diff --git a/Cargo.lock b/Cargo.lock
index 880ed45..e203255 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -1,6 +1,6 @@
 # This file is automatically @generated by Cargo.
 # It is not intended for manual editing.
-version = 3
+version = 4
 
 [[package]]
 name = "addr2line"
@@ -121,9 +121,15 @@ dependencies = [
 
 [[package]]
 name = "anyhow"
-version = "1.0.95"
+version = "1.0.97"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "34ac096ce696dc2fcabef30516bb13c0a68a11d30131d3df6f04711467681b04"
+checksum = "dcfed56ad506cb2c684a14971b8861fdc3baaaae314b9e5f9bb532cbe3ba7a4f"
+
+[[package]]
+name = "arrayvec"
+version = "0.7.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50"
 
 [[package]]
 name = "async-channel"
@@ -189,7 +195,7 @@ dependencies = [
  "futures-lite",
  "parking",
  "polling",
- "rustix",
+ "rustix 0.38.44",
  "slab",
  "tracing",
  "windows-sys 0.59.0",
@@ -208,9 +214,9 @@ dependencies = [
 
 [[package]]
 name = "async-std"
-version = "1.13.0"
+version = "1.13.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "c634475f29802fde2b8f0b505b1bd00dfe4df7d4a000f0b36f7671197d5c3615"
+checksum = "730294c1c08c2e0f85759590518f6333f0d5a0a766a27d519c1b244c3dfd8a24"
 dependencies = [
  "async-channel 1.9.0",
  "async-global-executor",
@@ -223,7 +229,7 @@ dependencies = [
  "futures-lite",
  "gloo-timers",
  "kv-log-macro",
- "log 0.4.25",
+ "log 0.4.27",
  "memchr",
  "once_cell",
  "pin-project-lite",
@@ -240,13 +246,13 @@ checksum = "8b75356056920673b02621b35afd0f7dda9306d03c79a30f5c56c44cf256e3de"
 
 [[package]]
 name = "async-trait"
-version = "0.1.85"
+version = "0.1.88"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "3f934833b4b7233644e5848f235df3f57ed8c80f1528a26c3dfa13d2147fa056"
+checksum = "e539d3fca749fcee5236ab05e93a52867dd549cc157c8cb7f99595f3cedffdb5"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -271,7 +277,7 @@ dependencies = [
  "base64ct",
  "clap",
  "cool_asserts",
- "serde 1.0.218",
+ "serde 1.0.219",
  "sha2",
  "sqlx",
  "thiserror 1.0.69",
@@ -305,10 +311,10 @@ dependencies = [
  "bytes",
  "form_urlencoded",
  "futures-util",
- "http 1.2.0",
+ "http 1.3.1",
  "http-body 1.0.1",
  "http-body-util",
- "hyper 1.5.2",
+ "hyper 1.6.0",
  "hyper-util",
  "itoa",
  "matchit",
@@ -317,7 +323,7 @@ dependencies = [
  "percent-encoding 2.3.1",
  "pin-project-lite",
  "rustversion",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "serde_path_to_error",
  "serde_urlencoded",
@@ -337,7 +343,7 @@ checksum = "df1362f362fd16024ae199c1970ce98f9661bf5ef94b9808fee734bc3698b733"
 dependencies = [
  "bytes",
  "futures-util",
- "http 1.2.0",
+ "http 1.3.1",
  "http-body 1.0.1",
  "http-body-util",
  "mime 0.3.17",
@@ -357,7 +363,7 @@ checksum = "604fde5e028fea851ce1d8570bbdc034bec850d157f7569d10f347d06808c05c"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -375,12 +381,21 @@ dependencies = [
  "addr2line",
  "cfg-if",
  "libc",
- "miniz_oxide 0.8.3",
+ "miniz_oxide 0.8.5",
  "object",
  "rustc-demangle",
  "windows-targets 0.52.6",
 ]
 
+[[package]]
+name = "bare-metal"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5deb64efa5bd81e31fcd1938615a6d98c82eafcbcd787162b6f63b91d6bac5b3"
+dependencies = [
+ "rustc_version 0.2.3",
+]
+
 [[package]]
 name = "base64"
 version = "0.9.3"
@@ -391,6 +406,12 @@ dependencies = [
  "safemem",
 ]
 
+[[package]]
+name = "base64"
+version = "0.13.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9e1b586273c5702936fe7b7d6896644d8be71e6314cfe09d3167c95f712589e8"
+
 [[package]]
 name = "base64"
 version = "0.21.7"
@@ -405,9 +426,9 @@ checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6"
 
 [[package]]
 name = "base64ct"
-version = "1.6.0"
+version = "1.7.3"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "8c3c1a368f70d6cf7302d78f8f7093da241fb8e8807c05cc9e51a125895a6d5b"
+checksum = "89e25b6adfb930f02d1981565a6e5d9c547ac15a96606256d3b59040e5cd4ca3"
 
 [[package]]
 name = "bincode"
@@ -415,7 +436,7 @@ version = "1.3.3"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "b1f45e9417d87227c7a56d22e471c6206462cba514c7590c09aff4cf6d1ddcad"
 dependencies = [
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
@@ -425,10 +446,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "271383c67ccabffb7381723dea0672a673f292304fcb45c01cc648c7a8d58088"
 dependencies = [
  "annotate-snippets",
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "cexpr",
  "clang-sys",
- "itertools",
+ "itertools 0.12.1",
  "lazy_static",
  "lazycell",
  "proc-macro2",
@@ -436,7 +457,7 @@ dependencies = [
  "regex",
  "rustc-hash",
  "shlex",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -460,6 +481,18 @@ version = "0.10.2"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "dc827186963e592360843fb5ba4b973e145841266c1357f7180c43526f2e5b61"
 
+[[package]]
+name = "bitfield"
+version = "0.13.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "46afbd2983a5d5a7bd740ccb198caf5b82f45c40c09c0eed36052d91cb92e719"
+
+[[package]]
+name = "bitfield"
+version = "0.14.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2d7e60934ceec538daadb9d8432424ed043a904d8e0243f3c6446bce549a46ac"
+
 [[package]]
 name = "bitflags"
 version = "1.3.2"
@@ -468,11 +501,11 @@ checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
 
 [[package]]
 name = "bitflags"
-version = "2.8.0"
+version = "2.9.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "8f68f53c83ab957f72c32642f3868eec03eb974d1fb82e453128456482613d36"
+checksum = "5c8214115b7bf84099f1309324e63141d4c5d7cc26862f97a0a857dbefe165bd"
 dependencies = [
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
@@ -499,21 +532,21 @@ dependencies = [
 
 [[package]]
 name = "build_html"
-version = "2.5.0"
+version = "2.6.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "225eb82ce9e70dcc0cfa6e404d0f353326b6e163bf500ec4711cec317d11935c"
+checksum = "01b01f54cbdd56298a506b086691594ded3b68dcbc9437adc87c616a35e7fc89"
 
 [[package]]
 name = "bumpalo"
-version = "3.16.0"
+version = "3.17.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c"
+checksum = "1628fb46dfa0b37568d12e5edd512553eccf6a22a78e8bde00bb4aed84d5bdbf"
 
 [[package]]
 name = "bytemuck"
-version = "1.21.0"
+version = "1.22.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ef657dfab802224e671f5818e9a4935f9b1957ed18e58292690cc39e7a4092a3"
+checksum = "b6b1fc10dbac614ebc03540c9dbd60e83887fda27794998c6528f1782047d540"
 
 [[package]]
 name = "byteorder"
@@ -523,9 +556,9 @@ checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b"
 
 [[package]]
 name = "bytes"
-version = "1.9.0"
+version = "1.10.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "325918d6fe32f23b19878fe4b34794ae41fc19ddbe53b10571a4874d44ffd39b"
+checksum = "d71b6127be86fdcfddb610f7182ac57211d4b18a3e9c82eb2d17662f2227ad6a"
 
 [[package]]
 name = "cairo-rs"
@@ -533,9 +566,9 @@ version = "0.18.5"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "8ca26ef0159422fb77631dc9d17b102f253b876fe1586b03b803e63a309b4ee2"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "cairo-sys-rs",
- "glib",
+ "glib 0.18.5",
  "libc",
  "once_cell",
  "thiserror 1.0.69",
@@ -547,16 +580,16 @@ version = "0.18.2"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "685c9fa8e590b8b3d678873528d83411db17242a73fccaed827770ea0fedda51"
 dependencies = [
- "glib-sys",
+ "glib-sys 0.18.1",
  "libc",
- "system-deps",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
 name = "cc"
-version = "1.2.10"
+version = "1.2.17"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "13208fcbb66eaeffe09b99fffbe1af420f00a7b35aa99ad683dfc1aa76145229"
+checksum = "1fcb57c740ae1daf453ae85f16e37396f672b039e00d9d866e07ddb24e328e3a"
 dependencies = [
  "shlex",
 ]
@@ -580,6 +613,16 @@ dependencies = [
  "target-lexicon",
 ]
 
+[[package]]
+name = "cfg-expr"
+version = "0.17.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8d4ba6e40bd1184518716a6e1a781bf9160e286d219ccdb8ab2612e74cfe4789"
+dependencies = [
+ "smallvec",
+ "target-lexicon",
+]
+
 [[package]]
 name = "cfg-if"
 version = "1.0.0"
@@ -590,22 +633,22 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
 name = "changeset"
 version = "0.1.0"
 dependencies = [
- "uuid 0.8.2",
+ "uuid 1.16.0",
 ]
 
 [[package]]
 name = "chrono"
-version = "0.4.39"
+version = "0.4.40"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "7e36cc9d416881d2e24f9a963be5fb1cd90966419ac844274161d10488b3e825"
+checksum = "1a7964611d71df112cb1730f2ee67324fcf4d0fc6606acbbe9bfe06df124637c"
 dependencies = [
  "android-tzdata",
  "iana-time-zone",
  "js-sys",
  "num-traits",
- "serde 1.0.218",
+ "serde 1.0.219",
  "wasm-bindgen",
- "windows-targets 0.52.6",
+ "windows-link",
 ]
 
 [[package]]
@@ -617,7 +660,7 @@ dependencies = [
  "chrono",
  "chrono-tz-build",
  "phf 0.11.3",
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
@@ -644,9 +687,9 @@ dependencies = [
 
 [[package]]
 name = "clap"
-version = "4.5.26"
+version = "4.5.34"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "a8eb5e908ef3a6efbe1ed62520fb7287959888c88485abe072543190ecc66783"
+checksum = "e958897981290da2a852763fe9cdb89cd36977a5d729023127095fa94d95e2ff"
 dependencies = [
  "clap_builder",
  "clap_derive",
@@ -654,9 +697,9 @@ dependencies = [
 
 [[package]]
 name = "clap_builder"
-version = "4.5.26"
+version = "4.5.34"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "96b01801b5fc6a0a232407abc821660c9c6d25a1cafc0d4f85f29fb8d9afc121"
+checksum = "83b0f35019843db2160b5bb19ae09b4e6411ac33fc6a712003c33e03090e2489"
 dependencies = [
  "anstream",
  "anstyle",
@@ -666,14 +709,14 @@ dependencies = [
 
 [[package]]
 name = "clap_derive"
-version = "4.5.24"
+version = "4.5.32"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "54b755194d6389280185988721fffba69495eed5ee9feeee9a599b53db80318c"
+checksum = "09176aae279615badda0765c0c0b3f6ed53f4709118af73cf4655d85d1530cd7"
 dependencies = [
  "heck 0.5.0",
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -718,7 +761,7 @@ version = "0.1.0"
 dependencies = [
  "config-derive",
  "cool_asserts",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "thiserror 1.0.69",
 ]
@@ -741,6 +784,12 @@ dependencies = [
  "wasm-bindgen",
 ]
 
+[[package]]
+name = "const-default"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0b396d1f76d455557e1218ec8066ae14bba60b4b36ecd55577ba979f5db7ecaa"
+
 [[package]]
 name = "const-oid"
 version = "0.9.6"
@@ -762,7 +811,7 @@ version = "0.17.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "7efb37c3e1ccb1ff97164ad95ac1606e8ccd35b3fa0a7d99a304c7f4a428cc24"
 dependencies = [
- "time 0.3.37",
+ "time 0.3.41",
  "version_check 0.9.5",
 ]
 
@@ -810,10 +859,42 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b"
 
 [[package]]
-name = "cpufeatures"
-version = "0.2.16"
+name = "cortex-m"
+version = "0.7.7"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "16b80225097f2e5ae4e7179dd2266824648f3e2f49d9134d584b76389d31c4c3"
+checksum = "8ec610d8f49840a5b376c69663b6369e71f4b34484b9b2eb29fb918d92516cb9"
+dependencies = [
+ "bare-metal",
+ "bitfield 0.13.2",
+ "embedded-hal 0.2.7",
+ "volatile-register",
+]
+
+[[package]]
+name = "cortex-m-rt"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "801d4dec46b34c299ccf6b036717ae0fce602faa4f4fe816d9013b9a7c9f5ba6"
+dependencies = [
+ "cortex-m-rt-macros",
+]
+
+[[package]]
+name = "cortex-m-rt-macros"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e37549a379a9e0e6e576fd208ee60394ccb8be963889eebba3ffe0980364f472"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.100",
+]
+
+[[package]]
+name = "cpufeatures"
+version = "0.2.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "59ed5838eebb26a2bb2e58f6d5b5316989ae9d08bab10e0e6d103e656d1b0280"
 dependencies = [
  "libc",
 ]
@@ -827,6 +908,15 @@ dependencies = [
  "crc-catalog",
 ]
 
+[[package]]
+name = "crc-any"
+version = "2.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a62ec9ff5f7965e4d7280bd5482acd20aadb50d632cf6c1d74493856b011fa73"
+dependencies = [
+ "debug-helper",
+]
+
 [[package]]
 name = "crc-catalog"
 version = "2.4.0"
@@ -842,6 +932,12 @@ dependencies = [
  "cfg-if",
 ]
 
+[[package]]
+name = "critical-section"
+version = "1.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "790eea4361631c5e7d22598ecd5723ff611904e3344ce8720784c93e3d83d40b"
+
 [[package]]
 name = "crossbeam-deque"
 version = "0.8.6"
@@ -878,9 +974,9 @@ checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28"
 
 [[package]]
 name = "crunchy"
-version = "0.2.2"
+version = "0.2.3"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7"
+checksum = "43da5946c66ffcc7745f48db692ffbb10a83bfe0afd96235c5c2a4fb23994929"
 
 [[package]]
 name = "crypto-common"
@@ -899,10 +995,10 @@ dependencies = [
  "async-std",
  "cairo-rs",
  "cyberpunk",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "gtk4",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_yml",
 ]
 
@@ -911,8 +1007,8 @@ name = "cyberpunk"
 version = "0.1.0"
 dependencies = [
  "cairo-rs",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "gtk4",
 ]
 
@@ -923,8 +1019,8 @@ dependencies = [
  "async-std",
  "cairo-rs",
  "cyberpunk",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "gtk4",
 ]
 
@@ -940,15 +1036,15 @@ dependencies = [
  "futures",
  "gdk4",
  "geo-types",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "glib-build-tools 0.18.0",
  "gtk4",
  "lazy_static",
  "libadwaita",
  "memorycache",
  "reqwest",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "tokio",
  "unic-langid",
@@ -956,9 +1052,15 @@ dependencies = [
 
 [[package]]
 name = "data-encoding"
-version = "2.7.0"
+version = "2.8.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "0e60eed09d8c01d3cee5b7d30acb059b76614c918fa0f992e0dd6eeb10daad6f"
+checksum = "575f75dfd25738df5b91b8e43e14d44bda14637a58fae779fd2b064f8bf3e010"
+
+[[package]]
+name = "debug-helper"
+version = "0.3.13"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f578e8e2c440e7297e008bb5486a3a8a194775224bbc23729b0dbdfaeebf162e"
 
 [[package]]
 name = "deflate"
@@ -983,9 +1085,9 @@ dependencies = [
 
 [[package]]
 name = "deranged"
-version = "0.3.11"
+version = "0.4.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "b42b6fa04a440b495c8b04d0e71b707c585f83cb9cb28cf8cd0d976c315e31b4"
+checksum = "28cfac68e08048ae1883171632c2aef3ebc555621ae56fbccce1cbf22dd7f058"
 dependencies = [
  "powerfmt",
 ]
@@ -1010,7 +1112,7 @@ checksum = "2517b0555262aeeda0d107a40ecfbbcf185921180ffb4acf316ebe0887467e26"
 dependencies = [
  "generic-array 0.11.2",
  "num-traits",
- "serde 1.0.218",
+ "serde 1.0.219",
  "typenum",
 ]
 
@@ -1022,7 +1124,7 @@ checksum = "a0b0a86c5d31c93238ff4b694fa31f3acdf67440770dc314c57d90e433914397"
 dependencies = [
  "generic-array 0.14.7",
  "num-traits",
- "serde 1.0.218",
+ "serde 1.0.219",
  "typenum",
 ]
 
@@ -1034,7 +1136,7 @@ checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -1045,13 +1147,75 @@ checksum = "1aaf95b3e5c8f23aa320147307562d361db0ae0d51242340f558153b4eb2439b"
 
 [[package]]
 name = "either"
-version = "1.13.0"
+version = "1.15.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "60b1af1c220855b6ceac025d3f6ecdd2b7c4894bfe9cd9bda4fbb4bc7c0d4cf0"
+checksum = "48c757948c5ede0e46177b7add2e67155f70e33c07fea8284df6576da70b3719"
 dependencies = [
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
+[[package]]
+name = "embedded-alloc"
+version = "0.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8f2de9133f68db0d4627ad69db767726c99ff8585272716708227008d3f1bddd"
+dependencies = [
+ "const-default",
+ "critical-section",
+ "linked_list_allocator",
+ "rlsf",
+]
+
+[[package]]
+name = "embedded-dma"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "994f7e5b5cb23521c22304927195f236813053eb9c065dd2226a32ba64695446"
+dependencies = [
+ "stable_deref_trait",
+]
+
+[[package]]
+name = "embedded-hal"
+version = "0.2.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "35949884794ad573cf46071e41c9b60efb0cb311e3ca01f7af807af1debc66ff"
+dependencies = [
+ "nb 0.1.3",
+ "void",
+]
+
+[[package]]
+name = "embedded-hal"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "361a90feb7004eca4019fb28352a9465666b24f840f5c3cddf0ff13920590b89"
+
+[[package]]
+name = "embedded-hal-async"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0c4c685bbef7fe13c3c6dd4da26841ed3980ef33e841cddfa15ce8a8fb3f1884"
+dependencies = [
+ "embedded-hal 1.0.0",
+]
+
+[[package]]
+name = "embedded-hal-nb"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fba4268c14288c828995299e59b12babdbe170f6c6d73731af1b4648142e8605"
+dependencies = [
+ "embedded-hal 1.0.0",
+ "nb 1.1.0",
+]
+
+[[package]]
+name = "embedded-io"
+version = "0.6.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "edd0f118536f44f5ccd48bcb8b111bdc3de888b58c74639dfb034a357d0f206d"
+
 [[package]]
 name = "emseries"
 version = "0.6.0"
@@ -1059,7 +1223,7 @@ dependencies = [
  "chrono",
  "chrono-tz",
  "dimensioned 0.7.0",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_derive",
  "serde_json",
  "tempfile",
@@ -1084,16 +1248,16 @@ checksum = "4cd405aab171cb85d6735e5c8d9db038c17d3ca007a4d2c25f337935c3d90580"
 dependencies = [
  "humantime",
  "is-terminal",
- "log 0.4.25",
+ "log 0.4.27",
  "regex",
  "termcolor",
 ]
 
 [[package]]
 name = "equivalent"
-version = "1.0.1"
+version = "1.0.2"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5"
+checksum = "877a4ace8713b0bcf2a4e7eec82529c029f1d0619886d18145fea96c3ffe5c0f"
 
 [[package]]
 name = "errno"
@@ -1135,9 +1299,9 @@ dependencies = [
 
 [[package]]
 name = "event-listener-strategy"
-version = "0.5.3"
+version = "0.5.4"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "3c3e4e0dd3673c1139bf041f3008816d9cf2946bbfac2945c09e523b8d7b05b2"
+checksum = "8be9f3dfaaffdae2972880079a491a1a8bb7cbed0b8dd7a347f668b4150a3b93"
 dependencies = [
  "event-listener 5.4.0",
  "pin-project-lite",
@@ -1152,7 +1316,7 @@ dependencies = [
  "bit_field",
  "half",
  "lebe",
- "miniz_oxide 0.8.3",
+ "miniz_oxide 0.8.5",
  "rayon-core",
  "smallvec",
  "zune-inflate",
@@ -1180,7 +1344,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "38e2275cc4e4fc009b0669731a1e5ab7ebf11f469eaede2bab9309a5b4d6057f"
 dependencies = [
  "memoffset",
- "rustc_version",
+ "rustc_version 0.4.1",
 ]
 
 [[package]]
@@ -1199,12 +1363,12 @@ dependencies = [
  "hex-string",
  "http 0.2.12",
  "image 0.23.14",
- "log 0.4.25",
+ "log 0.4.27",
  "logger",
  "mime 0.3.17",
  "mime_guess 2.0.5",
  "pretty_env_logger",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "sha2",
  "tempdir",
@@ -1226,8 +1390,8 @@ dependencies = [
  "emseries",
  "ft-core",
  "gdk4",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "glib-build-tools 0.18.0",
  "gtk4",
  "libadwaita",
@@ -1237,9 +1401,9 @@ dependencies = [
 
 [[package]]
 name = "fixed"
-version = "1.28.0"
+version = "1.29.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "85c6e0b89bf864acd20590dbdbad56f69aeb898abfc9443008fd7bd48b2cc85a"
+checksum = "707070ccf8c4173548210893a0186e29c266901b71ed20cd9e2ca0193dfe95c3"
 dependencies = [
  "az",
  "bytemuck",
@@ -1249,12 +1413,12 @@ dependencies = [
 
 [[package]]
 name = "flate2"
-version = "1.0.35"
+version = "1.1.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "c936bfdafb507ebbf50b8074c54fa31c5be9a1e7e5f467dd659697041407d07c"
+checksum = "11faaf5a5236997af9848be0bef4db95824b1d534ebc64d0f0c6cf3e67bd38dc"
 dependencies = [
  "crc32fast",
- "miniz_oxide 0.8.3",
+ "miniz_oxide 0.8.5",
 ]
 
 [[package]]
@@ -1330,9 +1494,9 @@ checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1"
 
 [[package]]
 name = "foldhash"
-version = "0.1.4"
+version = "0.1.5"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "a0d2fde1f7b3d48b8395d5f2de76c18a528bd6a9cdde438df747bfcba3e05d6f"
+checksum = "d9c4f5dac5e15c24eb999c26181a6ca40b39fe946cbe4c263c7209467bc83af2"
 
 [[package]]
 name = "foreign-types"
@@ -1358,6 +1522,45 @@ dependencies = [
  "percent-encoding 2.3.1",
 ]
 
+[[package]]
+name = "frunk"
+version = "0.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "874b6a17738fc273ec753618bac60ddaeac48cb1d7684c3e7bd472e57a28b817"
+dependencies = [
+ "frunk_core",
+ "frunk_derives",
+]
+
+[[package]]
+name = "frunk_core"
+version = "0.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3529a07095650187788833d585c219761114005d5976185760cf794d265b6a5c"
+
+[[package]]
+name = "frunk_derives"
+version = "0.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e99b8b3c28ae0e84b604c75f721c21dc77afb3706076af5e8216d15fd1deaae3"
+dependencies = [
+ "frunk_proc_macro_helpers",
+ "quote",
+ "syn 2.0.100",
+]
+
+[[package]]
+name = "frunk_proc_macro_helpers"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "05a956ef36c377977e512e227dcad20f68c2786ac7a54dacece3746046fea5ce"
+dependencies = [
+ "frunk_core",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.100",
+]
+
 [[package]]
 name = "ft-core"
 version = "0.1.0"
@@ -1366,7 +1569,7 @@ dependencies = [
  "chrono-tz",
  "dimensioned 0.8.0",
  "emseries",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "tempfile",
 ]
@@ -1377,6 +1580,15 @@ version = "0.1.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "a06f77d526c1a601b7c4cdd98f54b5eaabffc14d5f2f0296febdc7f357c6d3ba"
 
+[[package]]
+name = "fugit"
+version = "0.3.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "17186ad64927d5ac8f02c1e77ccefa08ccd9eaa314d5a4772278aa204a22f7e7"
+dependencies = [
+ "gcd",
+]
+
 [[package]]
 name = "futures"
 version = "0.3.31"
@@ -1457,7 +1669,7 @@ checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -1490,6 +1702,12 @@ dependencies = [
  "slab",
 ]
 
+[[package]]
+name = "gcd"
+version = "2.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1d758ba1b47b00caf47f24925c0074ecb20d6dfcffe7f6d53395c0465674841a"
+
 [[package]]
 name = "gdk-pixbuf"
 version = "0.18.5"
@@ -1497,8 +1715,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "50e1f5f1b0bfb830d6ccc8066d18db35c487b1b2b1e8589b5dfe9f07e8defaec"
 dependencies = [
  "gdk-pixbuf-sys",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "libc",
  "once_cell",
 ]
@@ -1509,11 +1727,11 @@ version = "0.18.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "3f9839ea644ed9c97a34d129ad56d38a25e6756f99f3a88e15cd39c20629caf7"
 dependencies = [
- "gio-sys",
- "glib-sys",
- "gobject-sys",
+ "gio-sys 0.18.1",
+ "glib-sys 0.18.1",
+ "gobject-sys 0.18.0",
  "libc",
- "system-deps",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
@@ -1525,10 +1743,10 @@ dependencies = [
  "cairo-rs",
  "gdk-pixbuf",
  "gdk4-sys",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "libc",
- "pango",
+ "pango 0.18.3",
 ]
 
 [[package]]
@@ -1539,13 +1757,13 @@ checksum = "dbab43f332a3cf1df9974da690b5bb0e26720ed09a228178ce52175372dcfef0"
 dependencies = [
  "cairo-sys-rs",
  "gdk-pixbuf-sys",
- "gio-sys",
- "glib-sys",
- "gobject-sys",
+ "gio-sys 0.18.1",
+ "glib-sys 0.18.1",
+ "gobject-sys 0.18.0",
  "libc",
- "pango-sys",
+ "pango-sys 0.18.0",
  "pkg-config",
- "system-deps",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
@@ -1586,14 +1804,14 @@ dependencies = [
 
 [[package]]
 name = "getrandom"
-version = "0.3.1"
+version = "0.3.2"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "43a49c392881ce6d5c3b8cb70f98717b7c07aabbdff06687b9030dbfbe2725f8"
+checksum = "73fea8450eea4bac3940448fb7ae50d91f034f941199fcd9d909a5a07aa455f0"
 dependencies = [
  "cfg-if",
  "libc",
- "wasi 0.13.3+wasi-0.2.2",
- "windows-targets 0.52.6",
+ "r-efi",
+ "wasi 0.14.2+wasi-0.2.4",
 ]
 
 [[package]]
@@ -1632,8 +1850,8 @@ dependencies = [
  "futures-core",
  "futures-io",
  "futures-util",
- "gio-sys",
- "glib",
+ "gio-sys 0.18.1",
+ "glib 0.18.5",
  "libc",
  "once_cell",
  "pin-project-lite",
@@ -1641,35 +1859,65 @@ dependencies = [
  "thiserror 1.0.69",
 ]
 
+[[package]]
+name = "gio"
+version = "0.20.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a4f00c70f8029d84ea7572dd0e1aaa79e5329667b4c17f329d79ffb1e6277487"
+dependencies = [
+ "futures-channel",
+ "futures-core",
+ "futures-io",
+ "futures-util",
+ "gio-sys 0.20.9",
+ "glib 0.20.9",
+ "libc",
+ "pin-project-lite",
+ "smallvec",
+]
+
 [[package]]
 name = "gio-sys"
 version = "0.18.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "37566df850baf5e4cb0dfb78af2e4b9898d817ed9263d1090a2df958c64737d2"
 dependencies = [
- "glib-sys",
- "gobject-sys",
+ "glib-sys 0.18.1",
+ "gobject-sys 0.18.0",
  "libc",
- "system-deps",
+ "system-deps 6.2.2",
  "winapi",
 ]
 
+[[package]]
+name = "gio-sys"
+version = "0.20.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "160eb5250a26998c3e1b54e6a3d4ea15c6c7762a6062a19a7b63eff6e2b33f9e"
+dependencies = [
+ "glib-sys 0.20.9",
+ "gobject-sys 0.20.9",
+ "libc",
+ "system-deps 7.0.3",
+ "windows-sys 0.59.0",
+]
+
 [[package]]
 name = "glib"
 version = "0.18.5"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "233daaf6e83ae6a12a52055f568f9d7cf4671dabb78ff9560ab6da230ce00ee5"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "futures-channel",
  "futures-core",
  "futures-executor",
  "futures-task",
  "futures-util",
- "gio-sys",
- "glib-macros",
- "glib-sys",
- "gobject-sys",
+ "gio-sys 0.18.1",
+ "glib-macros 0.18.5",
+ "glib-sys 0.18.1",
+ "gobject-sys 0.18.0",
  "libc",
  "memchr",
  "once_cell",
@@ -1677,6 +1925,27 @@ dependencies = [
  "thiserror 1.0.69",
 ]
 
+[[package]]
+name = "glib"
+version = "0.20.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "707b819af8059ee5395a2de9f2317d87a53dbad8846a2f089f0bb44703f37686"
+dependencies = [
+ "bitflags 2.9.0",
+ "futures-channel",
+ "futures-core",
+ "futures-executor",
+ "futures-task",
+ "futures-util",
+ "gio-sys 0.20.9",
+ "glib-macros 0.20.7",
+ "glib-sys 0.20.9",
+ "gobject-sys 0.20.9",
+ "libc",
+ "memchr",
+ "smallvec",
+]
+
 [[package]]
 name = "glib-build-tools"
 version = "0.16.3"
@@ -1702,11 +1971,24 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "0bb0228f477c0900c880fd78c8759b95c7636dbd7842707f49e132378aa2acdc"
 dependencies = [
  "heck 0.4.1",
- "proc-macro-crate 2.0.2",
+ "proc-macro-crate 2.0.0",
  "proc-macro-error",
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
+]
+
+[[package]]
+name = "glib-macros"
+version = "0.20.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "715601f8f02e71baef9c1f94a657a9a77c192aea6097cf9ae7e5e177cd8cde68"
+dependencies = [
+ "heck 0.5.0",
+ "proc-macro-crate 3.3.0",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -1716,7 +1998,17 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "063ce2eb6a8d0ea93d2bf8ba1957e78dbab6be1c2220dd3daca57d5a9d869898"
 dependencies = [
  "libc",
- "system-deps",
+ "system-deps 6.2.2",
+]
+
+[[package]]
+name = "glib-sys"
+version = "0.20.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a8928869a44cfdd1fccb17d6746e4ff82c8f82e41ce705aa026a52ca8dc3aefb"
+dependencies = [
+ "libc",
+ "system-deps 7.0.3",
 ]
 
 [[package]]
@@ -1752,7 +2044,7 @@ checksum = "2a17868f56b4a24f677b17c8cb69958385102fa879418052d60b50bc1727e261"
 dependencies = [
  "gloo-utils",
  "js-sys",
- "serde 1.0.218",
+ "serde 1.0.219",
  "wasm-bindgen",
  "web-sys",
 ]
@@ -1798,7 +2090,7 @@ dependencies = [
  "getrandom 0.2.15",
  "gloo-events",
  "gloo-utils",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde-wasm-bindgen",
  "serde_urlencoded",
  "thiserror 1.0.69",
@@ -1819,7 +2111,7 @@ dependencies = [
  "http 0.2.12",
  "js-sys",
  "pin-project",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "thiserror 1.0.69",
  "wasm-bindgen",
@@ -1837,10 +2129,10 @@ dependencies = [
  "futures-core",
  "futures-sink",
  "gloo-utils",
- "http 1.2.0",
+ "http 1.3.1",
  "js-sys",
  "pin-project",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "thiserror 1.0.69",
  "wasm-bindgen",
@@ -1866,7 +2158,7 @@ checksum = "fbc8031e8c92758af912f9bc08fbbadd3c6f3cfcbf6b64cdf3d6a81f0139277a"
 dependencies = [
  "gloo-utils",
  "js-sys",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "thiserror 1.0.69",
  "wasm-bindgen",
@@ -1892,7 +2184,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "0b5555354113b18c547c1d3a98fbf7fb32a9ff4f6fa112ce823a21641a0ba3aa"
 dependencies = [
  "js-sys",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "wasm-bindgen",
  "web-sys",
@@ -1910,7 +2202,7 @@ dependencies = [
  "gloo-worker-macros",
  "js-sys",
  "pinned",
- "serde 1.0.218",
+ "serde 1.0.219",
  "thiserror 1.0.69",
  "wasm-bindgen",
  "wasm-bindgen-futures",
@@ -1926,7 +2218,7 @@ dependencies = [
  "proc-macro-crate 1.3.1",
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -1937,12 +2229,12 @@ dependencies = [
  "config-derive",
  "futures",
  "gdk4",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "glib-build-tools 0.16.3",
  "gtk4",
  "libadwaita",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "tokio",
 ]
@@ -1952,7 +2244,7 @@ name = "gm-dash"
 version = "0.1.0"
 dependencies = [
  "pipewire",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "tokio",
  "warp",
@@ -1964,9 +2256,20 @@ version = "0.18.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "0850127b514d1c4a4654ead6dedadb18198999985908e6ffe4436f53c785ce44"
 dependencies = [
- "glib-sys",
+ "glib-sys 0.18.1",
  "libc",
- "system-deps",
+ "system-deps 6.2.2",
+]
+
+[[package]]
+name = "gobject-sys"
+version = "0.20.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c773a3cb38a419ad9c26c81d177d96b4b08980e8bdbbf32dace883e96e96e7e3"
+dependencies = [
+ "glib-sys 0.20.9",
+ "libc",
+ "system-deps 7.0.3",
 ]
 
 [[package]]
@@ -1975,7 +2278,7 @@ version = "0.18.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "3b2228cda1505613a7a956cca69076892cfbda84fc2b7a62b94a41a272c0c401"
 dependencies = [
- "glib",
+ "glib 0.18.5",
  "graphene-sys",
  "libc",
 ]
@@ -1986,10 +2289,10 @@ version = "0.18.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "cc4144cee8fc8788f2a9b73dc5f1d4e1189d1f95305c4cb7bd9c1af1cfa31f59"
 dependencies = [
- "glib-sys",
+ "glib-sys 0.18.1",
  "libc",
  "pkg-config",
- "system-deps",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
@@ -2009,11 +2312,11 @@ checksum = "0d958e351d2f210309b32d081c832d7de0aca0b077aa10d88336c6379bd01f7e"
 dependencies = [
  "cairo-rs",
  "gdk4",
- "glib",
+ "glib 0.18.5",
  "graphene-rs",
  "gsk4-sys",
  "libc",
- "pango",
+ "pango 0.18.3",
 ]
 
 [[package]]
@@ -2024,12 +2327,12 @@ checksum = "12bd9e3effea989f020e8f1ff3fa3b8c63ba93d43b899c11a118868853a56d55"
 dependencies = [
  "cairo-sys-rs",
  "gdk4-sys",
- "glib-sys",
- "gobject-sys",
+ "glib-sys 0.18.1",
+ "gobject-sys 0.18.0",
  "graphene-sys",
  "libc",
- "pango-sys",
- "system-deps",
+ "pango-sys 0.18.0",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
@@ -2043,14 +2346,14 @@ dependencies = [
  "futures-channel",
  "gdk-pixbuf",
  "gdk4",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "graphene-rs",
  "gsk4",
  "gtk4-macros",
  "gtk4-sys",
  "libc",
- "pango",
+ "pango 0.18.3",
 ]
 
 [[package]]
@@ -2076,14 +2379,14 @@ dependencies = [
  "cairo-sys-rs",
  "gdk-pixbuf-sys",
  "gdk4-sys",
- "gio-sys",
- "glib-sys",
- "gobject-sys",
+ "gio-sys 0.18.1",
+ "glib-sys 0.18.1",
+ "gobject-sys 0.18.0",
  "graphene-sys",
  "gsk4-sys",
  "libc",
- "pango-sys",
- "system-deps",
+ "pango-sys 0.18.0",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
@@ -2107,14 +2410,23 @@ dependencies = [
 
 [[package]]
 name = "half"
-version = "2.4.1"
+version = "2.5.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "6dd08c532ae367adf81c312a4580bc67f1d0fe8bc9c460520283f4c0ff277888"
+checksum = "7db2ff139bba50379da6aa0766b52fdcb62cb5b263009b09ed58ba604e14bbd1"
 dependencies = [
  "cfg-if",
  "crunchy",
 ]
 
+[[package]]
+name = "hash32"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "47d60b12902ba28e2730cd37e95b8c9223af2808df9e902d4df49588d1470606"
+dependencies = [
+ "byteorder",
+]
+
 [[package]]
 name = "hashbrown"
 version = "0.15.2"
@@ -2159,6 +2471,16 @@ dependencies = [
  "http 0.2.12",
 ]
 
+[[package]]
+name = "heapless"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0bfb9eb618601c89945a70e254898da93b13be0388091d42117462b265bb3fad"
+dependencies = [
+ "hash32",
+ "stable_deref_trait",
+]
+
 [[package]]
 name = "heck"
 version = "0.4.1"
@@ -2183,6 +2505,12 @@ version = "0.4.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "fbf6a919d6cf397374f7dfeeea91d974c7c0a7221d0d0f4f20d859d329e53fcc"
 
+[[package]]
+name = "hermit-abi"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fbd780fe5cc30f81464441920d82ac8740e2e46b29a6fad543ddd075229ce37e"
+
 [[package]]
 name = "hex"
 version = "0.4.3"
@@ -2195,8 +2523,8 @@ version = "0.1.0"
 dependencies = [
  "cairo-rs",
  "coordinates",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "glib-build-tools 0.18.0",
  "gtk4",
  "image 0.24.9",
@@ -2248,9 +2576,9 @@ dependencies = [
 
 [[package]]
 name = "http"
-version = "1.2.0"
+version = "1.3.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f16ca2af56261c99fba8bac40a10251ce8188205a4c448fbb745a2e4daa76fea"
+checksum = "f4a85d31aea989eead29a3aaf9e1115a180df8282431156e533de47660892565"
 dependencies = [
  "bytes",
  "fnv",
@@ -2275,27 +2603,27 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184"
 dependencies = [
  "bytes",
- "http 1.2.0",
+ "http 1.3.1",
 ]
 
 [[package]]
 name = "http-body-util"
-version = "0.1.2"
+version = "0.1.3"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f"
+checksum = "b021d93e26becf5dc7e1b75b1bed1fd93124b374ceb73f43d4d4eafec896a64a"
 dependencies = [
  "bytes",
- "futures-util",
- "http 1.2.0",
+ "futures-core",
+ "http 1.3.1",
  "http-body 1.0.1",
  "pin-project-lite",
 ]
 
 [[package]]
 name = "httparse"
-version = "1.9.5"
+version = "1.10.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "7d71d3574edd2771538b901e6549113b4006ece66150fb69c0fb6d9a2adae946"
+checksum = "6dbf3de79e51f3d586ab4cb9d5c3e2c14aa28ed23d180cf89b4df0454a69cc87"
 
 [[package]]
 name = "httpdate"
@@ -2305,9 +2633,9 @@ checksum = "df3b46402a9d5adb4c86a0cf463f42e19994e3ee891101b1841f30a545cb49a9"
 
 [[package]]
 name = "humantime"
-version = "2.1.0"
+version = "2.2.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4"
+checksum = "9b112acc8b3adf4b107a8ec20977da0273a8c386765a3ec0229bd500a1443f9f"
 
 [[package]]
 name = "hyper"
@@ -2354,14 +2682,14 @@ dependencies = [
 
 [[package]]
 name = "hyper"
-version = "1.5.2"
+version = "1.6.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "256fb8d4bd6413123cc9d91832d78325c48ff41677595be797d90f42969beae0"
+checksum = "cc2b571658e38e0c01b1fdca3bbbe93c00d3d71693ff2770043f8c29bc7d6f80"
 dependencies = [
  "bytes",
  "futures-channel",
  "futures-util",
- "http 1.2.0",
+ "http 1.3.1",
  "http-body 1.0.1",
  "httparse",
  "httpdate",
@@ -2392,9 +2720,9 @@ checksum = "df2dcfbe0677734ab2f3ffa7fa7bfd4706bfdc1ef393f2ee30184aed67e631b4"
 dependencies = [
  "bytes",
  "futures-util",
- "http 1.2.0",
+ "http 1.3.1",
  "http-body 1.0.1",
- "hyper 1.5.2",
+ "hyper 1.6.0",
  "pin-project-lite",
  "tokio",
  "tower-service",
@@ -2402,14 +2730,15 @@ dependencies = [
 
 [[package]]
 name = "iana-time-zone"
-version = "0.1.61"
+version = "0.1.62"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "235e081f3925a06703c2d0117ea8b91f042756fd6e7a6e5d901e8ca1a996b220"
+checksum = "b2fd658b06e56721792c5df4475705b6cda790e9298d19d2f8af083457bcd127"
 dependencies = [
  "android_system_properties",
  "core-foundation-sys",
  "iana-time-zone-haiku",
  "js-sys",
+ "log 0.4.27",
  "wasm-bindgen",
  "windows-core",
 ]
@@ -2427,8 +2756,8 @@ dependencies = [
 name = "icon-test"
 version = "0.1.0"
 dependencies = [
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "gtk4",
  "libadwaita",
 ]
@@ -2453,6 +2782,7 @@ checksum = "13acbb8371917fc971be86fc8057c41a64b521c184808a698c02acc242dbf637"
 dependencies = [
  "displaydoc",
  "litemap",
+ "serde 1.0.219",
  "tinystr",
  "writeable",
  "zerovec",
@@ -2474,9 +2804,9 @@ dependencies = [
 
 [[package]]
 name = "icu_locid_transform_data"
-version = "1.5.0"
+version = "1.5.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "fdc8ff3388f852bede6b579ad4e978ab004f139284d7b28715f773507b946f6e"
+checksum = "7515e6d781098bf9f7205ab3fc7e9709d34554ae0b21ddbcb5febfa4bc7df11d"
 
 [[package]]
 name = "icu_normalizer"
@@ -2498,9 +2828,9 @@ dependencies = [
 
 [[package]]
 name = "icu_normalizer_data"
-version = "1.5.0"
+version = "1.5.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f8cafbf7aa791e9b22bec55a167906f9e1215fd475cd22adfcf660e03e989516"
+checksum = "c5e8338228bdc8ab83303f16b797e177953730f601a96c25d10cb3ab0daa0cb7"
 
 [[package]]
 name = "icu_properties"
@@ -2519,9 +2849,9 @@ dependencies = [
 
 [[package]]
 name = "icu_properties_data"
-version = "1.5.0"
+version = "1.5.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "67a8effbc3dd3e4ba1afa8ad918d5684b8868b3b26500753effea8d2eed19569"
+checksum = "85fb8799753b75aee8d2a21d7c14d9f38921b54b3dbda10f5a3c7a7b82dba5e2"
 
 [[package]]
 name = "icu_provider"
@@ -2548,7 +2878,7 @@ checksum = "1ec89e9337638ecdc08744df490b221a7399bf8d164eb52a665454e60e075ad6"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -2622,9 +2952,9 @@ dependencies = [
 
 [[package]]
 name = "implicit-clone"
-version = "0.5.0"
+version = "0.5.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "1bd41bf647018e1da0e32dac34d02135d61d7204cee650e4633eddbd0b23ec38"
+checksum = "88e4c5423e5b38b6f175b42561a08f6ddbc8ed3a3275f48eb350ba2dfe9a6b60"
 dependencies = [
  "implicit-clone-derive",
  "indexmap",
@@ -2632,12 +2962,12 @@ dependencies = [
 
 [[package]]
 name = "implicit-clone-derive"
-version = "0.1.1"
+version = "0.1.2"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "9311685eb9a34808bbb0608ad2fcab9ae216266beca5848613e95553ac914e3b"
+checksum = "699c1b6d335e63d0ba5c1e1c7f647371ce989c3bcbe1f7ed2b85fa56e3bd1a21"
 dependencies = [
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -2648,9 +2978,9 @@ checksum = "0cfe9645a18782869361d9c8732246be7b410ad4e919d3609ebabdac00ba12c3"
 
 [[package]]
 name = "indexmap"
-version = "2.7.0"
+version = "2.8.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "62f822373a4fe84d4bb149bf54e584a7f4abec90e072ed49cda0edea5b95471f"
+checksum = "3954d50fe15b02142bf25d3b8bdadb634ec3948f103d04ffe3031bc8fe9d7058"
 dependencies = [
  "equivalent",
  "hashbrown",
@@ -2677,9 +3007,9 @@ dependencies = [
 
 [[package]]
 name = "ipnet"
-version = "2.10.1"
+version = "2.11.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ddc24109865250148c2e0f3d25d4f0f479571723792d3802153c60922a4fb708"
+checksum = "469fb0b9cefa57e3ef31275ee7cacb78f2fdca44e4765491884a2b119d4eb130"
 
 [[package]]
 name = "iron"
@@ -2699,13 +3029,13 @@ dependencies = [
 
 [[package]]
 name = "is-terminal"
-version = "0.4.13"
+version = "0.4.16"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "261f68e344040fbd0edea105bef17c66edf46f984ddb1115b775ce31be948f4b"
+checksum = "e04d7f318608d35d4b61ddd75cbdaee86b023ebe2bd5a66ee0915f0bf93095a9"
 dependencies = [
- "hermit-abi 0.4.0",
+ "hermit-abi 0.5.0",
  "libc",
- "windows-sys 0.52.0",
+ "windows-sys 0.59.0",
 ]
 
 [[package]]
@@ -2714,6 +3044,15 @@ version = "1.70.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "7943c866cc5cd64cbc25b2e01621d07fa8eb2a1a23160ee81ce38704e97b8ecf"
 
+[[package]]
+name = "itertools"
+version = "0.10.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b0fd2260e829bddf4cb6ea802289de2f86d6a7a690192fbe91b3f46e0f2c8473"
+dependencies = [
+ "either",
+]
+
 [[package]]
 name = "itertools"
 version = "0.12.1"
@@ -2725,9 +3064,9 @@ dependencies = [
 
 [[package]]
 name = "itoa"
-version = "1.0.14"
+version = "1.0.15"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "d75a2a4b1b190afb6f5425f10f6a8f959d2ea0b9c2b1d79553551850539e4674"
+checksum = "4a5f13b858c8d314ee3e8f639011f7ccefe71f97f96e50151fb991f267928e2c"
 
 [[package]]
 name = "jpeg-decoder"
@@ -2763,7 +3102,22 @@ version = "1.0.7"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "0de8b303297635ad57c9f5059fd9cee7a47f8e8daa09df0fcd07dd39fb22977f"
 dependencies = [
- "log 0.4.25",
+ "log 0.4.27",
+]
+
+[[package]]
+name = "l10n-db"
+version = "0.1.0"
+dependencies = [
+ "chrono",
+ "clap",
+ "icu_locid",
+ "serde 1.0.219",
+ "serde_json",
+ "tempfile",
+ "thiserror 2.0.12",
+ "toml",
+ "xml-rs",
 ]
 
 [[package]]
@@ -2801,12 +3155,12 @@ checksum = "2fe7e70c06507ed10a16cda707f358fbe60fe0dc237498f78c686ade92fd979c"
 dependencies = [
  "gdk-pixbuf",
  "gdk4",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "gtk4",
  "libadwaita-sys",
  "libc",
- "pango",
+ "pango 0.18.3",
 ]
 
 [[package]]
@@ -2816,20 +3170,20 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "5e10aaa38de1d53374f90deeb4535209adc40cc5dba37f9704724169bceec69a"
 dependencies = [
  "gdk4-sys",
- "gio-sys",
- "glib-sys",
- "gobject-sys",
+ "gio-sys 0.18.1",
+ "glib-sys 0.18.1",
+ "gobject-sys 0.18.0",
  "gtk4-sys",
  "libc",
- "pango-sys",
- "system-deps",
+ "pango-sys 0.18.0",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
 name = "libc"
-version = "0.2.169"
+version = "0.2.171"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "b5aba8db14291edd000dfcc4d620c7ebfb122c613afb886ca8803fa4e128a20a"
+checksum = "c19937216e9d3aa9956d9bb8dfc0b0c8beb6058fc4f7a4dc4d850edf86a237d6"
 
 [[package]]
 name = "libloading"
@@ -2853,7 +3207,7 @@ version = "0.8.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "65f3a4b81b2a2d8c7f300643676202debd1b7c929dbf5c9bb89402ea11d19810"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "cc",
  "convert_case",
  "cookie-factory",
@@ -2861,7 +3215,7 @@ dependencies = [
  "libspa-sys",
  "nix",
  "nom",
- "system-deps",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
@@ -2872,7 +3226,7 @@ checksum = "bf0d9716420364790e85cbb9d3ac2c950bde16a7dd36f3209b7dfdfc4a24d01f"
 dependencies = [
  "bindgen",
  "cc",
- "system-deps",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
@@ -2904,6 +3258,12 @@ dependencies = [
  "fixed",
 ]
 
+[[package]]
+name = "linked_list_allocator"
+version = "0.10.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9afa463f5405ee81cdb9cc2baf37e08ec7e4c8209442b5d72c04cfb2cd6e6286"
+
 [[package]]
 name = "linux-raw-sys"
 version = "0.4.15"
@@ -2911,10 +3271,16 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "d26c52dbd32dccf2d10cac7725f8eae5296885fb5703b261f7d0a0739ec807ab"
 
 [[package]]
-name = "litemap"
-version = "0.7.4"
+name = "linux-raw-sys"
+version = "0.9.3"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "4ee93343901ab17bd981295f2cf0026d4ad018c7c31ba84549a4ddbb47a45104"
+checksum = "fe7db12097d22ec582439daf8618b8fdd1a7bef6270e9af3b1ebcd30893cf413"
+
+[[package]]
+name = "litemap"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "23fb14cb19457329c82206317a5663005a4d404783dc74f4252769b0d5f42856"
 
 [[package]]
 name = "lock_api"
@@ -2932,14 +3298,14 @@ version = "0.3.9"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "e19e8d5c34a3e0e2223db8e060f9e8264aeeb5c5fc64a4ee9965c062211c024b"
 dependencies = [
- "log 0.4.25",
+ "log 0.4.27",
 ]
 
 [[package]]
 name = "log"
-version = "0.4.25"
+version = "0.4.27"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "04cbf5b083de1c7e0222a7a51dbfdba1cbe1c6ab0b15e29fff3f6c077fd9cd9f"
+checksum = "13dc2df351e3202783a1fe0d44375f7295ffb4049267b0f3018346dc122a1d94"
 dependencies = [
  "value-bag",
 ]
@@ -2998,7 +3364,7 @@ version = "0.1.0"
 dependencies = [
  "chrono",
  "futures",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_derive",
  "tokio",
 ]
@@ -3067,9 +3433,9 @@ dependencies = [
 
 [[package]]
 name = "miniz_oxide"
-version = "0.8.3"
+version = "0.8.5"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "b8402cab7aefae129c6977bb0ff1b8fd9a04eb5b51efc50a70bea51cda0c7924"
+checksum = "8e3e04debbb59698c15bacbb6d93584a8c0ca9cc3213cb423d31f760d8843ce5"
 dependencies = [
  "adler2",
  "simd-adler32",
@@ -3103,7 +3469,7 @@ dependencies = [
  "futures-util",
  "http 0.2.12",
  "httparse",
- "log 0.4.25",
+ "log 0.4.27",
  "memchr",
  "mime 0.3.17",
  "spin",
@@ -3122,12 +3488,12 @@ dependencies = [
 
 [[package]]
 name = "native-tls"
-version = "0.2.12"
+version = "0.2.14"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "a8614eb2c83d59d1c8cc974dd3f920198647674a0a035e1af1fa58707e317466"
+checksum = "87de3442987e9dbec73158d5c715e7ad9072fda936bb03d19d7fa10e00520f0e"
 dependencies = [
  "libc",
- "log 0.4.25",
+ "log 0.4.27",
  "openssl",
  "openssl-probe",
  "openssl-sys",
@@ -3137,13 +3503,28 @@ dependencies = [
  "tempfile",
 ]
 
+[[package]]
+name = "nb"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "801d31da0513b6ec5214e9bf433a77966320625a37860f910be265be6e18d06f"
+dependencies = [
+ "nb 1.1.0",
+]
+
+[[package]]
+name = "nb"
+version = "1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8d5439c4ad607c3c23abf66de8c8bf57ba8adcd1f129e699851a6e43935d339d"
+
 [[package]]
 name = "nix"
 version = "0.27.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "2eb04e9c688eff1c89d72b407f168cf79bb9e867a9d3323ed6c01519eb9cc053"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "cfg-if",
  "libc",
 ]
@@ -3246,6 +3627,26 @@ dependencies = [
  "libc",
 ]
 
+[[package]]
+name = "num_enum"
+version = "0.5.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1f646caf906c20226733ed5b1374287eb97e3c2a5c227ce668c1f2ce20ae57c9"
+dependencies = [
+ "num_enum_derive",
+]
+
+[[package]]
+name = "num_enum_derive"
+version = "0.5.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dcbff9bc912032c62bf65ef1d5aea88983b420f4f839db1e9b0c281a25c9c799"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
 [[package]]
 name = "object"
 version = "0.36.7"
@@ -3257,17 +3658,17 @@ dependencies = [
 
 [[package]]
 name = "once_cell"
-version = "1.20.2"
+version = "1.21.2"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "1261fe7e33c73b354eab43b1273a57c8f967d0391e80353e51f764ac02cf6775"
+checksum = "c2806eaa3524762875e21c3dcd057bc4b7bfa01ce4da8d46be1cd43649e1cc6b"
 
 [[package]]
 name = "openssl"
-version = "0.10.68"
+version = "0.10.71"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "6174bc48f102d208783c2c84bf931bb75927a617866870de8a4ea85597f871f5"
+checksum = "5e14130c6a98cd258fdcb0fb6d744152343ff729cbfcb28c656a9d12b999fbcd"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "cfg-if",
  "foreign-types",
  "libc",
@@ -3284,20 +3685,20 @@ checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
 name = "openssl-probe"
-version = "0.1.5"
+version = "0.1.6"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf"
+checksum = "d05e27ee213611ffe7d6348b942e8f942b37114c00cc03cec254295a4a17852e"
 
 [[package]]
 name = "openssl-sys"
-version = "0.9.104"
+version = "0.9.106"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "45abf306cbf99debc8195b66b7346498d7b10c210de50418b5ccd7ceba08c741"
+checksum = "8bb61ea9811cc39e3c2069f40b8b8e2e70d8569b361f879786cc7ed48b777cdd"
 dependencies = [
  "cc",
  "libc",
@@ -3316,7 +3717,7 @@ dependencies = [
  "cool_asserts",
  "grid",
  "nary_tree",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "sgf",
  "thiserror 1.0.69",
@@ -3330,14 +3731,14 @@ dependencies = [
  "async-channel 2.3.1",
  "async-std",
  "cairo-rs",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "glib-build-tools 0.17.10",
  "gtk4",
  "image 0.24.9",
  "libadwaita",
  "otg-core",
- "pango",
+ "pango 0.20.9",
  "sgf",
  "tokio",
  "uuid 0.8.2",
@@ -3349,11 +3750,23 @@ version = "0.18.3"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "7ca27ec1eb0457ab26f3036ea52229edbdb74dee1edd29063f5b9b010e7ebee4"
 dependencies = [
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "libc",
  "once_cell",
- "pango-sys",
+ "pango-sys 0.18.0",
+]
+
+[[package]]
+name = "pango"
+version = "0.20.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6b1f5dc1b8cf9bc08bfc0843a04ee0fa2e78f1e1fa4b126844a383af4f25f0ec"
+dependencies = [
+ "gio 0.20.9",
+ "glib 0.20.9",
+ "libc",
+ "pango-sys 0.20.9",
 ]
 
 [[package]]
@@ -3362,12 +3775,30 @@ version = "0.18.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "436737e391a843e5933d6d9aa102cb126d501e815b83601365a948a518555dc5"
 dependencies = [
- "glib-sys",
- "gobject-sys",
+ "glib-sys 0.18.1",
+ "gobject-sys 0.18.0",
  "libc",
- "system-deps",
+ "system-deps 6.2.2",
 ]
 
+[[package]]
+name = "pango-sys"
+version = "0.20.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0dbb9b751673bd8fe49eb78620547973a1e719ed431372122b20abd12445bab5"
+dependencies = [
+ "glib-sys 0.20.9",
+ "gobject-sys 0.20.9",
+ "libc",
+ "system-deps 7.0.3",
+]
+
+[[package]]
+name = "panic-halt"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a513e167849a384b7f9b746e517604398518590a9142f4846a32e3c2a4de7b11"
+
 [[package]]
 name = "parking"
 version = "2.2.1"
@@ -3406,6 +3837,12 @@ dependencies = [
  "regex",
 ]
 
+[[package]]
+name = "paste"
+version = "1.0.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a"
+
 [[package]]
 name = "pem-rfc7468"
 version = "0.7.0"
@@ -3504,24 +3941,37 @@ dependencies = [
  "siphasher 1.0.1",
 ]
 
+[[package]]
+name = "pico-st7789"
+version = "0.1.0"
+dependencies = [
+ "bitflags 2.9.0",
+ "cortex-m-rt",
+ "embedded-alloc",
+ "embedded-hal 1.0.0",
+ "fugit",
+ "panic-halt",
+ "rp-pico",
+]
+
 [[package]]
 name = "pin-project"
-version = "1.1.8"
+version = "1.1.10"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "1e2ec53ad785f4d35dac0adea7f7dc6f1bb277ad84a680c7afefeae05d1f5916"
+checksum = "677f1add503faace112b9f1373e43e9e054bfdd22ff1a63c1bc485eaec6a6a8a"
 dependencies = [
  "pin-project-internal",
 ]
 
 [[package]]
 name = "pin-project-internal"
-version = "1.1.8"
+version = "1.1.10"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "d56a66c0c55993aa927429d0f8a0abfd74f084e4d9c192cffed01e418d83eefb"
+checksum = "6e918e4ff8c4549eb882f14b3a4bc8c8bc93de829416eacf579f1207a8fbf861"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -3547,6 +3997,17 @@ dependencies = [
  "thiserror 1.0.69",
 ]
 
+[[package]]
+name = "pio"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "76e09694b50f89f302ed531c1f2a7569f0be5867aee4ab4f8f729bbeec0078e3"
+dependencies = [
+ "arrayvec",
+ "num_enum",
+ "paste",
+]
+
 [[package]]
 name = "piper"
 version = "0.2.4"
@@ -3565,7 +4026,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "08e645ba5c45109106d56610b3ee60eb13a6f2beb8b74f8dc8186cf261788dda"
 dependencies = [
  "anyhow",
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "libc",
  "libspa",
  "libspa-sys",
@@ -3583,7 +4044,7 @@ checksum = "849e188f90b1dda88fe2bfe1ad31fe5f158af2c98f80fb5d13726c44f3f01112"
 dependencies = [
  "bindgen",
  "libspa-sys",
- "system-deps",
+ "system-deps 6.2.2",
 ]
 
 [[package]]
@@ -3609,9 +4070,9 @@ dependencies = [
 
 [[package]]
 name = "pkg-config"
-version = "0.3.31"
+version = "0.3.32"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "953ec861398dccce10c670dfeaf3ec4911ca479e9c02154b3a215178c5f566f2"
+checksum = "7edddbd0b52d732b21ad9a5fab5c704c14cd949e5e9a1ec5929a24fded1b904c"
 
 [[package]]
 name = "plugin"
@@ -3644,7 +4105,7 @@ dependencies = [
  "crc32fast",
  "fdeflate",
  "flate2",
- "miniz_oxide 0.8.3",
+ "miniz_oxide 0.8.5",
 ]
 
 [[package]]
@@ -3657,11 +4118,17 @@ dependencies = [
  "concurrent-queue",
  "hermit-abi 0.4.0",
  "pin-project-lite",
- "rustix",
+ "rustix 0.38.44",
  "tracing",
  "windows-sys 0.59.0",
 ]
 
+[[package]]
+name = "portable-atomic"
+version = "1.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "350e9b48cbc6b0e028b0473b114454c6316e57336ee184ceab6e53f72c178b3e"
+
 [[package]]
 name = "powerfmt"
 version = "0.2.0"
@@ -3670,9 +4137,9 @@ checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391"
 
 [[package]]
 name = "ppv-lite86"
-version = "0.2.20"
+version = "0.2.21"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04"
+checksum = "85eae3c4ed2f50dcfe72643da4befc30deadb458a9b590d720cde2f2b1e97da9"
 dependencies = [
  "zerocopy",
 ]
@@ -3684,17 +4151,17 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "865724d4dbe39d9f3dd3b52b88d859d66bcb2d6a0acfd5ea68a65fb66d4bdc1c"
 dependencies = [
  "env_logger",
- "log 0.4.25",
+ "log 0.4.27",
 ]
 
 [[package]]
 name = "prettyplease"
-version = "0.2.29"
+version = "0.2.31"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "6924ced06e1f7dfe3fa48d57b9f74f55d8915f5036121bef647ef4b204895fac"
+checksum = "5316f57387668042f561aae71480de936257848f9c43ce528e311d89a07cadeb"
 dependencies = [
  "proc-macro2",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -3709,12 +4176,20 @@ dependencies = [
 
 [[package]]
 name = "proc-macro-crate"
-version = "2.0.2"
+version = "2.0.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "b00f26d3400549137f92511a46ac1cd8ce37cb5598a96d382381458b992a5d24"
+checksum = "7e8366a6159044a37876a2b9817124296703c586a5c92e2c53751fa06d8d43e8"
 dependencies = [
- "toml_datetime",
- "toml_edit 0.20.2",
+ "toml_edit 0.20.7",
+]
+
+[[package]]
+name = "proc-macro-crate"
+version = "3.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "edce586971a4dfaa28950c6f18ed55e0406c1ab88bbce2c6f6293a7aaba73d35"
+dependencies = [
+ "toml_edit 0.22.24",
 ]
 
 [[package]]
@@ -3743,9 +4218,9 @@ dependencies = [
 
 [[package]]
 name = "proc-macro2"
-version = "1.0.93"
+version = "1.0.94"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "60946a68e5f9d28b0dc1c21bb8a97ee7d018a8b322fa57838ba31cc878e22d99"
+checksum = "a31971752e70b8b2686d7e46ec17fb38dad4051d94024c88df49b667caea9c84"
 dependencies = [
  "unicode-ident",
 ]
@@ -3758,7 +4233,7 @@ checksum = "14cae93065090804185d3b75f0bf93b8eeda30c7a9b4a33d3bdb3988d6229e50"
 dependencies = [
  "bit-set",
  "bit-vec",
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "lazy_static",
  "num-traits",
  "rand 0.8.5",
@@ -3787,13 +4262,19 @@ checksum = "a1d01941d82fa2ab50be1e79e6714289dd7cde78eba4c074bc5a4374f650dfe0"
 
 [[package]]
 name = "quote"
-version = "1.0.38"
+version = "1.0.40"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "0e4dccaaaf89514f546c693ddc140f729f958c247918a13380cccc6078391acc"
+checksum = "1885c039570dc00dcb4ff087a89e185fd56bae234ddc7f056a945bf36467248d"
 dependencies = [
  "proc-macro2",
 ]
 
+[[package]]
+name = "r-efi"
+version = "5.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "74765f6d916ee2faa39bc8e68e4f3ed8949b48cccdac59983d287a7cb71ce9c5"
+
 [[package]]
 name = "rand"
 version = "0.3.23"
@@ -3993,11 +4474,11 @@ dependencies = [
 
 [[package]]
 name = "redox_syscall"
-version = "0.5.8"
+version = "0.5.10"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "03a862b389f93e68874fbf580b9de08dd02facb9a788ebadaf4a3fd33cf58834"
+checksum = "0b8c0c260b63a8219631167be35e6a988e9554dbd323f8bd08439c8ed1302bd1"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
 ]
 
 [[package]]
@@ -4056,14 +4537,14 @@ dependencies = [
  "hyper-tls",
  "ipnet",
  "js-sys",
- "log 0.4.25",
+ "log 0.4.27",
  "mime 0.3.17",
  "native-tls",
  "once_cell",
  "percent-encoding 2.3.1",
  "pin-project-lite",
  "rustls-pemfile",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "serde_urlencoded",
  "sync_wrapper 0.1.2",
@@ -4086,10 +4567,97 @@ dependencies = [
 ]
 
 [[package]]
-name = "rsa"
-version = "0.9.7"
+name = "rlsf"
+version = "0.2.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "47c75d7c5c6b673e58bf54d8544a9f432e3a925b0e80f7cd3602ab5c50c55519"
+checksum = "222fb240c3286247ecdee6fa5341e7cdad0ffdf8e7e401d9937f2d58482a20bf"
+dependencies = [
+ "cfg-if",
+ "const-default",
+ "libc",
+ "svgbobdoc",
+]
+
+[[package]]
+name = "rp-pico"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b9342d3ac7011ac688300979e9b52a81f0add1d05feb02868cf94bfee0705b28"
+dependencies = [
+ "cortex-m-rt",
+ "fugit",
+ "rp2040-boot2",
+ "rp2040-hal",
+ "usb-device",
+]
+
+[[package]]
+name = "rp2040-boot2"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7c92f344f63f950ee36cf4080050e4dce850839b9175da38f9d2ffb69b4dbb21"
+dependencies = [
+ "crc-any",
+]
+
+[[package]]
+name = "rp2040-hal"
+version = "0.10.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d11e711940087f2cdff8aeae9f4b902e2014c06a00b39a1092686b81ec973d6f"
+dependencies = [
+ "bitfield 0.14.0",
+ "cortex-m",
+ "critical-section",
+ "embedded-dma",
+ "embedded-hal 0.2.7",
+ "embedded-hal 1.0.0",
+ "embedded-hal-async",
+ "embedded-hal-nb",
+ "embedded-io",
+ "frunk",
+ "fugit",
+ "itertools 0.10.5",
+ "nb 1.1.0",
+ "paste",
+ "pio",
+ "rand_core 0.6.4",
+ "rp2040-hal-macros",
+ "rp2040-pac",
+ "usb-device",
+ "vcell",
+ "void",
+]
+
+[[package]]
+name = "rp2040-hal-macros"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "86479063e497efe1ae81995ef9071f54fd1c7427e04d6c5b84cde545ff672a5e"
+dependencies = [
+ "cortex-m-rt",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "rp2040-pac"
+version = "0.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "83cbcd3f7a0ca7bbe61dc4eb7e202842bee4e27b769a7bf3a4a72fa399d6e404"
+dependencies = [
+ "cortex-m",
+ "cortex-m-rt",
+ "critical-section",
+ "vcell",
+]
+
+[[package]]
+name = "rsa"
+version = "0.9.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "78928ac1ed176a5ca1d17e578a1825f3d81ca54cf41053a592584b020cfd691b"
 dependencies = [
  "const-oid",
  "digest",
@@ -4117,25 +4685,47 @@ version = "1.1.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2"
 
+[[package]]
+name = "rustc_version"
+version = "0.2.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "138e3e0acb6c9fb258b19b67cb8abd63c00679d2851805ea151465464fe9030a"
+dependencies = [
+ "semver 0.9.0",
+]
+
 [[package]]
 name = "rustc_version"
 version = "0.4.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "cfcb3a22ef46e85b45de6ee7e79d063319ebb6594faafcf1c225ea92ab6e9b92"
 dependencies = [
- "semver",
+ "semver 1.0.26",
 ]
 
 [[package]]
 name = "rustix"
-version = "0.38.43"
+version = "0.38.44"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "a78891ee6bf2340288408954ac787aa063d8e8817e9f53abb37c695c6d834ef6"
+checksum = "fdb5bc1ae2baa591800df16c9ca78619bf65c0488b41b96ccec5d11220d8c154"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "errno",
  "libc",
- "linux-raw-sys",
+ "linux-raw-sys 0.4.15",
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "rustix"
+version = "1.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e56a18552996ac8d29ecc3b190b4fdbb2d91ca4ec396de7bbffaf43f3d637e96"
+dependencies = [
+ "bitflags 2.9.0",
+ "errno",
+ "libc",
+ "linux-raw-sys 0.9.3",
  "windows-sys 0.59.0",
 ]
 
@@ -4150,9 +4740,9 @@ dependencies = [
 
 [[package]]
 name = "rustversion"
-version = "1.0.19"
+version = "1.0.20"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f7c45b9784283f1b2e7fb61b42047c2fd678ef0960d4f6f1eba131594cc369d4"
+checksum = "eded382c5f5f786b989652c49544c4877d9f015cc22e145a5ea8ea66c2921cd2"
 
 [[package]]
 name = "rusty-fork"
@@ -4168,9 +4758,9 @@ dependencies = [
 
 [[package]]
 name = "ryu"
-version = "1.0.18"
+version = "1.0.20"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f3cb5ba0dc43242ce17de99c180e96db90b235b8a9fdc9543c96d2209116bd9f"
+checksum = "28d3b2b1366ec20994f1fd18c3c594f05c5dd4bc44d8bb0c1c632c8d6829481f"
 
 [[package]]
 name = "safemem"
@@ -4210,7 +4800,7 @@ name = "screenplay"
 version = "0.1.0"
 dependencies = [
  "async-std",
- "glib",
+ "glib 0.18.5",
  "gtk4",
 ]
 
@@ -4220,7 +4810,7 @@ version = "2.11.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "897b2245f0b511c87893af39b033e5ca9cce68824c4d7e7630b5a1d339658d02"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "core-foundation",
  "core-foundation-sys",
  "libc",
@@ -4254,9 +4844,24 @@ checksum = "c2fdfc24bc566f839a2da4c4295b82db7d25a24253867d5c64355abb5799bdbe"
 
 [[package]]
 name = "semver"
-version = "1.0.24"
+version = "0.9.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "3cb6eb87a131f756572d7fb904f6e7b68633f09cca868c5df1c4b8d1a694bbba"
+checksum = "1d7eb9ef2c18661902cc47e535f9bc51b78acd254da71d375c2f6720d9a40403"
+dependencies = [
+ "semver-parser",
+]
+
+[[package]]
+name = "semver"
+version = "1.0.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "56e6fa9c48d24d85fb3de5ad847117517440f6beceb7798af16b4a87d616b8d0"
+
+[[package]]
+name = "semver-parser"
+version = "0.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "388a1df253eca08550bef6c72392cfe7c30914bf41df5269b68cbd6ff8f570a3"
 
 [[package]]
 name = "serde"
@@ -4266,9 +4871,9 @@ checksum = "34b623917345a631dc9608d5194cc206b3fe6c3554cd1c75b937e55e285254af"
 
 [[package]]
 name = "serde"
-version = "1.0.218"
+version = "1.0.219"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "e8dfc9d19bdbf6d17e22319da49161d5d0108e4188e8b680aef6299eed22df60"
+checksum = "5f0e2c6ed6606019b4e29e69dbaba95b11854410e5347d525002456dbbb786b6"
 dependencies = [
  "serde_derive",
 ]
@@ -4280,41 +4885,41 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "8302e169f0eddcc139c70f139d19d6467353af16f9fce27e8c30158036a1e16b"
 dependencies = [
  "js-sys",
- "serde 1.0.218",
+ "serde 1.0.219",
  "wasm-bindgen",
 ]
 
 [[package]]
 name = "serde_derive"
-version = "1.0.218"
+version = "1.0.219"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f09503e191f4e797cb8aac08e9a4a4695c5edf6a2e70e376d961ddd5c969f82b"
+checksum = "5b0276cf7f2c73365f7157c8123c21cd9a50fbbd844757af28ca1f5925fc2a00"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
 name = "serde_json"
-version = "1.0.138"
+version = "1.0.140"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "d434192e7da787e94a6ea7e9670b26a036d0ca41e0b7efb2676dd32bae872949"
+checksum = "20068b6e96dc6c9bd23e01df8827e6c7e1f2fddd43c21810382803c136b99373"
 dependencies = [
  "itoa",
  "memchr",
  "ryu",
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
 name = "serde_path_to_error"
-version = "0.1.16"
+version = "0.1.17"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "af99884400da37c88f5e9146b7f1fd0fbcae8f6eec4e9da38b67d05486f814a6"
+checksum = "59fab13f937fa393d08645bf3a84bdfe86e296747b506ada67bb15f10f218b2a"
 dependencies = [
  "itoa",
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
@@ -4323,7 +4928,7 @@ version = "0.6.8"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "87607cb1398ed59d48732e575a4c28a7a8ebf2454b964fe3f224f2afc07909e1"
 dependencies = [
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
@@ -4335,7 +4940,7 @@ dependencies = [
  "form_urlencoded",
  "itoa",
  "ryu",
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
@@ -4349,7 +4954,7 @@ dependencies = [
  "libyml",
  "memchr",
  "ryu",
- "serde 1.0.218",
+ "serde 1.0.219",
  "version_check 0.9.5",
 ]
 
@@ -4359,12 +4964,12 @@ version = "0.1.0"
 dependencies = [
  "axum",
  "result-extended",
- "serde 1.0.218",
- "thiserror 2.0.11",
+ "serde 1.0.219",
+ "thiserror 2.0.12",
  "tokio",
  "tower-http",
  "typeshare",
- "uuid 1.13.2",
+ "uuid 1.16.0",
  "visions-types",
 ]
 
@@ -4376,7 +4981,7 @@ dependencies = [
  "cool_asserts",
  "nary_tree",
  "nom",
- "serde 1.0.218",
+ "serde 1.0.219",
  "thiserror 1.0.69",
  "typeshare",
  "uuid 0.8.2",
@@ -4442,12 +5047,12 @@ dependencies = [
  "async-std",
  "cairo-rs",
  "fixed",
- "gio",
- "glib",
+ "gio 0.18.4",
+ "glib 0.18.5",
  "gtk4",
  "libadwaita",
  "lights-core",
- "pango",
+ "pango 0.20.9",
 ]
 
 [[package]]
@@ -4473,11 +5078,11 @@ dependencies = [
 
 [[package]]
 name = "smallvec"
-version = "1.13.2"
+version = "1.14.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67"
+checksum = "7fcf8323ef1faaee30a44a340193b1ac6814fd9b7b4e88e9d4519a3e4abe1cfd"
 dependencies = [
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
@@ -4546,15 +5151,15 @@ dependencies = [
  "hashbrown",
  "hashlink",
  "indexmap",
- "log 0.4.25",
+ "log 0.4.27",
  "memchr",
  "once_cell",
  "percent-encoding 2.3.1",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "sha2",
  "smallvec",
- "thiserror 2.0.11",
+ "thiserror 2.0.12",
  "tokio",
  "tokio-stream",
  "tracing",
@@ -4571,7 +5176,7 @@ dependencies = [
  "quote",
  "sqlx-core",
  "sqlx-macros-core",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -4587,14 +5192,14 @@ dependencies = [
  "once_cell",
  "proc-macro2",
  "quote",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "sha2",
  "sqlx-core",
  "sqlx-mysql",
  "sqlx-postgres",
  "sqlx-sqlite",
- "syn 2.0.96",
+ "syn 2.0.100",
  "tempfile",
  "tokio",
  "url 2.5.4",
@@ -4608,7 +5213,7 @@ checksum = "4560278f0e00ce64938540546f59f590d60beee33fffbd3b9cd47851e5fff233"
 dependencies = [
  "atoi",
  "base64 0.22.1",
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "byteorder",
  "bytes",
  "crc",
@@ -4624,20 +5229,20 @@ dependencies = [
  "hkdf",
  "hmac",
  "itoa",
- "log 0.4.25",
+ "log 0.4.27",
  "md-5",
  "memchr",
  "once_cell",
  "percent-encoding 2.3.1",
  "rand 0.8.5",
  "rsa",
- "serde 1.0.218",
+ "serde 1.0.219",
  "sha1",
  "sha2",
  "smallvec",
  "sqlx-core",
  "stringprep",
- "thiserror 2.0.11",
+ "thiserror 2.0.12",
  "tracing",
  "whoami",
 ]
@@ -4650,7 +5255,7 @@ checksum = "c5b98a57f363ed6764d5b3a12bfedf62f07aa16e1856a7ddc2a0bb190a959613"
 dependencies = [
  "atoi",
  "base64 0.22.1",
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "byteorder",
  "crc",
  "dotenvy",
@@ -4663,18 +5268,18 @@ dependencies = [
  "hmac",
  "home",
  "itoa",
- "log 0.4.25",
+ "log 0.4.27",
  "md-5",
  "memchr",
  "once_cell",
  "rand 0.8.5",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "sha2",
  "smallvec",
  "sqlx-core",
  "stringprep",
- "thiserror 2.0.11",
+ "thiserror 2.0.12",
  "tracing",
  "whoami",
 ]
@@ -4693,9 +5298,9 @@ dependencies = [
  "futures-intrusive",
  "futures-util",
  "libsqlite3-sys",
- "log 0.4.25",
+ "log 0.4.27",
  "percent-encoding 2.3.1",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_urlencoded",
  "sqlx-core",
  "tracing",
@@ -4731,6 +5336,19 @@ version = "2.6.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292"
 
+[[package]]
+name = "svgbobdoc"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f2c04b93fc15d79b39c63218f15e3fdffaa4c227830686e3b7c5f41244eb3e50"
+dependencies = [
+ "base64 0.13.1",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+ "unicode-width",
+]
+
 [[package]]
 name = "syn"
 version = "1.0.109"
@@ -4744,9 +5362,9 @@ dependencies = [
 
 [[package]]
 name = "syn"
-version = "2.0.96"
+version = "2.0.100"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "d5d0adab1ae378d7f53bdebc67a39f1f151407ef230f0ce2883572f5d8985c80"
+checksum = "b09a44accad81e1ba1cd74a32461ba89dee89095ba17b32f5d03683b1b1fc2a0"
 dependencies = [
  "proc-macro2",
  "quote",
@@ -4773,7 +5391,7 @@ checksum = "c8af7666ab7b6390ab78131fb5b0fce11d6b7a6951602017c35fa82800708971"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -4803,7 +5421,20 @@ version = "6.2.2"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "a3e535eb8dded36d55ec13eddacd30dec501792ff23a0b1682c38601b8cf2349"
 dependencies = [
- "cfg-expr",
+ "cfg-expr 0.15.8",
+ "heck 0.5.0",
+ "pkg-config",
+ "toml",
+ "version-compare",
+]
+
+[[package]]
+name = "system-deps"
+version = "7.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "66d23aaf9f331227789a99e8de4c91bf46703add012bdfd45fdecdfb2975a005"
+dependencies = [
+ "cfg-expr 0.17.2",
  "heck 0.5.0",
  "pkg-config",
  "toml",
@@ -4828,15 +5459,14 @@ dependencies = [
 
 [[package]]
 name = "tempfile"
-version = "3.15.0"
+version = "3.19.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "9a8a559c81686f576e8cd0290cd2a24a2a9ad80c98b3478856500fcbd7acd704"
+checksum = "7437ac7763b9b123ccf33c338a5cc1bac6f69b45a136c19bdd8a65e3916435bf"
 dependencies = [
- "cfg-if",
  "fastrand",
- "getrandom 0.2.15",
+ "getrandom 0.3.2",
  "once_cell",
- "rustix",
+ "rustix 1.0.3",
  "windows-sys 0.59.0",
 ]
 
@@ -4860,11 +5490,11 @@ dependencies = [
 
 [[package]]
 name = "thiserror"
-version = "2.0.11"
+version = "2.0.12"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "d452f284b73e6d76dd36758a0c8684b1d5be31f92b89d07fd5822175732206fc"
+checksum = "567b8a2dae586314f7be2a752ec7474332959c6460e02bde30d702a66d488708"
 dependencies = [
- "thiserror-impl 2.0.11",
+ "thiserror-impl 2.0.12",
 ]
 
 [[package]]
@@ -4875,18 +5505,18 @@ checksum = "4fee6c4efc90059e10f81e6d42c60a18f76588c3d74cb83a0b242a2b6c7504c1"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
 name = "thiserror-impl"
-version = "2.0.11"
+version = "2.0.12"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "26afc1baea8a989337eeb52b6e72a039780ce45c3edfcc9c5b9d112feeb173c2"
+checksum = "7f7cf42b4507d8ea322120659672cf1b9dbb93f8f2d4ecfd6e51350ff5b17a1d"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -4924,30 +5554,30 @@ dependencies = [
 
 [[package]]
 name = "time"
-version = "0.3.37"
+version = "0.3.41"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "35e7868883861bd0e56d9ac6efcaaca0d6d5d82a2a7ec8209ff492c07cf37b21"
+checksum = "8a7619e19bc266e0f9c5e6686659d394bc57973859340060a69221e57dbc0c40"
 dependencies = [
  "deranged",
  "itoa",
  "num-conv",
  "powerfmt",
- "serde 1.0.218",
+ "serde 1.0.219",
  "time-core",
  "time-macros",
 ]
 
 [[package]]
 name = "time-core"
-version = "0.1.2"
+version = "0.1.4"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ef927ca75afb808a4d64dd374f00a2adf8d0fcff8e7b184af886c3c87ec4a3f3"
+checksum = "c9e9a38711f559d9e3ce1cdb06dd7c5b8ea546bc90052da6d06bb76da74bb07c"
 
 [[package]]
 name = "time-macros"
-version = "0.2.19"
+version = "0.2.22"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "2834e6017e3e5e4b9834939793b282bc03b37a3336245fa820e35e233e2a85de"
+checksum = "3526739392ec93fd8b359c8e98514cb3e8e021beb4e5f597b00a0221f8ed8a49"
 dependencies = [
  "num-conv",
  "time-core",
@@ -4968,14 +5598,15 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "9117f5d4db391c1cf6927e7bea3db74b9a1c1add8f7eda9ffd5364f40f57b82f"
 dependencies = [
  "displaydoc",
+ "serde 1.0.219",
  "zerovec",
 ]
 
 [[package]]
 name = "tinyvec"
-version = "1.8.1"
+version = "1.9.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "022db8904dfa342efe721985167e9fcd16c29b226db4397ed752a761cfce81e8"
+checksum = "09b3661f17e86524eccd4371ab0429194e0d7c008abb45f7a7495b1719463c71"
 dependencies = [
  "tinyvec_macros",
 ]
@@ -4988,9 +5619,9 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20"
 
 [[package]]
 name = "tokio"
-version = "1.43.0"
+version = "1.44.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "3d61fa4ffa3de412bfea335c6ecff681de2b609ba3c77ef3e00e521813a9ed9e"
+checksum = "f382da615b842244d4b8738c82ed1275e6c5dd90c459a30941cd07080b06c91a"
 dependencies = [
  "backtrace",
  "bytes",
@@ -5012,7 +5643,7 @@ checksum = "6e06d43f1345a3bcd39f6a56dbb7dcab2ba47e68e8ac134855e7e2bdbaf8cab8"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -5043,16 +5674,16 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "c83b561d025642014097b66e6c1bb422783339e0909e4429cde4749d1990bc38"
 dependencies = [
  "futures-util",
- "log 0.4.25",
+ "log 0.4.27",
  "tokio",
  "tungstenite",
 ]
 
 [[package]]
 name = "tokio-util"
-version = "0.7.13"
+version = "0.7.14"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "d7fcaa8d55a2bdd6b83ace262b016eca0d79ee02818c5c1bcdf0305114081078"
+checksum = "6b9590b93e6fcc1739458317cccd391ad3955e2bde8913edf6f95f9e65a8f034"
 dependencies = [
  "bytes",
  "futures-core",
@@ -5063,9 +5694,9 @@ dependencies = [
 
 [[package]]
 name = "tokise"
-version = "0.1.0"
+version = "0.2.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "fae352452702497bdbde0ce9c79a27fa488c73be42c77e0e9ee4a1b5cce97596"
+checksum = "decf97738ce15b9e9cc1671ea29b0f6c56538719e1a092d19cc2134bf144e40e"
 dependencies = [
  "futures",
  "gloo",
@@ -5080,23 +5711,23 @@ dependencies = [
 
 [[package]]
 name = "toml"
-version = "0.8.2"
+version = "0.8.20"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "185d8ab0dfbb35cf1399a6344d8484209c088f75f8f68230da55d48d95d43e3d"
+checksum = "cd87a5cdd6ffab733b2f74bc4fd7ee5fff6634124999ac278c35fc78c6120148"
 dependencies = [
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_spanned",
  "toml_datetime",
- "toml_edit 0.20.2",
+ "toml_edit 0.22.24",
 ]
 
 [[package]]
 name = "toml_datetime"
-version = "0.6.3"
+version = "0.6.8"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "7cda73e2f1397b1262d6dfdcef8aafae14d1de7748d66822d3bfeeb6d03e5e4b"
+checksum = "0dd7358ecb8fc2f8d014bf86f6f638ce72ba252a2c3a2572f2a795f1d23efb41"
 dependencies = [
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
@@ -5107,20 +5738,31 @@ checksum = "1b5bb770da30e5cbfde35a2d7b9b8a2c4b8ef89548a7a6aeab5c9a576e3e7421"
 dependencies = [
  "indexmap",
  "toml_datetime",
- "winnow",
+ "winnow 0.5.40",
 ]
 
 [[package]]
 name = "toml_edit"
-version = "0.20.2"
+version = "0.20.7"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "396e4d48bbb2b7554c944bde63101b5ae446cff6ec4a24227428f15eb72ef338"
+checksum = "70f427fce4d84c72b5b732388bf4a9f4531b53f74e2887e3ecb2481f68f66d81"
 dependencies = [
  "indexmap",
- "serde 1.0.218",
+ "toml_datetime",
+ "winnow 0.5.40",
+]
+
+[[package]]
+name = "toml_edit"
+version = "0.22.24"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "17b4795ff5edd201c7cd6dca065ae59972ce77d1b80fa0a84d94950ece7d1474"
+dependencies = [
+ "indexmap",
+ "serde 1.0.219",
  "serde_spanned",
  "toml_datetime",
- "winnow",
+ "winnow 0.7.4",
 ]
 
 [[package]]
@@ -5145,9 +5787,9 @@ version = "0.6.2"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "403fa3b783d4b626a8ad51d766ab03cb6d2dbfc46b1c5d4448395e6628dc9697"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
  "bytes",
- "http 1.2.0",
+ "http 1.3.1",
  "pin-project-lite",
  "tower-layer",
  "tower-service",
@@ -5171,7 +5813,7 @@ version = "0.1.41"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "784e0ac535deb450455cbfa28a6f0df145ea1bb7ae51b821cf5e7927fdcfbdd0"
 dependencies = [
- "log 0.4.25",
+ "log 0.4.27",
  "pin-project-lite",
  "tracing-attributes",
  "tracing-core",
@@ -5185,7 +5827,7 @@ checksum = "395ae124c09f9e6918a2310af6038fba074bcf474ac352496d5910dd59a2226d"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -5199,9 +5841,9 @@ dependencies = [
 
 [[package]]
 name = "traitobject"
-version = "0.1.0"
+version = "0.1.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "efd1f82c56340fdf16f2a953d7bda4f8fdffba13d93b00844c25572110b26079"
+checksum = "04a79e25382e2e852e8da874249358d382ebaf259d0d34e75d8db16a7efabbc7"
 
 [[package]]
 name = "tree"
@@ -5222,9 +5864,9 @@ dependencies = [
  "byteorder",
  "bytes",
  "data-encoding",
- "http 1.2.0",
+ "http 1.3.1",
  "httparse",
- "log 0.4.25",
+ "log 0.4.27",
  "rand 0.8.5",
  "sha1",
  "thiserror 1.0.69",
@@ -5258,9 +5900,9 @@ dependencies = [
 
 [[package]]
 name = "typenum"
-version = "1.17.0"
+version = "1.18.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825"
+checksum = "1dccffe3ce07af9386bfd29e80c0ab1a8205a2fc34e4bcd40364df902cfa8f3f"
 
 [[package]]
 name = "typeshare"
@@ -5269,7 +5911,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "19be0f411120091e76e13e5a0186d8e2bcc3e7e244afdb70152197f1a8486ceb"
 dependencies = [
  "chrono",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "typeshare-annotation",
 ]
@@ -5281,7 +5923,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "a615d6c2764852a2e88a4f16e9ce1ea49bb776b5872956309e170d63a042a34f"
 dependencies = [
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -5331,9 +5973,9 @@ checksum = "5c1cb5db39152898a79168971543b1cb5020dff7fe43c8dc468b0885f5e29df5"
 
 [[package]]
 name = "unicode-ident"
-version = "1.0.14"
+version = "1.0.18"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "adb9e6ca4f869e1180728b7950e35922a7fc6397f7b641499e8f3ef06e50dc83"
+checksum = "5a5f39404a5da50712a4c1eecf25e90dd62b613502b7e925fd4e4d19b5c96512"
 
 [[package]]
 name = "unicode-normalization"
@@ -5393,6 +6035,16 @@ dependencies = [
  "percent-encoding 2.3.1",
 ]
 
+[[package]]
+name = "usb-device"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "98816b1accafbb09085168b90f27e93d790b4bfa19d883466b5e53315b5f06a6"
+dependencies = [
+ "heapless",
+ "portable-atomic",
+]
+
 [[package]]
 name = "utf-8"
 version = "0.7.6"
@@ -5434,25 +6086,31 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "bc5cf98d8186244414c848017f0e2676b3fcb46807f6668a97dfe67359a3c4b7"
 dependencies = [
  "getrandom 0.2.15",
- "serde 1.0.218",
+ "serde 1.0.219",
 ]
 
 [[package]]
 name = "uuid"
-version = "1.13.2"
+version = "1.16.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "8c1f41ffb7cf259f1ecc2876861a17e7142e63ead296f671f81f6ae85903e0d6"
+checksum = "458f7a779bf54acc9f347480ac654f68407d3aab21269a6e3c9f922acd9e2da9"
 dependencies = [
- "getrandom 0.3.1",
+ "getrandom 0.3.2",
  "js-sys",
  "wasm-bindgen",
 ]
 
 [[package]]
 name = "value-bag"
-version = "1.10.0"
+version = "1.11.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "3ef4c4aa54d5d05a279399bfa921ec387b7aba77caf7a682ae8d86785b8fdad2"
+checksum = "943ce29a8a743eb10d6082545d861b24f9d1b160b7d741e0f2cdf726bec909c5"
+
+[[package]]
+name = "vcell"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "77439c1b53d2303b20d9459b1ade71a83c716e3f9c34f3228c00e6f185d6c002"
 
 [[package]]
 name = "vcpkg"
@@ -5484,7 +6142,7 @@ version = "0.1.0"
 dependencies = [
  "gloo-console",
  "gloo-net 0.6.0",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde-wasm-bindgen",
  "serde_json",
  "visions-types",
@@ -5498,15 +6156,30 @@ dependencies = [
 name = "visions-types"
 version = "0.1.0"
 dependencies = [
- "serde 1.0.218",
- "uuid 1.13.2",
+ "serde 1.0.219",
+ "uuid 1.16.0",
+]
+
+[[package]]
+name = "void"
+version = "1.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6a02e4885ed3bc0f2de90ea6dd45ebcbb66dacffe03547fadbb0eeae2770887d"
+
+[[package]]
+name = "volatile-register"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "de437e2a6208b014ab52972a27e59b33fa2920d3e00fe05026167a1c509d19cc"
+dependencies = [
+ "vcell",
 ]
 
 [[package]]
 name = "wait-timeout"
-version = "0.2.0"
+version = "0.2.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "9f200f5b12eb75f8c1ed65abd4b2db8a6e1b138a20de009dacee265a2498f3f6"
+checksum = "09ac3b126d3914f9849036f826e054cbabdc8519970b8998ddaf3b5bd3c65f11"
 dependencies = [
  "libc",
 ]
@@ -5532,14 +6205,14 @@ dependencies = [
  "headers",
  "http 0.2.12",
  "hyper 0.14.32",
- "log 0.4.25",
+ "log 0.4.27",
  "mime 0.3.17",
  "mime_guess 2.0.5",
  "multer",
  "percent-encoding 2.3.1",
  "pin-project",
  "scoped-tls",
- "serde 1.0.218",
+ "serde 1.0.219",
  "serde_json",
  "serde_urlencoded",
  "tokio",
@@ -5563,9 +6236,9 @@ checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423"
 
 [[package]]
 name = "wasi"
-version = "0.13.3+wasi-0.2.2"
+version = "0.14.2+wasi-0.2.4"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "26816d2e1a4a36a2940b96c5296ce403917633dff8f3440e9b236ed6f6bacad2"
+checksum = "9683f9a5a998d873c0d21fcbe3c083009670149a8fab228644b8bd36b2c48cb3"
 dependencies = [
  "wit-bindgen-rt",
 ]
@@ -5595,10 +6268,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "2f0a0651a5c2bc21487bde11ee802ccaf4c51935d0d3d42a6101f98161700bc6"
 dependencies = [
  "bumpalo",
- "log 0.4.25",
+ "log 0.4.27",
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
  "wasm-bindgen-shared",
 ]
 
@@ -5633,7 +6306,7 @@ checksum = "8ae87ea40c9f689fc23f209965b6fb8a99ad69aeeb0231408be24920604395de"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
  "wasm-bindgen-backend",
  "wasm-bindgen-shared",
 ]
@@ -5665,9 +6338,9 @@ checksum = "53a85b86a771b1c87058196170769dd264f66c0782acf1ae6cc51bfd64b39082"
 
 [[package]]
 name = "whoami"
-version = "1.5.2"
+version = "1.6.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "372d5b87f58ec45c384ba03563b03544dc5fadc3983e434b286913f5b4a9bb6d"
+checksum = "6994d13118ab492c3c80c1f81928718159254c53c472bf9ce36f8dae4add02a7"
 dependencies = [
  "redox_syscall",
  "wasite",
@@ -5713,6 +6386,12 @@ dependencies = [
  "windows-targets 0.52.6",
 ]
 
+[[package]]
+name = "windows-link"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "76840935b766e1b0a05c0066835fb9ec80071d4c09a16f6bd5f7e655e3c14c38"
+
 [[package]]
 name = "windows-sys"
 version = "0.48.0"
@@ -5870,6 +6549,15 @@ dependencies = [
  "memchr",
 ]
 
+[[package]]
+name = "winnow"
+version = "0.7.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0e97b544156e9bebe1a0ffbc03484fc1ffe3100cbce3ffb17eac35f7cdd7ab36"
+dependencies = [
+ "memchr",
+]
+
 [[package]]
 name = "winreg"
 version = "0.50.0"
@@ -5882,11 +6570,11 @@ dependencies = [
 
 [[package]]
 name = "wit-bindgen-rt"
-version = "0.33.0"
+version = "0.39.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "3268f3d866458b787f390cf61f4bbb563b922d091359f9608842999eaee3943c"
+checksum = "6f42320e61fe2cfd34354ecb597f86f413484a798ba44a8ca1165c58d42da6c1"
 dependencies = [
- "bitflags 2.8.0",
+ "bitflags 2.9.0",
 ]
 
 [[package]]
@@ -5901,6 +6589,12 @@ version = "0.5.5"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "1e9df38ee2d2c3c5948ea468a8406ff0db0b29ae1ffde1bcf20ef305bcc95c51"
 
+[[package]]
+name = "xml-rs"
+version = "0.8.25"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c5b940ebc25896e71dd073bad2dbaa2abfe97b0a391415e22ad1326d9c54e3c4"
+
 [[package]]
 name = "yansi-term"
 version = "0.1.2"
@@ -5913,7 +6607,7 @@ dependencies = [
 [[package]]
 name = "yew"
 version = "0.21.0"
-source = "git+https://github.com/yewstack/yew/#69e3b5f8140180acd1c2de41e421960cbacf768f"
+source = "git+https://github.com/yewstack/yew/#b0d065626175f009cde687980f3724de8dc27240"
 dependencies = [
  "console_error_panic_hook",
  "futures",
@@ -5922,9 +6616,9 @@ dependencies = [
  "indexmap",
  "js-sys",
  "rustversion",
- "serde 1.0.218",
+ "serde 1.0.219",
  "slab",
- "thiserror 1.0.69",
+ "thiserror 2.0.12",
  "tokio",
  "tokise",
  "tracing",
@@ -5934,27 +6628,18 @@ dependencies = [
  "yew-macro",
 ]
 
-[[package]]
-name = "yew-app"
-version = "0.1.0"
-dependencies = [
- "gloo-net 0.6.0",
- "serde 1.0.218",
- "wasm-bindgen-futures",
- "yew",
-]
-
 [[package]]
 name = "yew-macro"
 version = "0.21.0"
-source = "git+https://github.com/yewstack/yew/#69e3b5f8140180acd1c2de41e421960cbacf768f"
+source = "git+https://github.com/yewstack/yew/#b0d065626175f009cde687980f3724de8dc27240"
 dependencies = [
  "once_cell",
  "prettyplease",
  "proc-macro-error",
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "rustversion",
+ "syn 2.0.100",
 ]
 
 [[package]]
@@ -5963,7 +6648,7 @@ version = "0.7.5"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "120e6aef9aa629e3d4f52dc8cc43a015c7724194c97dfaf45180d2daf2b77f40"
 dependencies = [
- "serde 1.0.218",
+ "serde 1.0.219",
  "stable_deref_trait",
  "yoke-derive",
  "zerofrom",
@@ -5977,49 +6662,48 @@ checksum = "2380878cad4ac9aac1e2435f3eb4020e8374b5f13c296cb75b4620ff8e229154"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
  "synstructure",
 ]
 
 [[package]]
 name = "zerocopy"
-version = "0.7.35"
+version = "0.8.24"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0"
+checksum = "2586fea28e186957ef732a5f8b3be2da217d65c5969d4b1e17f973ebbe876879"
 dependencies = [
- "byteorder",
  "zerocopy-derive",
 ]
 
 [[package]]
 name = "zerocopy-derive"
-version = "0.7.35"
+version = "0.8.24"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e"
+checksum = "a996a8f63c5c4448cd959ac1bab0aaa3306ccfd060472f85943ee0750f0169be"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
 name = "zerofrom"
-version = "0.1.5"
+version = "0.1.6"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "cff3ee08c995dee1859d998dea82f7374f2826091dd9cd47def953cae446cd2e"
+checksum = "50cc42e0333e05660c3587f3bf9d0478688e15d870fab3346451ce7f8c9fbea5"
 dependencies = [
  "zerofrom-derive",
 ]
 
 [[package]]
 name = "zerofrom-derive"
-version = "0.1.5"
+version = "0.1.6"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "595eed982f7d355beb85837f651fa22e90b3c044842dc7f2c2842c086f295808"
+checksum = "d71e5d6e06ab090c67b5e44993ec16b72dcbaabc526db883a360057678b48502"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
  "synstructure",
 ]
 
@@ -6048,7 +6732,7 @@ checksum = "6eafa6dfb17584ea3e2bd6e76e0cc15ad7af12b09abdd1ca55961bed9b1063c6"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.96",
+ "syn 2.0.100",
 ]
 
 [[package]]
diff --git a/Cargo.nix b/Cargo.nix
index 5e3a985..5af2594 100644
--- a/Cargo.nix
+++ b/Cargo.nix
@@ -225,6 +225,16 @@ rec {
       # File a bug if you depend on any for non-debug work!
       debug = internal.debugCrate { inherit packageId; };
     };
+    "l10n-db" = rec {
+      packageId = "l10n-db";
+      build = internal.buildRustCrateWithFeatures {
+        packageId = "l10n-db";
+      };
+
+      # Debug support which might change between releases.
+      # File a bug if you depend on any for non-debug work!
+      debug = internal.debugCrate { inherit packageId; };
+    };
     "lights-core" = rec {
       packageId = "lights-core";
       build = internal.buildRustCrateWithFeatures {
@@ -295,6 +305,16 @@ rec {
       # File a bug if you depend on any for non-debug work!
       debug = internal.debugCrate { inherit packageId; };
     };
+    "server" = rec {
+      packageId = "server";
+      build = internal.buildRustCrateWithFeatures {
+        packageId = "server";
+      };
+
+      # Debug support which might change between releases.
+      # File a bug if you depend on any for non-debug work!
+      debug = internal.debugCrate { inherit packageId; };
+    };
     "sgf" = rec {
       packageId = "sgf";
       build = internal.buildRustCrateWithFeatures {
@@ -331,16 +351,6 @@ rec {
         packageId = "tree";
       };
 
-      # Debug support which might change between releases.
-      # File a bug if you depend on any for non-debug work!
-      debug = internal.debugCrate { inherit packageId; };
-    };
-    "visions" = rec {
-      packageId = "visions";
-      build = internal.buildRustCrateWithFeatures {
-        packageId = "visions";
-      };
-
       # Debug support which might change between releases.
       # File a bug if you depend on any for non-debug work!
       debug = internal.debugCrate { inherit packageId; };
@@ -448,49 +458,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" "std" ];
       };
-      "ahash" = rec {
-        crateName = "ahash";
-        version = "0.8.11";
-        edition = "2018";
-        sha256 = "04chdfkls5xmhp1d48gnjsmglbqibizs3bpbj6rsj604m10si7g8";
-        authors = [
-          "Tom Kaitchuck <Tom.Kaitchuck@gmail.com>"
-        ];
-        dependencies = [
-          {
-            name = "cfg-if";
-            packageId = "cfg-if";
-          }
-          {
-            name = "once_cell";
-            packageId = "once_cell";
-            usesDefaultFeatures = false;
-            target = { target, features }: (!(("arm" == target."arch") && ("none" == target."os")));
-            features = [ "alloc" ];
-          }
-          {
-            name = "zerocopy";
-            packageId = "zerocopy";
-            usesDefaultFeatures = false;
-            features = [ "simd" ];
-          }
-        ];
-        buildDependencies = [
-          {
-            name = "version_check";
-            packageId = "version_check 0.9.5";
-          }
-        ];
-        features = {
-          "atomic-polyfill" = [ "dep:atomic-polyfill" "once_cell/atomic-polyfill" ];
-          "compile-time-rng" = [ "const-random" ];
-          "const-random" = [ "dep:const-random" ];
-          "default" = [ "std" "runtime-rng" ];
-          "getrandom" = [ "dep:getrandom" ];
-          "runtime-rng" = [ "getrandom" ];
-          "serde" = [ "dep:serde" ];
-        };
-      };
       "aho-corasick" = rec {
         crateName = "aho-corasick";
         version = "1.1.3";
@@ -721,34 +688,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" "std" ];
       };
-      "assert-json-diff" = rec {
-        crateName = "assert-json-diff";
-        version = "2.0.2";
-        edition = "2018";
-        sha256 = "04mg3w0rh3schpla51l18362hsirl23q93aisws2irrj32wg5r27";
-        libName = "assert_json_diff";
-        authors = [
-          "David Pedersen <david.pdrsn@gmail.com>"
-        ];
-        dependencies = [
-          {
-            name = "serde";
-            packageId = "serde 1.0.217";
-          }
-          {
-            name = "serde_json";
-            packageId = "serde_json";
-          }
-        ];
-        devDependencies = [
-          {
-            name = "serde";
-            packageId = "serde 1.0.217";
-            features = [ "derive" ];
-          }
-        ];
-
-      };
       "async-channel 1.9.0" = rec {
         crateName = "async-channel";
         version = "1.9.0";
@@ -1267,7 +1206,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -1301,17 +1240,6 @@ rec {
           }
         ];
 
-      };
-      "auto-future" = rec {
-        crateName = "auto-future";
-        version = "1.0.0";
-        edition = "2021";
-        sha256 = "0wykbakzh227vz6frx9p48zsq0wpswgmb7v3917m53m7gr2pw7iw";
-        libName = "auto_future";
-        authors = [
-          "Joseph Lenton <josephlenton@gmail.com>"
-        ];
-
       };
       "autocfg 0.1.8" = rec {
         crateName = "autocfg";
@@ -1339,429 +1267,6 @@ rec {
         ];
 
       };
-      "axum" = rec {
-        crateName = "axum";
-        version = "0.7.9";
-        edition = "2021";
-        sha256 = "07z7wqczi9i8xb4460rvn39p4wjqwr32hx907crd1vwb2fy8ijpd";
-        dependencies = [
-          {
-            name = "async-trait";
-            packageId = "async-trait";
-          }
-          {
-            name = "axum-core";
-            packageId = "axum-core";
-          }
-          {
-            name = "axum-macros";
-            packageId = "axum-macros";
-            optional = true;
-          }
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            usesDefaultFeatures = false;
-            features = [ "alloc" ];
-          }
-          {
-            name = "http";
-            packageId = "http 1.2.0";
-          }
-          {
-            name = "http-body";
-            packageId = "http-body 1.0.1";
-          }
-          {
-            name = "http-body-util";
-            packageId = "http-body-util";
-          }
-          {
-            name = "hyper";
-            packageId = "hyper 1.5.2";
-            optional = true;
-          }
-          {
-            name = "hyper-util";
-            packageId = "hyper-util";
-            optional = true;
-            features = [ "tokio" "server" "service" ];
-          }
-          {
-            name = "itoa";
-            packageId = "itoa";
-          }
-          {
-            name = "matchit";
-            packageId = "matchit";
-          }
-          {
-            name = "memchr";
-            packageId = "memchr";
-          }
-          {
-            name = "mime";
-            packageId = "mime 0.3.17";
-          }
-          {
-            name = "percent-encoding";
-            packageId = "percent-encoding 2.3.1";
-          }
-          {
-            name = "pin-project-lite";
-            packageId = "pin-project-lite";
-          }
-          {
-            name = "rustversion";
-            packageId = "rustversion";
-          }
-          {
-            name = "serde";
-            packageId = "serde 1.0.217";
-          }
-          {
-            name = "serde_json";
-            packageId = "serde_json";
-            optional = true;
-            features = [ "raw_value" ];
-          }
-          {
-            name = "serde_path_to_error";
-            packageId = "serde_path_to_error";
-            optional = true;
-          }
-          {
-            name = "serde_urlencoded";
-            packageId = "serde_urlencoded";
-            optional = true;
-          }
-          {
-            name = "sync_wrapper";
-            packageId = "sync_wrapper 1.0.2";
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            rename = "tokio";
-            optional = true;
-            features = [ "time" ];
-          }
-          {
-            name = "tower";
-            packageId = "tower";
-            usesDefaultFeatures = false;
-            features = [ "util" ];
-          }
-          {
-            name = "tower-layer";
-            packageId = "tower-layer";
-          }
-          {
-            name = "tower-service";
-            packageId = "tower-service";
-          }
-          {
-            name = "tracing";
-            packageId = "tracing";
-            optional = true;
-            usesDefaultFeatures = false;
-          }
-        ];
-        devDependencies = [
-          {
-            name = "axum-macros";
-            packageId = "axum-macros";
-            features = [ "__private" ];
-          }
-          {
-            name = "serde";
-            packageId = "serde 1.0.217";
-            features = [ "derive" ];
-          }
-          {
-            name = "serde_json";
-            packageId = "serde_json";
-            features = [ "raw_value" ];
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            rename = "tokio";
-            features = [ "macros" "rt" "rt-multi-thread" "net" "test-util" ];
-          }
-          {
-            name = "tower";
-            packageId = "tower";
-            rename = "tower";
-            features = [ "util" "timeout" "limit" "load-shed" "steer" "filter" ];
-          }
-          {
-            name = "tracing";
-            packageId = "tracing";
-          }
-        ];
-        features = {
-          "__private_docs" = [ "axum-core/__private_docs" "tower/full" "dep:tower-http" ];
-          "default" = [ "form" "http1" "json" "matched-path" "original-uri" "query" "tokio" "tower-log" "tracing" ];
-          "form" = [ "dep:serde_urlencoded" ];
-          "http1" = [ "dep:hyper" "hyper?/http1" "hyper-util?/http1" ];
-          "http2" = [ "dep:hyper" "hyper?/http2" "hyper-util?/http2" ];
-          "json" = [ "dep:serde_json" "dep:serde_path_to_error" ];
-          "macros" = [ "dep:axum-macros" ];
-          "multipart" = [ "dep:multer" ];
-          "query" = [ "dep:serde_urlencoded" ];
-          "tokio" = [ "dep:hyper-util" "dep:tokio" "tokio/net" "tokio/rt" "tower/make" "tokio/macros" ];
-          "tower-log" = [ "tower/log" ];
-          "tracing" = [ "dep:tracing" "axum-core/tracing" ];
-          "ws" = [ "dep:hyper" "tokio" "dep:tokio-tungstenite" "dep:sha1" "dep:base64" ];
-        };
-        resolvedDefaultFeatures = [ "default" "form" "http1" "json" "macros" "matched-path" "original-uri" "query" "tokio" "tower-log" "tracing" ];
-      };
-      "axum-core" = rec {
-        crateName = "axum-core";
-        version = "0.4.5";
-        edition = "2021";
-        sha256 = "16b1496c4gm387q20hkv5ic3k5bd6xmnvk50kwsy6ymr8rhvvwh9";
-        libName = "axum_core";
-        dependencies = [
-          {
-            name = "async-trait";
-            packageId = "async-trait";
-          }
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            usesDefaultFeatures = false;
-            features = [ "alloc" ];
-          }
-          {
-            name = "http";
-            packageId = "http 1.2.0";
-          }
-          {
-            name = "http-body";
-            packageId = "http-body 1.0.1";
-          }
-          {
-            name = "http-body-util";
-            packageId = "http-body-util";
-          }
-          {
-            name = "mime";
-            packageId = "mime 0.3.17";
-          }
-          {
-            name = "pin-project-lite";
-            packageId = "pin-project-lite";
-          }
-          {
-            name = "rustversion";
-            packageId = "rustversion";
-          }
-          {
-            name = "sync_wrapper";
-            packageId = "sync_wrapper 1.0.2";
-          }
-          {
-            name = "tower-layer";
-            packageId = "tower-layer";
-          }
-          {
-            name = "tower-service";
-            packageId = "tower-service";
-          }
-          {
-            name = "tracing";
-            packageId = "tracing";
-            optional = true;
-            usesDefaultFeatures = false;
-          }
-        ];
-        devDependencies = [
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            usesDefaultFeatures = false;
-            features = [ "alloc" ];
-          }
-        ];
-        features = {
-          "__private_docs" = [ "dep:tower-http" ];
-          "tracing" = [ "dep:tracing" ];
-        };
-        resolvedDefaultFeatures = [ "tracing" ];
-      };
-      "axum-macros" = rec {
-        crateName = "axum-macros";
-        version = "0.4.2";
-        edition = "2021";
-        sha256 = "1klv77c889jm05bzayaaiinalarhvh2crc2w4nvp3l581xaj7lap";
-        procMacro = true;
-        libName = "axum_macros";
-        dependencies = [
-          {
-            name = "proc-macro2";
-            packageId = "proc-macro2";
-          }
-          {
-            name = "quote";
-            packageId = "quote";
-          }
-          {
-            name = "syn";
-            packageId = "syn 2.0.96";
-            features = [ "full" "parsing" "extra-traits" ];
-          }
-        ];
-        devDependencies = [
-          {
-            name = "syn";
-            packageId = "syn 2.0.96";
-            features = [ "full" "extra-traits" ];
-          }
-        ];
-        features = {
-          "__private" = [ "syn/visit-mut" ];
-        };
-        resolvedDefaultFeatures = [ "default" ];
-      };
-      "axum-test" = rec {
-        crateName = "axum-test";
-        version = "16.4.1";
-        edition = "2021";
-        sha256 = "1p5qxacvxsagnqq30nr2wznjyhgb8svsfb925ah3d2b0s91s9qv3";
-        libName = "axum_test";
-        authors = [
-          "Joseph Lenton <josephlenton@gmail.com>"
-        ];
-        dependencies = [
-          {
-            name = "anyhow";
-            packageId = "anyhow";
-          }
-          {
-            name = "assert-json-diff";
-            packageId = "assert-json-diff";
-          }
-          {
-            name = "auto-future";
-            packageId = "auto-future";
-          }
-          {
-            name = "axum";
-            packageId = "axum";
-          }
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "bytesize";
-            packageId = "bytesize";
-          }
-          {
-            name = "cookie";
-            packageId = "cookie 0.18.1";
-          }
-          {
-            name = "http";
-            packageId = "http 1.2.0";
-          }
-          {
-            name = "http-body-util";
-            packageId = "http-body-util";
-          }
-          {
-            name = "hyper";
-            packageId = "hyper 1.5.2";
-            features = [ "http1" ];
-          }
-          {
-            name = "hyper-util";
-            packageId = "hyper-util";
-            features = [ "client" "http1" "client-legacy" ];
-          }
-          {
-            name = "mime";
-            packageId = "mime 0.3.17";
-          }
-          {
-            name = "pretty_assertions";
-            packageId = "pretty_assertions";
-            optional = true;
-          }
-          {
-            name = "reserve-port";
-            packageId = "reserve-port";
-          }
-          {
-            name = "rust-multipart-rfc7578_2";
-            packageId = "rust-multipart-rfc7578_2";
-          }
-          {
-            name = "serde";
-            packageId = "serde 1.0.217";
-          }
-          {
-            name = "serde_json";
-            packageId = "serde_json";
-          }
-          {
-            name = "serde_urlencoded";
-            packageId = "serde_urlencoded";
-          }
-          {
-            name = "smallvec";
-            packageId = "smallvec";
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            features = [ "rt" ];
-          }
-          {
-            name = "tower";
-            packageId = "tower";
-            features = [ "util" "make" ];
-          }
-          {
-            name = "url";
-            packageId = "url 2.5.4";
-          }
-        ];
-        devDependencies = [
-          {
-            name = "axum";
-            packageId = "axum";
-            features = [ "multipart" "tokio" "ws" ];
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            features = [ "rt" "rt-multi-thread" "sync" "time" "macros" ];
-          }
-        ];
-        features = {
-          "all" = [ "pretty-assertions" "yaml" "msgpack" "reqwest" "shuttle" "typed-routing" "ws" ];
-          "default" = [ "pretty-assertions" ];
-          "msgpack" = [ "dep:rmp-serde" ];
-          "pretty-assertions" = [ "dep:pretty_assertions" ];
-          "reqwest" = [ "dep:reqwest" ];
-          "shuttle" = [ "dep:shuttle-axum" ];
-          "typed-routing" = [ "dep:axum-extra" ];
-          "ws" = [ "axum/ws" "tokio/time" "dep:uuid" "dep:base64" "dep:tokio-tungstenite" "dep:futures-util" ];
-          "yaml" = [ "dep:serde_yaml" ];
-        };
-        resolvedDefaultFeatures = [ "default" "pretty-assertions" ];
-      };
       "az" = rec {
         crateName = "az";
         version = "1.2.1";
@@ -2055,7 +1560,7 @@ rec {
         dependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             usesDefaultFeatures = false;
           }
@@ -2199,19 +1704,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" "std" ];
       };
-      "bytesize" = rec {
-        crateName = "bytesize";
-        version = "1.3.0";
-        edition = "2015";
-        sha256 = "1k3aak70iwz4s2gsjbxf0ws4xnixqbdz6p2ha96s06748fpniqx3";
-        authors = [
-          "Hyunsik Choi <hyunsik.choi@gmail.com>"
-        ];
-        features = {
-          "serde" = [ "dep:serde" ];
-        };
-        resolvedDefaultFeatures = [ "default" ];
-      };
       "cairo-rs" = rec {
         crateName = "cairo-rs";
         version = "0.18.5";
@@ -2434,7 +1926,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             usesDefaultFeatures = false;
           }
@@ -2494,7 +1986,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             usesDefaultFeatures = false;
           }
@@ -2615,10 +2107,10 @@ rec {
       };
       "clap" = rec {
         crateName = "clap";
-        version = "4.5.26";
+        version = "4.5.30";
         edition = "2021";
         crateBin = [];
-        sha256 = "10v7qvn90calfbhap1c4r249i5c7fbxj09fn3szfz9pkis85xsx8";
+        sha256 = "0vcyrn4ymq2gd56sl3xnfki8q8llg64sj3rj3qx33mgsf66v3dwj";
         dependencies = [
           {
             name = "clap_builder";
@@ -2647,6 +2139,7 @@ rec {
           "unicode" = [ "clap_builder/unicode" ];
           "unstable-doc" = [ "clap_builder/unstable-doc" "derive" ];
           "unstable-ext" = [ "clap_builder/unstable-ext" ];
+          "unstable-markdown" = [ "clap_derive/unstable-markdown" ];
           "unstable-styles" = [ "clap_builder/unstable-styles" ];
           "unstable-v5" = [ "clap_builder/unstable-v5" "clap_derive?/unstable-v5" "deprecated" ];
           "usage" = [ "clap_builder/usage" ];
@@ -2656,9 +2149,9 @@ rec {
       };
       "clap_builder" = rec {
         crateName = "clap_builder";
-        version = "4.5.26";
+        version = "4.5.30";
         edition = "2021";
-        sha256 = "08f1mzcvi7zjhm7hvz6al4jnv70ccqhwiaq74hihlspwnl0iic4n";
+        sha256 = "0369xis2ar46icsaxqyy37976mlb62alzyx4j53k99vq2w3v4pd3";
         dependencies = [
           {
             name = "anstream";
@@ -2695,9 +2188,9 @@ rec {
       };
       "clap_derive" = rec {
         crateName = "clap_derive";
-        version = "4.5.24";
+        version = "4.5.28";
         edition = "2021";
-        sha256 = "131ih3dm76srkbpfx7zfspp9b556zgzj31wqhl0ji2b39lcmbdsl";
+        sha256 = "1vgigkhljp3r8r5lwdrn1ij93nafmjwh8cx77nppb9plqsaysk5z";
         procMacro = true;
         dependencies = [
           {
@@ -2720,6 +2213,7 @@ rec {
         ];
         features = {
           "raw-deprecated" = [ "deprecated" ];
+          "unstable-markdown" = [ "dep:pulldown-cmark" "dep:anstyle" ];
           "unstable-v5" = [ "deprecated" ];
         };
         resolvedDefaultFeatures = [ "default" ];
@@ -2811,7 +2305,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -2887,7 +2381,7 @@ rec {
           "random" = [ "rand" ];
         };
       };
-      "cookie 0.17.0" = rec {
+      "cookie" = rec {
         crateName = "cookie";
         version = "0.17.0";
         edition = "2018";
@@ -2926,45 +2420,6 @@ rec {
           "subtle" = [ "dep:subtle" ];
         };
       };
-      "cookie 0.18.1" = rec {
-        crateName = "cookie";
-        version = "0.18.1";
-        edition = "2018";
-        sha256 = "0iy749flficrlvgr3hjmf3igr738lk81n5akzf4ym4cs6cxg7pjd";
-        authors = [
-          "Sergio Benitez <sb@sergio.bz>"
-          "Alex Crichton <alex@alexcrichton.com>"
-        ];
-        dependencies = [
-          {
-            name = "time";
-            packageId = "time 0.3.37";
-            usesDefaultFeatures = false;
-            features = [ "std" "parsing" "formatting" "macros" ];
-          }
-        ];
-        buildDependencies = [
-          {
-            name = "version_check";
-            packageId = "version_check 0.9.5";
-          }
-        ];
-        features = {
-          "aes-gcm" = [ "dep:aes-gcm" ];
-          "base64" = [ "dep:base64" ];
-          "hkdf" = [ "dep:hkdf" ];
-          "hmac" = [ "dep:hmac" ];
-          "key-expansion" = [ "sha2" "hkdf" ];
-          "percent-encode" = [ "percent-encoding" ];
-          "percent-encoding" = [ "dep:percent-encoding" ];
-          "private" = [ "aes-gcm" "base64" "rand" "subtle" ];
-          "rand" = [ "dep:rand" ];
-          "secure" = [ "private" "signed" "key-expansion" ];
-          "sha2" = [ "dep:sha2" ];
-          "signed" = [ "hmac" "sha2" "base64" "rand" "subtle" ];
-          "subtle" = [ "dep:subtle" ];
-        };
-      };
       "cookie-factory" = rec {
         crateName = "cookie-factory";
         version = "0.3.3";
@@ -3325,7 +2780,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -3495,7 +2950,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -3626,16 +3081,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "alloc" "powerfmt" "std" ];
       };
-      "diff" = rec {
-        crateName = "diff";
-        version = "0.1.13";
-        edition = "2015";
-        sha256 = "1j0nzjxci2zqx63hdcihkp0a4dkdmzxd7my4m7zk6cjyfy34j9an";
-        authors = [
-          "Utkarsh Kukreti <utkarshkukreti@gmail.com>"
-        ];
-
-      };
       "digest" = rec {
         crateName = "digest";
         version = "0.10.7";
@@ -3702,7 +3147,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
           }
           {
@@ -3742,7 +3187,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
           }
           {
@@ -3821,7 +3266,7 @@ rec {
         dependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             features = [ "derive" ];
           }
@@ -3857,7 +3302,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
           }
           {
             name = "serde_derive";
@@ -4145,33 +3590,6 @@ rec {
         ];
 
       };
-      "fallible-iterator" = rec {
-        crateName = "fallible-iterator";
-        version = "0.3.0";
-        edition = "2018";
-        sha256 = "0ja6l56yka5vn4y4pk6hn88z0bpny7a8k1919aqjzp0j1yhy9k1a";
-        libName = "fallible_iterator";
-        authors = [
-          "Steven Fackler <sfackler@gmail.com>"
-        ];
-        features = {
-          "default" = [ "alloc" ];
-          "std" = [ "alloc" ];
-        };
-        resolvedDefaultFeatures = [ "alloc" "default" ];
-      };
-      "fallible-streaming-iterator" = rec {
-        crateName = "fallible-streaming-iterator";
-        version = "0.1.9";
-        edition = "2015";
-        sha256 = "0nj6j26p71bjy8h42x6jahx1hn0ng6mc2miwpgwnp8vnwqf4jq3k";
-        libName = "fallible_streaming_iterator";
-        authors = [
-          "Steven Fackler <sfackler@gmail.com>"
-        ];
-        features = {
-        };
-      };
       "fastrand" = rec {
         crateName = "fastrand";
         version = "2.3.0";
@@ -4277,7 +3695,7 @@ rec {
           }
           {
             name = "cookie";
-            packageId = "cookie 0.17.0";
+            packageId = "cookie";
           }
           {
             name = "futures-util";
@@ -4317,7 +3735,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -4841,7 +4259,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -5481,7 +4899,7 @@ rec {
         libName = "geo_types";
 
       };
-      "getrandom" = rec {
+      "getrandom 0.2.15" = rec {
         crateName = "getrandom";
         version = "0.2.15";
         edition = "2018";
@@ -5517,6 +4935,84 @@ rec {
         };
         resolvedDefaultFeatures = [ "std" ];
       };
+      "getrandom 0.3.1" = rec {
+        crateName = "getrandom";
+        version = "0.3.1";
+        edition = "2021";
+        sha256 = "1y154yzby383p63ndw6zpfm0fz3vf6c0zdwc7df6vkl150wrr923";
+        authors = [
+          "The Rand Project Developers"
+        ];
+        dependencies = [
+          {
+            name = "cfg-if";
+            packageId = "cfg-if";
+          }
+          {
+            name = "libc";
+            packageId = "libc";
+            usesDefaultFeatures = false;
+            target = { target, features }: ((("linux" == target."os") || ("android" == target."os")) && (!(("custom" == target."getrandom_backend") || ("rdrand" == target."getrandom_backend") || ("rndr" == target."getrandom_backend"))));
+          }
+          {
+            name = "libc";
+            packageId = "libc";
+            usesDefaultFeatures = false;
+            target = { target, features }: (("dragonfly" == target."os") || ("freebsd" == target."os") || ("hurd" == target."os") || ("illumos" == target."os") || (("horizon" == target."os") && ("arm" == target."arch")));
+          }
+          {
+            name = "libc";
+            packageId = "libc";
+            usesDefaultFeatures = false;
+            target = { target, features }: (("haiku" == target."os") || ("redox" == target."os") || ("nto" == target."os") || ("aix" == target."os"));
+          }
+          {
+            name = "libc";
+            packageId = "libc";
+            usesDefaultFeatures = false;
+            target = { target, features }: (("ios" == target."os") || ("visionos" == target."os") || ("watchos" == target."os") || ("tvos" == target."os"));
+          }
+          {
+            name = "libc";
+            packageId = "libc";
+            usesDefaultFeatures = false;
+            target = { target, features }: (("macos" == target."os") || ("openbsd" == target."os") || ("vita" == target."os") || ("emscripten" == target."os"));
+          }
+          {
+            name = "libc";
+            packageId = "libc";
+            usesDefaultFeatures = false;
+            target = { target, features }: ("netbsd" == target."os");
+          }
+          {
+            name = "libc";
+            packageId = "libc";
+            usesDefaultFeatures = false;
+            target = { target, features }: ("solaris" == target."os");
+          }
+          {
+            name = "libc";
+            packageId = "libc";
+            usesDefaultFeatures = false;
+            target = { target, features }: ("vxworks" == target."os");
+          }
+          {
+            name = "wasi";
+            packageId = "wasi 0.13.3+wasi-0.2.2";
+            usesDefaultFeatures = false;
+            target = { target, features }: (("wasm32" == target."arch") && ("wasi" == target."os") && ("p2" == target."env"));
+          }
+          {
+            name = "windows-targets";
+            packageId = "windows-targets 0.52.6";
+            target = { target, features }: ((target."windows" or false) && (!("win7" == target."vendor")));
+          }
+        ];
+        features = {
+          "rustc-dep-of-std" = [ "dep:compiler_builtins" "dep:core" ];
+          "wasm_js" = [ "dep:wasm-bindgen" "dep:js-sys" ];
+        };
+      };
       "gif 0.11.4" = rec {
         crateName = "gif";
         version = "0.11.4";
@@ -5868,7 +5364,7 @@ rec {
           }
           {
             name = "proc-macro-crate";
-            packageId = "proc-macro-crate 2.0.2";
+            packageId = "proc-macro-crate 2.0.0";
           }
           {
             name = "proc-macro-error";
@@ -6027,7 +5523,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
           }
           {
             name = "serde_json";
@@ -6075,7 +5571,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "alloc" "derive" ];
           }
           {
@@ -6642,39 +6138,7 @@ rec {
         };
         resolvedDefaultFeatures = [ "alloc" "default" "std" ];
       };
-      "hashbrown 0.14.5" = rec {
-        crateName = "hashbrown";
-        version = "0.14.5";
-        edition = "2021";
-        sha256 = "1wa1vy1xs3mp11bn3z9dv0jricgr6a2j0zkf1g19yz3vw4il89z5";
-        authors = [
-          "Amanieu d'Antras <amanieu@gmail.com>"
-        ];
-        dependencies = [
-          {
-            name = "ahash";
-            packageId = "ahash";
-            optional = true;
-            usesDefaultFeatures = false;
-          }
-        ];
-        features = {
-          "ahash" = [ "dep:ahash" ];
-          "alloc" = [ "dep:alloc" ];
-          "allocator-api2" = [ "dep:allocator-api2" ];
-          "compiler_builtins" = [ "dep:compiler_builtins" ];
-          "core" = [ "dep:core" ];
-          "default" = [ "ahash" "inline-more" "allocator-api2" ];
-          "equivalent" = [ "dep:equivalent" ];
-          "nightly" = [ "allocator-api2?/nightly" "bumpalo/allocator_api" ];
-          "rayon" = [ "dep:rayon" ];
-          "rkyv" = [ "dep:rkyv" ];
-          "rustc-dep-of-std" = [ "nightly" "core" "compiler_builtins" "alloc" "rustc-internal-api" ];
-          "serde" = [ "dep:serde" ];
-        };
-        resolvedDefaultFeatures = [ "ahash" "inline-more" ];
-      };
-      "hashbrown 0.15.2" = rec {
+      "hashbrown" = rec {
         crateName = "hashbrown";
         version = "0.15.2";
         edition = "2021";
@@ -6718,7 +6182,7 @@ rec {
         };
         resolvedDefaultFeatures = [ "allocator-api2" "default" "default-hasher" "equivalent" "inline-more" "raw-entry" ];
       };
-      "hashlink 0.10.0" = rec {
+      "hashlink" = rec {
         crateName = "hashlink";
         version = "0.10.0";
         edition = "2018";
@@ -6729,7 +6193,7 @@ rec {
         dependencies = [
           {
             name = "hashbrown";
-            packageId = "hashbrown 0.15.2";
+            packageId = "hashbrown";
             usesDefaultFeatures = false;
             features = [ "default-hasher" "inline-more" ];
           }
@@ -6739,27 +6203,6 @@ rec {
           "serde_impl" = [ "serde" ];
         };
       };
-      "hashlink 0.9.1" = rec {
-        crateName = "hashlink";
-        version = "0.9.1";
-        edition = "2018";
-        sha256 = "1byq4nyrflm5s6wdx5qwp96l1qbp2d0nljvrr5yqrsfy51qzz93b";
-        authors = [
-          "kyren <kerriganw@gmail.com>"
-        ];
-        dependencies = [
-          {
-            name = "hashbrown";
-            packageId = "hashbrown 0.14.5";
-            usesDefaultFeatures = false;
-            features = [ "ahash" "inline-more" ];
-          }
-        ];
-        features = {
-          "serde" = [ "dep:serde" ];
-          "serde_impl" = [ "serde" ];
-        };
-      };
       "headers" = rec {
         crateName = "headers";
         version = "0.3.9";
@@ -7073,7 +6516,7 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" "std" ];
       };
-      "http-body 0.4.6" = rec {
+      "http-body" = rec {
         crateName = "http-body";
         version = "0.4.6";
         edition = "2018";
@@ -7099,65 +6542,6 @@ rec {
           }
         ];
 
-      };
-      "http-body 1.0.1" = rec {
-        crateName = "http-body";
-        version = "1.0.1";
-        edition = "2018";
-        sha256 = "111ir5k2b9ihz5nr9cz7cwm7fnydca7dx4hc7vr16scfzghxrzhy";
-        libName = "http_body";
-        authors = [
-          "Carl Lerche <me@carllerche.com>"
-          "Lucio Franco <luciofranco14@gmail.com>"
-          "Sean McArthur <sean@seanmonstar.com>"
-        ];
-        dependencies = [
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "http";
-            packageId = "http 1.2.0";
-          }
-        ];
-
-      };
-      "http-body-util" = rec {
-        crateName = "http-body-util";
-        version = "0.1.2";
-        edition = "2018";
-        sha256 = "0kslwazg4400qnc2azkrgqqci0fppv12waicnsy5d8hncvbjjd3r";
-        libName = "http_body_util";
-        authors = [
-          "Carl Lerche <me@carllerche.com>"
-          "Lucio Franco <luciofranco14@gmail.com>"
-          "Sean McArthur <sean@seanmonstar.com>"
-        ];
-        dependencies = [
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            usesDefaultFeatures = false;
-          }
-          {
-            name = "http";
-            packageId = "http 1.2.0";
-          }
-          {
-            name = "http-body";
-            packageId = "http-body 1.0.1";
-          }
-          {
-            name = "pin-project-lite";
-            packageId = "pin-project-lite";
-          }
-        ];
-
       };
       "httparse" = rec {
         crateName = "httparse";
@@ -7288,7 +6672,7 @@ rec {
           }
           {
             name = "http-body";
-            packageId = "http-body 0.4.6";
+            packageId = "http-body";
           }
           {
             name = "httparse";
@@ -7357,108 +6741,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "client" "default" "h2" "http1" "http2" "runtime" "server" "socket2" "stream" "tcp" ];
       };
-      "hyper 1.5.2" = rec {
-        crateName = "hyper";
-        version = "1.5.2";
-        edition = "2021";
-        sha256 = "1q7akfb443yrjzkmnnbp2vs8zi15hgbk466rr4y144v4ppabhvr5";
-        authors = [
-          "Sean McArthur <sean@seanmonstar.com>"
-        ];
-        dependencies = [
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "futures-channel";
-            packageId = "futures-channel";
-            optional = true;
-          }
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            optional = true;
-            usesDefaultFeatures = false;
-          }
-          {
-            name = "http";
-            packageId = "http 1.2.0";
-          }
-          {
-            name = "http-body";
-            packageId = "http-body 1.0.1";
-          }
-          {
-            name = "httparse";
-            packageId = "httparse";
-            optional = true;
-          }
-          {
-            name = "httpdate";
-            packageId = "httpdate";
-            optional = true;
-          }
-          {
-            name = "itoa";
-            packageId = "itoa";
-            optional = true;
-          }
-          {
-            name = "pin-project-lite";
-            packageId = "pin-project-lite";
-            optional = true;
-          }
-          {
-            name = "smallvec";
-            packageId = "smallvec";
-            optional = true;
-            features = [ "const_generics" "const_new" ];
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            features = [ "sync" ];
-          }
-          {
-            name = "want";
-            packageId = "want";
-            optional = true;
-          }
-        ];
-        devDependencies = [
-          {
-            name = "futures-channel";
-            packageId = "futures-channel";
-            features = [ "sink" ];
-          }
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            usesDefaultFeatures = false;
-            features = [ "alloc" "sink" ];
-          }
-          {
-            name = "pin-project-lite";
-            packageId = "pin-project-lite";
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            features = [ "fs" "macros" "net" "io-std" "io-util" "rt" "rt-multi-thread" "sync" "time" "test-util" ];
-          }
-        ];
-        features = {
-          "client" = [ "dep:want" "dep:pin-project-lite" "dep:smallvec" ];
-          "ffi" = [ "dep:http-body-util" "futures-util?/alloc" ];
-          "full" = [ "client" "http1" "http2" "server" ];
-          "http1" = [ "dep:futures-channel" "dep:futures-util" "dep:httparse" "dep:itoa" ];
-          "http2" = [ "dep:futures-channel" "dep:futures-util" "dep:h2" ];
-          "server" = [ "dep:httpdate" "dep:pin-project-lite" "dep:smallvec" ];
-          "tracing" = [ "dep:tracing" ];
-        };
-        resolvedDefaultFeatures = [ "client" "default" "http1" "server" ];
-      };
       "hyper-tls" = rec {
         crateName = "hyper-tls";
         version = "0.5.0";
@@ -7509,101 +6791,6 @@ rec {
           "vendored" = [ "native-tls/vendored" ];
         };
       };
-      "hyper-util" = rec {
-        crateName = "hyper-util";
-        version = "0.1.10";
-        edition = "2021";
-        sha256 = "1d1iwrkysjhq63pg54zk3vfby1j7zmxzm9zzyfr4lwvp0szcybfz";
-        libName = "hyper_util";
-        authors = [
-          "Sean McArthur <sean@seanmonstar.com>"
-        ];
-        dependencies = [
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "futures-channel";
-            packageId = "futures-channel";
-            optional = true;
-          }
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            usesDefaultFeatures = false;
-          }
-          {
-            name = "http";
-            packageId = "http 1.2.0";
-          }
-          {
-            name = "http-body";
-            packageId = "http-body 1.0.1";
-          }
-          {
-            name = "hyper";
-            packageId = "hyper 1.5.2";
-          }
-          {
-            name = "pin-project-lite";
-            packageId = "pin-project-lite";
-          }
-          {
-            name = "socket2";
-            packageId = "socket2";
-            optional = true;
-            features = [ "all" ];
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            optional = true;
-            usesDefaultFeatures = false;
-          }
-          {
-            name = "tower-service";
-            packageId = "tower-service";
-            optional = true;
-          }
-          {
-            name = "tracing";
-            packageId = "tracing";
-            optional = true;
-            usesDefaultFeatures = false;
-            features = [ "std" ];
-          }
-        ];
-        devDependencies = [
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "hyper";
-            packageId = "hyper 1.5.2";
-            features = [ "full" ];
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            features = [ "macros" "test-util" "signal" ];
-          }
-        ];
-        features = {
-          "client" = [ "hyper/client" "dep:tracing" "dep:futures-channel" "dep:tower-service" ];
-          "client-legacy" = [ "client" "dep:socket2" "tokio/sync" ];
-          "full" = [ "client" "client-legacy" "server" "server-auto" "server-graceful" "service" "http1" "http2" "tokio" ];
-          "http1" = [ "hyper/http1" ];
-          "http2" = [ "hyper/http2" ];
-          "server" = [ "hyper/server" ];
-          "server-auto" = [ "server" "http1" "http2" ];
-          "server-graceful" = [ "server" "tokio/sync" "futures-util/alloc" ];
-          "service" = [ "dep:tower-service" ];
-          "tokio" = [ "dep:tokio" "tokio/net" "tokio/rt" "tokio/time" ];
-        };
-        resolvedDefaultFeatures = [ "client" "client-legacy" "default" "http1" "server" "service" "tokio" ];
-      };
       "iana-time-zone" = rec {
         crateName = "iana-time-zone";
         version = "0.1.61";
@@ -7766,6 +6953,13 @@ rec {
             usesDefaultFeatures = false;
             features = [ "alloc" ];
           }
+          {
+            name = "serde";
+            packageId = "serde 1.0.218";
+            optional = true;
+            usesDefaultFeatures = false;
+            features = [ "alloc" "derive" ];
+          }
           {
             name = "tinystr";
             packageId = "tinystr";
@@ -7784,13 +6978,21 @@ rec {
             usesDefaultFeatures = false;
           }
         ];
+        devDependencies = [
+          {
+            name = "serde";
+            packageId = "serde 1.0.218";
+            usesDefaultFeatures = false;
+            features = [ "derive" ];
+          }
+        ];
         features = {
           "bench" = [ "serde" ];
           "databake" = [ "dep:databake" ];
           "serde" = [ "dep:serde" "tinystr/serde" ];
           "zerovec" = [ "dep:zerovec" ];
         };
-        resolvedDefaultFeatures = [ "zerovec" ];
+        resolvedDefaultFeatures = [ "serde" "zerovec" ];
       };
       "icu_locid_transform" = rec {
         crateName = "icu_locid_transform";
@@ -8369,49 +7571,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "bmp" "dds" "default" "dxt" "exr" "farbfeld" "gif" "hdr" "ico" "jpeg" "jpeg_rayon" "openexr" "png" "pnm" "qoi" "tga" "tiff" "webp" ];
       };
-      "include_dir" = rec {
-        crateName = "include_dir";
-        version = "0.7.4";
-        edition = "2021";
-        sha256 = "1pfh3g45z88kwq93skng0n6g3r7zkhq9ldqs9y8rvr7i11s12gcj";
-        authors = [
-          "Michael Bryan <michaelfbryan@gmail.com>"
-        ];
-        dependencies = [
-          {
-            name = "include_dir_macros";
-            packageId = "include_dir_macros";
-          }
-        ];
-        features = {
-          "glob" = [ "dep:glob" ];
-          "metadata" = [ "include_dir_macros/metadata" ];
-          "nightly" = [ "include_dir_macros/nightly" ];
-        };
-        resolvedDefaultFeatures = [ "default" ];
-      };
-      "include_dir_macros" = rec {
-        crateName = "include_dir_macros";
-        version = "0.7.4";
-        edition = "2021";
-        sha256 = "0x8smnf6knd86g69p19z5lpfsaqp8w0nx14kdpkz1m8bxnkqbavw";
-        procMacro = true;
-        authors = [
-          "Michael Bryan <michaelfbryan@gmail.com>"
-        ];
-        dependencies = [
-          {
-            name = "proc-macro2";
-            packageId = "proc-macro2";
-          }
-          {
-            name = "quote";
-            packageId = "quote";
-          }
-        ];
-        features = {
-        };
-      };
       "indent_write" = rec {
         crateName = "indent_write";
         version = "2.2.0";
@@ -8437,7 +7596,7 @@ rec {
           }
           {
             name = "hashbrown";
-            packageId = "hashbrown 0.15.2";
+            packageId = "hashbrown";
             usesDefaultFeatures = false;
           }
         ];
@@ -8732,6 +7891,67 @@ rec {
           }
         ];
 
+      };
+      "l10n-db" = rec {
+        crateName = "l10n-db";
+        version = "0.1.0";
+        edition = "2021";
+        crateBin = [
+          {
+            name = "main";
+            path = "src/bin/main.rs";
+            requiredFeatures = [ ];
+          }
+        ];
+        # We can't filter paths with references in Nix 2.4
+        # See https://github.com/NixOS/nix/issues/5410
+        src = if ((lib.versionOlder builtins.nixVersion "2.4pre20211007") || (lib.versionOlder "2.5" builtins.nixVersion ))
+          then lib.cleanSourceWith { filter = sourceFilter;  src = ./l10n-db; }
+          else ./l10n-db;
+        libName = "l10n_db";
+        dependencies = [
+          {
+            name = "chrono";
+            packageId = "chrono";
+            features = [ "serde" ];
+          }
+          {
+            name = "clap";
+            packageId = "clap";
+            features = [ "derive" ];
+          }
+          {
+            name = "icu_locid";
+            packageId = "icu_locid";
+            features = [ "serde" ];
+          }
+          {
+            name = "serde";
+            packageId = "serde 1.0.218";
+            features = [ "derive" ];
+          }
+          {
+            name = "serde_json";
+            packageId = "serde_json";
+          }
+          {
+            name = "tempfile";
+            packageId = "tempfile";
+          }
+          {
+            name = "thiserror";
+            packageId = "thiserror 2.0.11";
+          }
+          {
+            name = "toml";
+            packageId = "toml";
+          }
+          {
+            name = "xml-rs";
+            packageId = "xml-rs";
+          }
+        ];
+
       };
       "language-tags" = rec {
         crateName = "language-tags";
@@ -9090,7 +8310,7 @@ rec {
           "syn" = [ "dep:syn" ];
           "vcpkg" = [ "dep:vcpkg" ];
         };
-        resolvedDefaultFeatures = [ "bundled" "bundled_bindings" "cc" "default" "min_sqlite_version_3_14_0" "pkg-config" "unlock_notify" "vcpkg" ];
+        resolvedDefaultFeatures = [ "bundled" "bundled_bindings" "cc" "pkg-config" "unlock_notify" "vcpkg" ];
       };
       "libyml" = rec {
         crateName = "libyml";
@@ -9308,18 +8528,6 @@ rec {
         libPath = "lib.rs";
 
       };
-      "matchit" = rec {
-        crateName = "matchit";
-        version = "0.7.3";
-        edition = "2021";
-        sha256 = "156bgdmmlv4crib31qhgg49nsjk88dxkdqp80ha2pk2rk6n6ax0f";
-        authors = [
-          "Ibraheem Ahmed <ibraheem@ibraheem.ca>"
-        ];
-        features = {
-        };
-        resolvedDefaultFeatures = [ "default" ];
-      };
       "md-5" = rec {
         crateName = "md-5";
         version = "0.10.6";
@@ -9412,7 +8620,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
           }
           {
             name = "serde_derive";
@@ -10383,7 +9591,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -11349,31 +10557,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "simd" "std" ];
       };
-      "pretty_assertions" = rec {
-        crateName = "pretty_assertions";
-        version = "1.4.1";
-        edition = "2018";
-        sha256 = "0v8iq35ca4rw3rza5is3wjxwsf88303ivys07anc5yviybi31q9s";
-        authors = [
-          "Colin Kiegel <kiegel@gmx.de>"
-          "Florent Fayolle <florent.fayolle69@gmail.com>"
-          "Tom Milligan <code@tommilligan.net>"
-        ];
-        dependencies = [
-          {
-            name = "diff";
-            packageId = "diff";
-          }
-          {
-            name = "yansi";
-            packageId = "yansi";
-          }
-        ];
-        features = {
-          "default" = [ "std" ];
-        };
-        resolvedDefaultFeatures = [ "default" "std" ];
-      };
       "pretty_env_logger" = rec {
         crateName = "pretty_env_logger";
         version = "0.5.0";
@@ -11415,20 +10598,16 @@ rec {
         ];
 
       };
-      "proc-macro-crate 2.0.2" = rec {
+      "proc-macro-crate 2.0.0" = rec {
         crateName = "proc-macro-crate";
-        version = "2.0.2";
+        version = "2.0.0";
         edition = "2021";
-        sha256 = "092x5acqnic14cw6vacqap5kgknq3jn4c6jij9zi6j85839jc3xh";
+        sha256 = "1s23imns07vmacn2xjd5hv2h6rr94iqq3fd2frwa6i4h2nk6d0vy";
         libName = "proc_macro_crate";
         authors = [
           "Bastian Köcher <git@kchr.de>"
         ];
         dependencies = [
-          {
-            name = "toml_datetime";
-            packageId = "toml_datetime";
-          }
           {
             name = "toml_edit";
             packageId = "toml_edit 0.20.2";
@@ -11962,7 +11141,7 @@ rec {
         dependencies = [
           {
             name = "getrandom";
-            packageId = "getrandom";
+            packageId = "getrandom 0.2.15";
             optional = true;
           }
         ];
@@ -12447,7 +11626,7 @@ rec {
           }
           {
             name = "http-body";
-            packageId = "http-body 0.4.6";
+            packageId = "http-body";
             target = { target, features }: (!("wasm32" == target."arch"));
           }
           {
@@ -12513,7 +11692,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
           }
           {
             name = "serde_json";
@@ -12531,7 +11710,7 @@ rec {
           }
           {
             name = "sync_wrapper";
-            packageId = "sync_wrapper 0.1.2";
+            packageId = "sync_wrapper";
           }
           {
             name = "system-configuration";
@@ -12598,7 +11777,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             target = {target, features}: (!("wasm32" == target."arch"));
             features = [ "derive" ];
           }
@@ -12664,27 +11843,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "__tls" "default" "default-tls" "hyper-tls" "json" "native-tls-crate" "serde_json" "tokio-native-tls" ];
       };
-      "reserve-port" = rec {
-        crateName = "reserve-port";
-        version = "2.0.1";
-        edition = "2021";
-        sha256 = "10x21rdb1hjzp6n5flbbw3hfd7brmirckz1q0zsf3a7s5d516f4q";
-        libName = "reserve_port";
-        authors = [
-          "Joseph Lenton <josephlenton@gmail.com>"
-        ];
-        dependencies = [
-          {
-            name = "lazy_static";
-            packageId = "lazy_static";
-          }
-          {
-            name = "thiserror";
-            packageId = "thiserror 1.0.69";
-          }
-        ];
-
-      };
       "result-extended" = rec {
         crateName = "result-extended";
         version = "0.1.0";
@@ -12803,158 +11961,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" "pem" "std" "u64_digit" ];
       };
-      "rusqlite" = rec {
-        crateName = "rusqlite";
-        version = "0.32.1";
-        edition = "2021";
-        sha256 = "0vlx040bppl414pbjgbp7qr4jdxwszi9krx0m63zzf2f2whvflvp";
-        authors = [
-          "The rusqlite developers"
-        ];
-        dependencies = [
-          {
-            name = "bitflags";
-            packageId = "bitflags 2.8.0";
-          }
-          {
-            name = "fallible-iterator";
-            packageId = "fallible-iterator";
-          }
-          {
-            name = "fallible-streaming-iterator";
-            packageId = "fallible-streaming-iterator";
-          }
-          {
-            name = "hashlink";
-            packageId = "hashlink 0.9.1";
-          }
-          {
-            name = "libsqlite3-sys";
-            packageId = "libsqlite3-sys";
-          }
-          {
-            name = "smallvec";
-            packageId = "smallvec";
-          }
-        ];
-        features = {
-          "array" = [ "vtab" ];
-          "buildtime_bindgen" = [ "libsqlite3-sys/buildtime_bindgen" ];
-          "bundled" = [ "libsqlite3-sys/bundled" "modern_sqlite" ];
-          "bundled-full" = [ "modern-full" "bundled" ];
-          "bundled-sqlcipher" = [ "libsqlite3-sys/bundled-sqlcipher" "bundled" ];
-          "bundled-sqlcipher-vendored-openssl" = [ "libsqlite3-sys/bundled-sqlcipher-vendored-openssl" "bundled-sqlcipher" ];
-          "bundled-windows" = [ "libsqlite3-sys/bundled-windows" ];
-          "chrono" = [ "dep:chrono" ];
-          "csv" = [ "dep:csv" ];
-          "csvtab" = [ "csv" "vtab" ];
-          "in_gecko" = [ "modern_sqlite" "libsqlite3-sys/in_gecko" ];
-          "loadable_extension" = [ "libsqlite3-sys/loadable_extension" ];
-          "modern-full" = [ "array" "backup" "blob" "modern_sqlite" "chrono" "collation" "column_decltype" "csvtab" "extra_check" "functions" "hooks" "i128_blob" "limits" "load_extension" "serde_json" "series" "time" "trace" "unlock_notify" "url" "uuid" "vtab" "window" ];
-          "modern_sqlite" = [ "libsqlite3-sys/bundled_bindings" ];
-          "preupdate_hook" = [ "libsqlite3-sys/preupdate_hook" "hooks" ];
-          "rusqlite-macros" = [ "dep:rusqlite-macros" ];
-          "serde_json" = [ "dep:serde_json" ];
-          "serialize" = [ "modern_sqlite" ];
-          "series" = [ "vtab" ];
-          "session" = [ "libsqlite3-sys/session" "hooks" ];
-          "sqlcipher" = [ "libsqlite3-sys/sqlcipher" ];
-          "time" = [ "dep:time" ];
-          "unlock_notify" = [ "libsqlite3-sys/unlock_notify" ];
-          "url" = [ "dep:url" ];
-          "uuid" = [ "dep:uuid" ];
-          "wasm32-wasi-vfs" = [ "libsqlite3-sys/wasm32-wasi-vfs" ];
-          "window" = [ "functions" ];
-          "with-asan" = [ "libsqlite3-sys/with-asan" ];
-        };
-      };
-      "rusqlite_migration" = rec {
-        crateName = "rusqlite_migration";
-        version = "1.3.1";
-        edition = "2021";
-        sha256 = "076dm65g0sngzrb93r07va4l5zl3gjx9gq5mlsh21p7p0bl44fwj";
-        authors = [
-          "Clément Joly <foss@131719.xyz>"
-        ];
-        dependencies = [
-          {
-            name = "include_dir";
-            packageId = "include_dir";
-            optional = true;
-          }
-          {
-            name = "log";
-            packageId = "log 0.4.25";
-          }
-          {
-            name = "rusqlite";
-            packageId = "rusqlite";
-            usesDefaultFeatures = false;
-          }
-        ];
-        features = {
-          "alpha-async-tokio-rusqlite" = [ "dep:tokio-rusqlite" "dep:tokio" ];
-          "from-directory" = [ "dep:include_dir" ];
-        };
-        resolvedDefaultFeatures = [ "default" "from-directory" ];
-      };
-      "rust-multipart-rfc7578_2" = rec {
-        crateName = "rust-multipart-rfc7578_2";
-        version = "0.6.1";
-        edition = "2021";
-        sha256 = "0mwd3i2mk91n6diaxnkw28vyjbifhrm5ls73pcpfzz8a1i0lidq3";
-        libName = "rust_multipart_rfc7578_2";
-        authors = [
-          "Joseph Lenton <josephlenton@gmail.com>"
-          "Ferris Tseng <ferristseng@fastmail.fm>"
-        ];
-        dependencies = [
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "futures-core";
-            packageId = "futures-core";
-          }
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            usesDefaultFeatures = false;
-            features = [ "io" ];
-          }
-          {
-            name = "http";
-            packageId = "http 0.2.12";
-          }
-          {
-            name = "mime";
-            packageId = "mime 0.3.17";
-          }
-          {
-            name = "mime_guess";
-            packageId = "mime_guess 2.0.5";
-          }
-          {
-            name = "rand";
-            packageId = "rand 0.8.5";
-            features = [ "small_rng" ];
-          }
-          {
-            name = "thiserror";
-            packageId = "thiserror 1.0.69";
-          }
-        ];
-        devDependencies = [
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            usesDefaultFeatures = false;
-            features = [ "std" ];
-          }
-        ];
-
-      };
       "rustc-demangle" = rec {
         crateName = "rustc-demangle";
         version = "0.1.24";
@@ -13422,11 +12428,11 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" "std" ];
       };
-      "serde 1.0.217" = rec {
+      "serde 1.0.218" = rec {
         crateName = "serde";
-        version = "1.0.217";
+        version = "1.0.218";
         edition = "2018";
-        sha256 = "0w2ck1p1ajmrv1cf51qf7igjn2nc51r0izzc00fzmmhkvxjl5z02";
+        sha256 = "0q6z4bnrwagnms0bds4886711l6mc68s979i49zd3xnvkg8wkpz8";
         authors = [
           "Erick Tryzelaar <erick.tryzelaar@gmail.com>"
           "David Tolnay <dtolnay@gmail.com>"
@@ -13458,9 +12464,9 @@ rec {
       };
       "serde_derive" = rec {
         crateName = "serde_derive";
-        version = "1.0.217";
+        version = "1.0.218";
         edition = "2015";
-        sha256 = "180r3rj5gi5s1m23q66cr5wlfgc5jrs6n1mdmql2njnhk37zg6ss";
+        sha256 = "0azqd74xbpb1v5vf6w1fdbgmwp39ljjfj25cib5rgrzlj7hh75gh";
         procMacro = true;
         authors = [
           "Erick Tryzelaar <erick.tryzelaar@gmail.com>"
@@ -13492,9 +12498,9 @@ rec {
       };
       "serde_json" = rec {
         crateName = "serde_json";
-        version = "1.0.136";
+        version = "1.0.139";
         edition = "2021";
-        sha256 = "1lipcjhh1zazh283i4wsl4l14knh81q2rlkwmag8v8s2rwihqsik";
+        sha256 = "19kj3irpa22a7djz1jaf4wambzh7psiqa6zyafqnb76crhx6ry24";
         authors = [
           "Erick Tryzelaar <erick.tryzelaar@gmail.com>"
           "David Tolnay <dtolnay@gmail.com>"
@@ -13515,14 +12521,14 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             usesDefaultFeatures = false;
           }
         ];
         devDependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
         ];
@@ -13535,26 +12541,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" "raw_value" "std" ];
       };
-      "serde_path_to_error" = rec {
-        crateName = "serde_path_to_error";
-        version = "0.1.16";
-        edition = "2021";
-        sha256 = "19hlz2359l37ifirskpcds7sxg0gzpqvfilibs7whdys0128i6dg";
-        authors = [
-          "David Tolnay <dtolnay@gmail.com>"
-        ];
-        dependencies = [
-          {
-            name = "itoa";
-            packageId = "itoa";
-          }
-          {
-            name = "serde";
-            packageId = "serde 1.0.217";
-          }
-        ];
-
-      };
       "serde_spanned" = rec {
         crateName = "serde_spanned";
         version = "0.6.8";
@@ -13563,14 +12549,14 @@ rec {
         dependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
           }
         ];
         devDependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
           }
         ];
         features = {
@@ -13601,7 +12587,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
           }
         ];
 
@@ -13638,7 +12624,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
           }
         ];
         buildDependencies = [
@@ -13650,7 +12636,7 @@ rec {
         devDependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
         ];
@@ -13658,6 +12644,24 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" ];
       };
+      "server" = rec {
+        crateName = "server";
+        version = "0.1.0";
+        edition = "2021";
+        crateBin = [
+          {
+            name = "server";
+            path = "src/main.rs";
+            requiredFeatures = [ ];
+          }
+        ];
+        # We can't filter paths with references in Nix 2.4
+        # See https://github.com/NixOS/nix/issues/5410
+        src = if ((lib.versionOlder builtins.nixVersion "2.4pre20211007") || (lib.versionOlder "2.5" builtins.nixVersion ))
+          then lib.cleanSourceWith { filter = sourceFilter;  src = ./visions/server; }
+          else ./visions/server;
+
+      };
       "sgf" = rec {
         crateName = "sgf";
         version = "0.1.0";
@@ -13690,7 +12694,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -13992,7 +12996,7 @@ rec {
         dependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             usesDefaultFeatures = false;
           }
@@ -14245,11 +13249,11 @@ rec {
           }
           {
             name = "hashbrown";
-            packageId = "hashbrown 0.15.2";
+            packageId = "hashbrown";
           }
           {
             name = "hashlink";
-            packageId = "hashlink 0.10.0";
+            packageId = "hashlink";
           }
           {
             name = "indexmap";
@@ -14275,7 +13279,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             features = [ "derive" "rc" ];
           }
@@ -14478,7 +13482,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -14697,7 +13701,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
           }
           {
@@ -14864,7 +13868,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -14986,7 +13990,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             features = [ "derive" ];
           }
@@ -15152,7 +14156,7 @@ rec {
         };
         resolvedDefaultFeatures = [ "clone-impls" "default" "derive" "extra-traits" "fold" "full" "parsing" "printing" "proc-macro" "visit" "visit-mut" ];
       };
-      "sync_wrapper 0.1.2" = rec {
+      "sync_wrapper" = rec {
         crateName = "sync_wrapper";
         version = "0.1.2";
         edition = "2018";
@@ -15165,19 +14169,6 @@ rec {
           "futures-core" = [ "dep:futures-core" ];
         };
       };
-      "sync_wrapper 1.0.2" = rec {
-        crateName = "sync_wrapper";
-        version = "1.0.2";
-        edition = "2021";
-        sha256 = "0qvjyasd6w18mjg5xlaq5jgy84jsjfsvmnn12c13gypxbv75dwhb";
-        authors = [
-          "Actyx AG <developer@actyx.io>"
-        ];
-        features = {
-          "futures" = [ "futures-core" ];
-          "futures-core" = [ "dep:futures-core" ];
-        };
-      };
       "synstructure" = rec {
         crateName = "synstructure";
         version = "0.13.1";
@@ -15330,9 +14321,9 @@ rec {
       };
       "tempfile" = rec {
         crateName = "tempfile";
-        version = "3.15.0";
+        version = "3.17.1";
         edition = "2021";
-        sha256 = "016pmkbwn3shas44gcwq1kc9lajalb90qafhiip5fvv8h6f5b2ls";
+        sha256 = "0c52ggq5vy5mzgk5ly36cgzs1cig3cv6r1jarijmzxgkn6na1r92";
         authors = [
           "Steven Allen <steven@stebalien.com>"
           "The Rust Project Developers"
@@ -15350,7 +14341,7 @@ rec {
           }
           {
             name = "getrandom";
-            packageId = "getrandom";
+            packageId = "getrandom 0.3.1";
             optional = true;
             usesDefaultFeatures = false;
             target = { target, features }: ((target."unix" or false) || (target."windows" or false) || ("wasi" == target."os"));
@@ -15613,7 +14604,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             usesDefaultFeatures = false;
           }
@@ -15634,7 +14625,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             usesDefaultFeatures = false;
             features = [ "derive" ];
           }
@@ -15734,6 +14725,13 @@ rec {
             packageId = "displaydoc";
             usesDefaultFeatures = false;
           }
+          {
+            name = "serde";
+            packageId = "serde 1.0.218";
+            optional = true;
+            usesDefaultFeatures = false;
+            features = [ "alloc" ];
+          }
           {
             name = "zerovec";
             packageId = "zerovec";
@@ -15747,7 +14745,7 @@ rec {
           "serde" = [ "dep:serde" ];
           "zerovec" = [ "dep:zerovec" ];
         };
-        resolvedDefaultFeatures = [ "alloc" "default" "zerovec" ];
+        resolvedDefaultFeatures = [ "alloc" "default" "serde" "zerovec" ];
       };
       "tinyvec" = rec {
         crateName = "tinyvec";
@@ -16109,16 +15107,16 @@ rec {
       };
       "toml" = rec {
         crateName = "toml";
-        version = "0.8.2";
+        version = "0.8.20";
         edition = "2021";
-        sha256 = "0g9ysjaqvm2mv8q85xpqfn7hi710hj24sd56k49wyddvvyq8lp8q";
+        sha256 = "0j012b37iz1mihksr6a928s6dzszxvblzg3l5wxp7azzsv6sb1yd";
         authors = [
           "Alex Crichton <alex@alexcrichton.com>"
         ];
         dependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
           }
           {
             name = "serde_spanned";
@@ -16132,39 +15130,40 @@ rec {
           }
           {
             name = "toml_edit";
-            packageId = "toml_edit 0.20.2";
+            packageId = "toml_edit 0.22.24";
             optional = true;
+            usesDefaultFeatures = false;
             features = [ "serde" ];
           }
         ];
         devDependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
         ];
         features = {
           "default" = [ "parse" "display" ];
-          "display" = [ "dep:toml_edit" ];
+          "display" = [ "dep:toml_edit" "toml_edit?/display" ];
           "indexmap" = [ "dep:indexmap" ];
-          "parse" = [ "dep:toml_edit" ];
+          "parse" = [ "dep:toml_edit" "toml_edit?/parse" ];
           "preserve_order" = [ "indexmap" ];
         };
-        resolvedDefaultFeatures = [ "parse" ];
+        resolvedDefaultFeatures = [ "default" "display" "parse" ];
       };
       "toml_datetime" = rec {
         crateName = "toml_datetime";
-        version = "0.6.3";
+        version = "0.6.8";
         edition = "2021";
-        sha256 = "0jsy7v8bdvmzsci6imj8fzgd255fmy5fzp6zsri14yrry7i77nkw";
+        sha256 = "0hgv7v9g35d7y9r2afic58jvlwnf73vgd1mz2k8gihlgrf73bmqd";
         authors = [
           "Alex Crichton <alex@alexcrichton.com>"
         ];
         dependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
           }
         ];
@@ -16194,7 +15193,7 @@ rec {
           }
           {
             name = "winnow";
-            packageId = "winnow";
+            packageId = "winnow 0.5.40";
           }
         ];
         features = {
@@ -16212,6 +15211,36 @@ rec {
           "Andronik Ordian <write@reusable.software>"
           "Ed Page <eopage@gmail.com>"
         ];
+        dependencies = [
+          {
+            name = "indexmap";
+            packageId = "indexmap";
+            features = [ "std" ];
+          }
+          {
+            name = "toml_datetime";
+            packageId = "toml_datetime";
+          }
+          {
+            name = "winnow";
+            packageId = "winnow 0.5.40";
+          }
+        ];
+        features = {
+          "perf" = [ "dep:kstring" ];
+          "serde" = [ "dep:serde" "toml_datetime/serde" "dep:serde_spanned" ];
+        };
+        resolvedDefaultFeatures = [ "default" ];
+      };
+      "toml_edit 0.22.24" = rec {
+        crateName = "toml_edit";
+        version = "0.22.24";
+        edition = "2021";
+        sha256 = "0x0lgp70x5cl9nla03xqs5vwwwlrwmd0djkdrp3h3lpdymgpkd0p";
+        authors = [
+          "Andronik Ordian <write@reusable.software>"
+          "Ed Page <eopage@gmail.com>"
+        ];
         dependencies = [
           {
             name = "indexmap";
@@ -16220,7 +15249,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
           }
           {
@@ -16235,209 +15264,17 @@ rec {
           }
           {
             name = "winnow";
-            packageId = "winnow";
+            packageId = "winnow 0.7.3";
+            optional = true;
           }
         ];
         features = {
+          "default" = [ "parse" "display" ];
+          "parse" = [ "dep:winnow" ];
           "perf" = [ "dep:kstring" ];
           "serde" = [ "dep:serde" "toml_datetime/serde" "dep:serde_spanned" ];
         };
-        resolvedDefaultFeatures = [ "default" "serde" ];
-      };
-      "tower" = rec {
-        crateName = "tower";
-        version = "0.5.2";
-        edition = "2018";
-        sha256 = "1ybmd59nm4abl9bsvy6rx31m4zvzp5rja2slzpn712y9b68ssffh";
-        authors = [
-          "Tower Maintainers <team@tower-rs.com>"
-        ];
-        dependencies = [
-          {
-            name = "futures-core";
-            packageId = "futures-core";
-            optional = true;
-          }
-          {
-            name = "futures-util";
-            packageId = "futures-util";
-            optional = true;
-            usesDefaultFeatures = false;
-            features = [ "alloc" ];
-          }
-          {
-            name = "pin-project-lite";
-            packageId = "pin-project-lite";
-            optional = true;
-          }
-          {
-            name = "sync_wrapper";
-            packageId = "sync_wrapper 1.0.2";
-            optional = true;
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            optional = true;
-            features = [ "sync" ];
-          }
-          {
-            name = "tower-layer";
-            packageId = "tower-layer";
-          }
-          {
-            name = "tower-service";
-            packageId = "tower-service";
-          }
-          {
-            name = "tracing";
-            packageId = "tracing";
-            optional = true;
-            usesDefaultFeatures = false;
-            features = [ "std" ];
-          }
-        ];
-        devDependencies = [
-          {
-            name = "pin-project-lite";
-            packageId = "pin-project-lite";
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            features = [ "macros" "sync" "test-util" "rt-multi-thread" ];
-          }
-          {
-            name = "tracing";
-            packageId = "tracing";
-            usesDefaultFeatures = false;
-            features = [ "std" ];
-          }
-        ];
-        features = {
-          "__common" = [ "futures-core" "pin-project-lite" ];
-          "balance" = [ "discover" "load" "ready-cache" "make" "slab" "util" ];
-          "buffer" = [ "__common" "tokio/sync" "tokio/rt" "tokio-util" "tracing" ];
-          "discover" = [ "__common" ];
-          "filter" = [ "__common" "futures-util" ];
-          "full" = [ "balance" "buffer" "discover" "filter" "hedge" "limit" "load" "load-shed" "make" "ready-cache" "reconnect" "retry" "spawn-ready" "steer" "timeout" "util" ];
-          "futures-core" = [ "dep:futures-core" ];
-          "futures-util" = [ "dep:futures-util" ];
-          "hdrhistogram" = [ "dep:hdrhistogram" ];
-          "hedge" = [ "util" "filter" "futures-util" "hdrhistogram" "tokio/time" "tracing" ];
-          "indexmap" = [ "dep:indexmap" ];
-          "limit" = [ "__common" "tokio/time" "tokio/sync" "tokio-util" "tracing" ];
-          "load" = [ "__common" "tokio/time" "tracing" ];
-          "load-shed" = [ "__common" ];
-          "log" = [ "tracing/log" ];
-          "make" = [ "futures-util" "pin-project-lite" "tokio/io-std" ];
-          "pin-project-lite" = [ "dep:pin-project-lite" ];
-          "ready-cache" = [ "futures-core" "futures-util" "indexmap" "tokio/sync" "tracing" "pin-project-lite" ];
-          "reconnect" = [ "make" "tokio/io-std" "tracing" ];
-          "retry" = [ "__common" "tokio/time" "util" ];
-          "slab" = [ "dep:slab" ];
-          "spawn-ready" = [ "__common" "futures-util" "tokio/sync" "tokio/rt" "util" "tracing" ];
-          "sync_wrapper" = [ "dep:sync_wrapper" ];
-          "timeout" = [ "pin-project-lite" "tokio/time" ];
-          "tokio" = [ "dep:tokio" ];
-          "tokio-stream" = [ "dep:tokio-stream" ];
-          "tokio-util" = [ "dep:tokio-util" ];
-          "tracing" = [ "dep:tracing" ];
-          "util" = [ "__common" "futures-util" "pin-project-lite" "sync_wrapper" ];
-        };
-        resolvedDefaultFeatures = [ "__common" "futures-core" "futures-util" "log" "make" "pin-project-lite" "sync_wrapper" "tokio" "tracing" "util" ];
-      };
-      "tower-http" = rec {
-        crateName = "tower-http";
-        version = "0.6.2";
-        edition = "2018";
-        sha256 = "15wnvhl6cpir9125s73bqjzjsvfb0fmndmsimnl2ddnlhfvs6gs0";
-        libName = "tower_http";
-        authors = [
-          "Tower Maintainers <team@tower-rs.com>"
-        ];
-        dependencies = [
-          {
-            name = "bitflags";
-            packageId = "bitflags 2.8.0";
-          }
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-          {
-            name = "http";
-            packageId = "http 1.2.0";
-          }
-          {
-            name = "pin-project-lite";
-            packageId = "pin-project-lite";
-          }
-          {
-            name = "tower-layer";
-            packageId = "tower-layer";
-          }
-          {
-            name = "tower-service";
-            packageId = "tower-service";
-          }
-        ];
-        devDependencies = [
-          {
-            name = "bytes";
-            packageId = "bytes";
-          }
-        ];
-        features = {
-          "async-compression" = [ "dep:async-compression" ];
-          "auth" = [ "base64" "validate-request" ];
-          "base64" = [ "dep:base64" ];
-          "catch-panic" = [ "tracing" "futures-util/std" "dep:http-body" "dep:http-body-util" ];
-          "compression-br" = [ "async-compression/brotli" "futures-core" "dep:http-body" "tokio-util" "tokio" ];
-          "compression-deflate" = [ "async-compression/zlib" "futures-core" "dep:http-body" "tokio-util" "tokio" ];
-          "compression-full" = [ "compression-br" "compression-deflate" "compression-gzip" "compression-zstd" ];
-          "compression-gzip" = [ "async-compression/gzip" "futures-core" "dep:http-body" "tokio-util" "tokio" ];
-          "compression-zstd" = [ "async-compression/zstd" "futures-core" "dep:http-body" "tokio-util" "tokio" ];
-          "decompression-br" = [ "async-compression/brotli" "futures-core" "dep:http-body" "dep:http-body-util" "tokio-util" "tokio" ];
-          "decompression-deflate" = [ "async-compression/zlib" "futures-core" "dep:http-body" "dep:http-body-util" "tokio-util" "tokio" ];
-          "decompression-full" = [ "decompression-br" "decompression-deflate" "decompression-gzip" "decompression-zstd" ];
-          "decompression-gzip" = [ "async-compression/gzip" "futures-core" "dep:http-body" "dep:http-body-util" "tokio-util" "tokio" ];
-          "decompression-zstd" = [ "async-compression/zstd" "futures-core" "dep:http-body" "dep:http-body-util" "tokio-util" "tokio" ];
-          "follow-redirect" = [ "futures-util" "dep:http-body" "iri-string" "tower/util" ];
-          "fs" = [ "futures-util" "dep:http-body" "dep:http-body-util" "tokio/fs" "tokio-util/io" "tokio/io-util" "dep:http-range-header" "mime_guess" "mime" "percent-encoding" "httpdate" "set-status" "futures-util/alloc" "tracing" ];
-          "full" = [ "add-extension" "auth" "catch-panic" "compression-full" "cors" "decompression-full" "follow-redirect" "fs" "limit" "map-request-body" "map-response-body" "metrics" "normalize-path" "propagate-header" "redirect" "request-id" "sensitive-headers" "set-header" "set-status" "timeout" "trace" "util" "validate-request" ];
-          "futures-core" = [ "dep:futures-core" ];
-          "futures-util" = [ "dep:futures-util" ];
-          "httpdate" = [ "dep:httpdate" ];
-          "iri-string" = [ "dep:iri-string" ];
-          "limit" = [ "dep:http-body" "dep:http-body-util" ];
-          "metrics" = [ "dep:http-body" "tokio/time" ];
-          "mime" = [ "dep:mime" ];
-          "mime_guess" = [ "dep:mime_guess" ];
-          "percent-encoding" = [ "dep:percent-encoding" ];
-          "request-id" = [ "uuid" ];
-          "timeout" = [ "dep:http-body" "tokio/time" ];
-          "tokio" = [ "dep:tokio" ];
-          "tokio-util" = [ "dep:tokio-util" ];
-          "tower" = [ "dep:tower" ];
-          "trace" = [ "dep:http-body" "tracing" ];
-          "tracing" = [ "dep:tracing" ];
-          "util" = [ "tower" ];
-          "uuid" = [ "dep:uuid" ];
-          "validate-request" = [ "mime" ];
-        };
-        resolvedDefaultFeatures = [ "cors" "default" ];
-      };
-      "tower-layer" = rec {
-        crateName = "tower-layer";
-        version = "0.3.3";
-        edition = "2018";
-        sha256 = "03kq92fdzxin51w8iqix06dcfgydyvx7yr6izjq0p626v9n2l70j";
-        libName = "tower_layer";
-        authors = [
-          "Tower Maintainers <team@tower-rs.com>"
-        ];
-
+        resolvedDefaultFeatures = [ "display" "parse" "serde" ];
       };
       "tower-service" = rec {
         crateName = "tower-service";
@@ -16743,7 +15580,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             features = [ "derive" ];
           }
           {
@@ -17047,17 +15884,6 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" "std" ];
       };
-      "urlencoding" = rec {
-        crateName = "urlencoding";
-        version = "2.1.3";
-        edition = "2021";
-        sha256 = "1nj99jp37k47n0hvaz5fvz7z6jd0sb4ppvfy3nphr1zbnyixpy6s";
-        authors = [
-          "Kornel <kornel@geekhood.net>"
-          "Bertram Truong <b@bertramtruong.com>"
-        ];
-
-      };
       "utf-8" = rec {
         crateName = "utf-8";
         version = "0.7.6";
@@ -17146,12 +15972,12 @@ rec {
         dependencies = [
           {
             name = "getrandom";
-            packageId = "getrandom";
+            packageId = "getrandom 0.2.15";
             optional = true;
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             usesDefaultFeatures = false;
           }
@@ -17186,7 +16012,7 @@ rec {
         dependencies = [
           {
             name = "getrandom";
-            packageId = "getrandom";
+            packageId = "getrandom 0.2.15";
             optional = true;
           }
         ];
@@ -17278,129 +16104,6 @@ rec {
           "Sergio Benitez <sb@sergio.bz>"
         ];
 
-      };
-      "visions" = rec {
-        crateName = "visions";
-        version = "0.1.0";
-        edition = "2021";
-        crateBin = [
-          {
-            name = "visions";
-            path = "src/main.rs";
-            requiredFeatures = [ ];
-          }
-        ];
-        # We can't filter paths with references in Nix 2.4
-        # See https://github.com/NixOS/nix/issues/5410
-        src = if ((lib.versionOlder builtins.nixVersion "2.4pre20211007") || (lib.versionOlder "2.5" builtins.nixVersion ))
-          then lib.cleanSourceWith { filter = sourceFilter;  src = ./visions/server; }
-          else ./visions/server;
-        dependencies = [
-          {
-            name = "async-std";
-            packageId = "async-std";
-          }
-          {
-            name = "async-trait";
-            packageId = "async-trait";
-          }
-          {
-            name = "authdb";
-            packageId = "authdb";
-          }
-          {
-            name = "axum";
-            packageId = "axum";
-            features = [ "macros" ];
-          }
-          {
-            name = "futures";
-            packageId = "futures";
-          }
-          {
-            name = "include_dir";
-            packageId = "include_dir";
-          }
-          {
-            name = "lazy_static";
-            packageId = "lazy_static";
-          }
-          {
-            name = "mime";
-            packageId = "mime 0.3.17";
-          }
-          {
-            name = "mime_guess";
-            packageId = "mime_guess 2.0.5";
-          }
-          {
-            name = "pretty_env_logger";
-            packageId = "pretty_env_logger";
-          }
-          {
-            name = "result-extended";
-            packageId = "result-extended";
-          }
-          {
-            name = "rusqlite";
-            packageId = "rusqlite";
-          }
-          {
-            name = "rusqlite_migration";
-            packageId = "rusqlite_migration";
-            features = [ "from-directory" ];
-          }
-          {
-            name = "serde";
-            packageId = "serde 1.0.217";
-          }
-          {
-            name = "serde_json";
-            packageId = "serde_json";
-          }
-          {
-            name = "thiserror";
-            packageId = "thiserror 2.0.11";
-          }
-          {
-            name = "tokio";
-            packageId = "tokio";
-            features = [ "full" ];
-          }
-          {
-            name = "tokio-stream";
-            packageId = "tokio-stream";
-          }
-          {
-            name = "tower-http";
-            packageId = "tower-http";
-            features = [ "cors" ];
-          }
-          {
-            name = "typeshare";
-            packageId = "typeshare";
-          }
-          {
-            name = "urlencoding";
-            packageId = "urlencoding";
-          }
-          {
-            name = "uuid";
-            packageId = "uuid 1.12.0";
-            features = [ "v4" ];
-          }
-        ];
-        devDependencies = [
-          {
-            name = "axum-test";
-            packageId = "axum-test";
-          }
-          {
-            name = "cool_asserts";
-            packageId = "cool_asserts";
-          }
-        ];
-
       };
       "wait-timeout" = rec {
         crateName = "wait-timeout";
@@ -17505,7 +16208,7 @@ rec {
           }
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
           }
           {
             name = "serde_json";
@@ -17598,6 +16301,29 @@ rec {
         };
         resolvedDefaultFeatures = [ "default" "std" ];
       };
+      "wasi 0.13.3+wasi-0.2.2" = rec {
+        crateName = "wasi";
+        version = "0.13.3+wasi-0.2.2";
+        edition = "2021";
+        sha256 = "1lnapbvdcvi3kc749wzqvwrpd483win2kicn1faa4dja38p6v096";
+        authors = [
+          "The Cranelift Project Developers"
+        ];
+        dependencies = [
+          {
+            name = "wit-bindgen-rt";
+            packageId = "wit-bindgen-rt";
+            features = [ "bitflags" ];
+          }
+        ];
+        features = {
+          "compiler_builtins" = [ "dep:compiler_builtins" ];
+          "core" = [ "dep:core" ];
+          "default" = [ "std" ];
+          "rustc-dep-of-std" = [ "compiler_builtins" "core" "rustc-std-workspace-alloc" ];
+          "rustc-std-workspace-alloc" = [ "dep:rustc-std-workspace-alloc" ];
+        };
+      };
       "wasite" = rec {
         crateName = "wasite";
         version = "0.1.0";
@@ -19474,7 +18200,7 @@ rec {
         ];
 
       };
-      "winnow" = rec {
+      "winnow 0.5.40" = rec {
         crateName = "winnow";
         version = "0.5.40";
         edition = "2021";
@@ -19496,6 +18222,28 @@ rec {
         };
         resolvedDefaultFeatures = [ "alloc" "default" "std" ];
       };
+      "winnow 0.7.3" = rec {
+        crateName = "winnow";
+        version = "0.7.3";
+        edition = "2021";
+        sha256 = "1c9bmhpdwbdmll6b4l6skabz0296dchnmnxw84hh2y3ggyllwzqf";
+        dependencies = [
+          {
+            name = "memchr";
+            packageId = "memchr";
+            optional = true;
+            usesDefaultFeatures = false;
+          }
+        ];
+        features = {
+          "debug" = [ "std" "dep:anstream" "dep:anstyle" "dep:is-terminal" "dep:terminal_size" ];
+          "default" = [ "std" ];
+          "simd" = [ "dep:memchr" ];
+          "std" = [ "alloc" "memchr?/std" ];
+          "unstable-doc" = [ "alloc" "std" "simd" "unstable-recover" ];
+        };
+        resolvedDefaultFeatures = [ "alloc" "default" "std" ];
+      };
       "winreg" = rec {
         crateName = "winreg";
         version = "0.50.0";
@@ -19521,6 +18269,24 @@ rec {
           "serialization-serde" = [ "transactions" "serde" ];
         };
       };
+      "wit-bindgen-rt" = rec {
+        crateName = "wit-bindgen-rt";
+        version = "0.33.0";
+        edition = "2021";
+        sha256 = "0g4lwfp9x6a2i1hgjn8k14nr4fsnpd5izxhc75zpi2s5cvcg6s1j";
+        libName = "wit_bindgen_rt";
+        dependencies = [
+          {
+            name = "bitflags";
+            packageId = "bitflags 2.8.0";
+            optional = true;
+          }
+        ];
+        features = {
+          "bitflags" = [ "dep:bitflags" ];
+        };
+        resolvedDefaultFeatures = [ "bitflags" ];
+      };
       "write16" = rec {
         crateName = "write16";
         version = "1.0.0";
@@ -19544,23 +18310,17 @@ rec {
           "either" = [ "dep:either" ];
         };
       };
-      "yansi" = rec {
-        crateName = "yansi";
-        version = "1.0.1";
+      "xml-rs" = rec {
+        crateName = "xml-rs";
+        version = "0.8.25";
         edition = "2021";
-        sha256 = "0jdh55jyv0dpd38ij4qh60zglbw9aa8wafqai6m0wa7xaxk3mrfg";
+        crateBin = [];
+        sha256 = "1i73ajf6scni5bi1a51r19xykgrambdx5fkks0fyg5jqqbml1ff5";
+        libName = "xml";
         authors = [
-          "Sergio Benitez <sb@sergio.bz>"
+          "Vladimir Matveev <vmatveev@citrine.cc>"
         ];
-        features = {
-          "default" = [ "std" ];
-          "detect-env" = [ "std" ];
-          "detect-tty" = [ "is-terminal" "std" ];
-          "hyperlink" = [ "std" ];
-          "is-terminal" = [ "dep:is-terminal" ];
-          "std" = [ "alloc" ];
-        };
-        resolvedDefaultFeatures = [ "alloc" "default" "std" ];
+
       };
       "yansi-term" = rec {
         crateName = "yansi-term";
@@ -19598,7 +18358,7 @@ rec {
         dependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             optional = true;
             usesDefaultFeatures = false;
           }
@@ -19623,7 +18383,7 @@ rec {
         devDependencies = [
           {
             name = "serde";
-            packageId = "serde 1.0.217";
+            packageId = "serde 1.0.218";
             usesDefaultFeatures = false;
           }
         ];
diff --git a/Cargo.toml b/Cargo.toml
index 4ffc188..ec41881 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -22,10 +22,12 @@ members = [
     "gm-dash/server",
     "hex-grid",
     "icon-test",
+    "l10n-db",
     "memorycache",
     "nom-training",
     "otg/core",
     "otg/gtk",
+    "pico-st7789",
     "result-extended",
     "screenplay",
     "sgf",
@@ -34,6 +36,5 @@ members = [
     "visions/server",
     "visions/types",
     "visions/ui",
-    "visions/yew-app",
     # "bike-lights/bike",
 ]
diff --git a/crate-hashes.json b/crate-hashes.json
index 47b533e..480f3ca 100644
--- a/crate-hashes.json
+++ b/crate-hashes.json
@@ -3,7 +3,6 @@
   "registry+https://github.com/rust-lang/crates.io-index#adler2@2.0.0": "09r6drylvgy8vv8k20lnbvwq8gp09h7smfn6h1rxsy15pgh629si",
   "registry+https://github.com/rust-lang/crates.io-index#adler32@1.2.0": "0d7jq7jsjyhsgbhnfq5fvrlh9j0i9g1fqrl2735ibv5f75yjgqda",
   "registry+https://github.com/rust-lang/crates.io-index#adler@1.0.2": "1zim79cvzd5yrkzl3nyfx0avijwgk9fqv3yrscdy1cc79ih02qpj",
-  "registry+https://github.com/rust-lang/crates.io-index#ahash@0.8.11": "04chdfkls5xmhp1d48gnjsmglbqibizs3bpbj6rsj604m10si7g8",
   "registry+https://github.com/rust-lang/crates.io-index#aho-corasick@1.1.3": "05mrpkvdgp5d20y2p989f187ry9diliijgwrs254fs9s1m1x6q4f",
   "registry+https://github.com/rust-lang/crates.io-index#allocator-api2@0.2.21": "08zrzs022xwndihvzdn78yqarv2b9696y67i6h78nla3ww87jgb8",
   "registry+https://github.com/rust-lang/crates.io-index#android-tzdata@0.1.1": "1w7ynjxrfs97xg3qlcdns4kgfpwcdv824g611fq32cag4cdr96g9",
@@ -15,7 +14,6 @@
   "registry+https://github.com/rust-lang/crates.io-index#anstyle-wincon@3.0.7": "0kmf0fq4c8yribdpdpylzz1zccpy84hizmcsac3wrac1f7kk8dfa",
   "registry+https://github.com/rust-lang/crates.io-index#anstyle@1.0.10": "1yai2vppmd7zlvlrp9grwll60knrmscalf8l2qpfz8b7y5lkpk2m",
   "registry+https://github.com/rust-lang/crates.io-index#anyhow@1.0.95": "010vd1ki8w84dzgx6c81sc8qm9n02fxic1gkpv52zp4nwrn0kb1l",
-  "registry+https://github.com/rust-lang/crates.io-index#assert-json-diff@2.0.2": "04mg3w0rh3schpla51l18362hsirl23q93aisws2irrj32wg5r27",
   "registry+https://github.com/rust-lang/crates.io-index#async-channel@1.9.0": "0dbdlkzlncbibd3ij6y6jmvjd0cmdn48ydcfdpfhw09njd93r5c1",
   "registry+https://github.com/rust-lang/crates.io-index#async-channel@2.3.1": "0skvwxj6ysfc6d7bhczz9a2550260g62bm5gl0nmjxxyn007id49",
   "registry+https://github.com/rust-lang/crates.io-index#async-executor@1.13.1": "1v6w1dbvsmw6cs4dk4lxj5dvrikc6xi479wikwaab2qy3h09mjih",
@@ -27,13 +25,8 @@
   "registry+https://github.com/rust-lang/crates.io-index#async-trait@0.1.85": "0mm0gwad44zs7mna4a0m1z4dhzpmydfj73w4wm23c8xpnhrli4rz",
   "registry+https://github.com/rust-lang/crates.io-index#atoi@2.0.0": "0a05h42fggmy7h0ajjv6m7z72l924i7igbx13hk9d8pyign9k3gj",
   "registry+https://github.com/rust-lang/crates.io-index#atomic-waker@1.1.2": "1h5av1lw56m0jf0fd3bchxq8a30xv0b4wv8s4zkp4s0i7mfvs18m",
-  "registry+https://github.com/rust-lang/crates.io-index#auto-future@1.0.0": "0wykbakzh227vz6frx9p48zsq0wpswgmb7v3917m53m7gr2pw7iw",
   "registry+https://github.com/rust-lang/crates.io-index#autocfg@0.1.8": "0y4vw4l4izdxq1v0rrhvmlbqvalrqrmk60v1z0dqlgnlbzkl7phd",
   "registry+https://github.com/rust-lang/crates.io-index#autocfg@1.4.0": "09lz3by90d2hphbq56znag9v87gfpd9gb8nr82hll8z6x2nhprdc",
-  "registry+https://github.com/rust-lang/crates.io-index#axum-core@0.4.5": "16b1496c4gm387q20hkv5ic3k5bd6xmnvk50kwsy6ymr8rhvvwh9",
-  "registry+https://github.com/rust-lang/crates.io-index#axum-macros@0.4.2": "1klv77c889jm05bzayaaiinalarhvh2crc2w4nvp3l581xaj7lap",
-  "registry+https://github.com/rust-lang/crates.io-index#axum-test@16.4.1": "1p5qxacvxsagnqq30nr2wznjyhgb8svsfb925ah3d2b0s91s9qv3",
-  "registry+https://github.com/rust-lang/crates.io-index#axum@0.7.9": "07z7wqczi9i8xb4460rvn39p4wjqwr32hx907crd1vwb2fy8ijpd",
   "registry+https://github.com/rust-lang/crates.io-index#az@1.2.1": "0ww9k1w3al7x5qmb7f13v3s9c2pg1pdxbs8xshqy6zyrchj4qzkv",
   "registry+https://github.com/rust-lang/crates.io-index#backtrace@0.3.74": "06pfif7nwx66qf2zaanc2fcq7m64i91ki9imw9xd3bnz5hrwp0ld",
   "registry+https://github.com/rust-lang/crates.io-index#base64@0.21.7": "0rw52yvsk75kar9wgqfwgb414kvil1gn7mqkrhn9zf1537mpsacx",
@@ -53,7 +46,6 @@
   "registry+https://github.com/rust-lang/crates.io-index#bytemuck@1.21.0": "18wj81x9xhqcd6985r8qxmbik6szjfjfj62q3xklw8h2p3x7srgg",
   "registry+https://github.com/rust-lang/crates.io-index#byteorder@1.5.0": "0jzncxyf404mwqdbspihyzpkndfgda450l0893pz5xj685cg5l0z",
   "registry+https://github.com/rust-lang/crates.io-index#bytes@1.9.0": "16ykzx24v1x4f42v2lxyvlczqhdfji3v7r4ghwckpwijzvb1hn9j",
-  "registry+https://github.com/rust-lang/crates.io-index#bytesize@1.3.0": "1k3aak70iwz4s2gsjbxf0ws4xnixqbdz6p2ha96s06748fpniqx3",
   "registry+https://github.com/rust-lang/crates.io-index#cairo-rs@0.18.5": "1qjfkcq3mrh3p01nnn71dy3kn99g21xx3j8xcdvzn8ll2pq6x8lc",
   "registry+https://github.com/rust-lang/crates.io-index#cairo-sys-rs@0.18.2": "0lfsxl7ylw3phbnwmz3k58j1gnqi6kc2hdc7g3bb7f4hwnl9yp38",
   "registry+https://github.com/rust-lang/crates.io-index#cc@1.2.10": "0aaj2ivamhfzhgb9maasnfkh03s2mzhzpzwrkghgzbkfnv5qy80k",
@@ -64,9 +56,9 @@
   "registry+https://github.com/rust-lang/crates.io-index#chrono-tz@0.8.6": "0vlksnmpb6rd4h55245agnfhphnpslwnq9al3aw3is43dd3f16nm",
   "registry+https://github.com/rust-lang/crates.io-index#chrono@0.4.39": "09g8nf409lb184kl9j4s85k0kn8wzgjkp5ls9zid50b886fwqdky",
   "registry+https://github.com/rust-lang/crates.io-index#clang-sys@1.8.1": "1x1r9yqss76z8xwpdanw313ss6fniwc1r7dzb5ycjn0ph53kj0hb",
-  "registry+https://github.com/rust-lang/crates.io-index#clap@4.5.26": "10v7qvn90calfbhap1c4r249i5c7fbxj09fn3szfz9pkis85xsx8",
-  "registry+https://github.com/rust-lang/crates.io-index#clap_builder@4.5.26": "08f1mzcvi7zjhm7hvz6al4jnv70ccqhwiaq74hihlspwnl0iic4n",
-  "registry+https://github.com/rust-lang/crates.io-index#clap_derive@4.5.24": "131ih3dm76srkbpfx7zfspp9b556zgzj31wqhl0ji2b39lcmbdsl",
+  "registry+https://github.com/rust-lang/crates.io-index#clap@4.5.30": "0vcyrn4ymq2gd56sl3xnfki8q8llg64sj3rj3qx33mgsf66v3dwj",
+  "registry+https://github.com/rust-lang/crates.io-index#clap_builder@4.5.30": "0369xis2ar46icsaxqyy37976mlb62alzyx4j53k99vq2w3v4pd3",
+  "registry+https://github.com/rust-lang/crates.io-index#clap_derive@4.5.28": "1vgigkhljp3r8r5lwdrn1ij93nafmjwh8cx77nppb9plqsaysk5z",
   "registry+https://github.com/rust-lang/crates.io-index#clap_lex@0.7.4": "19nwfls5db269js5n822vkc8dw0wjq2h1wf0hgr06ld2g52d2spl",
   "registry+https://github.com/rust-lang/crates.io-index#cloudabi@0.0.3": "0kxcg83jlihy0phnd2g8c2c303px3l2p3pkjz357ll6llnd5pz6x",
   "registry+https://github.com/rust-lang/crates.io-index#color_quant@1.1.0": "12q1n427h2bbmmm1mnglr57jaz2dj9apk0plcxw7nwqiai7qjyrx",
@@ -76,7 +68,6 @@
   "registry+https://github.com/rust-lang/crates.io-index#convert_case@0.6.0": "1jn1pq6fp3rri88zyw6jlhwwgf6qiyc08d6gjv0qypgkl862n67c",
   "registry+https://github.com/rust-lang/crates.io-index#cookie-factory@0.3.3": "18mka6fk3843qq3jw1fdfvzyv05kx7kcmirfbs2vg2kbw9qzm1cq",
   "registry+https://github.com/rust-lang/crates.io-index#cookie@0.17.0": "096c52jg9iq4lfcps2psncswv33fc30mmnaa2sbzzcfcw71kgyvy",
-  "registry+https://github.com/rust-lang/crates.io-index#cookie@0.18.1": "0iy749flficrlvgr3hjmf3igr738lk81n5akzf4ym4cs6cxg7pjd",
   "registry+https://github.com/rust-lang/crates.io-index#cool_asserts@2.0.3": "1v18dg7ifx41k2f82j3gsnpm1fg9wk5s4zv7sf42c7pnad72b7zf",
   "registry+https://github.com/rust-lang/crates.io-index#core-foundation-sys@0.8.7": "12w8j73lazxmr1z0h98hf3z623kl8ms7g07jch7n4p8f9nwlhdkp",
   "registry+https://github.com/rust-lang/crates.io-index#core-foundation@0.9.4": "13zvbbj07yk3b61b8fhwfzhy35535a583irf23vlcg59j7h9bqci",
@@ -94,7 +85,6 @@
   "registry+https://github.com/rust-lang/crates.io-index#deflate@0.8.6": "0x6iqlayg129w63999kz97m279m0jj4x4sm6gkqlvmp73y70yxvk",
   "registry+https://github.com/rust-lang/crates.io-index#der@0.7.9": "1h4vzjfa1lczxdf8avfj9qlwh1qianqlxdy1g5rn762qnvkzhnzm",
   "registry+https://github.com/rust-lang/crates.io-index#deranged@0.3.11": "1d1ibqqnr5qdrpw8rclwrf1myn3wf0dygl04idf4j2s49ah6yaxl",
-  "registry+https://github.com/rust-lang/crates.io-index#diff@0.1.13": "1j0nzjxci2zqx63hdcihkp0a4dkdmzxd7my4m7zk6cjyfy34j9an",
   "registry+https://github.com/rust-lang/crates.io-index#digest@0.10.7": "14p2n6ih29x81akj097lvz7wi9b6b9hvls0lwrv7b6xwyy0s5ncy",
   "registry+https://github.com/rust-lang/crates.io-index#dimensioned@0.7.0": "09ky8s3higkf677lmyqg30hmj66gpg7hx907s6hfvbk2a9av05r5",
   "registry+https://github.com/rust-lang/crates.io-index#dimensioned@0.8.0": "15s3j4ry943xqlac63bp81sgdk9s3yilysabzww35j9ibmnaic50",
@@ -110,8 +100,6 @@
   "registry+https://github.com/rust-lang/crates.io-index#event-listener@2.5.3": "1q4w3pndc518crld6zsqvvpy9lkzwahp2zgza9kbzmmqh9gif1h2",
   "registry+https://github.com/rust-lang/crates.io-index#event-listener@5.4.0": "1bii2gn3vaa33s0gr2zph7cagiq0ppcfxcxabs24ri9z9kgar4il",
   "registry+https://github.com/rust-lang/crates.io-index#exr@1.73.0": "1q47yq78q9k210r6jy1wwrilxwwxqavik9l3l426rd17k7srfcgq",
-  "registry+https://github.com/rust-lang/crates.io-index#fallible-iterator@0.3.0": "0ja6l56yka5vn4y4pk6hn88z0bpny7a8k1919aqjzp0j1yhy9k1a",
-  "registry+https://github.com/rust-lang/crates.io-index#fallible-streaming-iterator@0.1.9": "0nj6j26p71bjy8h42x6jahx1hn0ng6mc2miwpgwnp8vnwqf4jq3k",
   "registry+https://github.com/rust-lang/crates.io-index#fastrand@2.3.0": "1ghiahsw1jd68df895cy5h3gzwk30hndidn3b682zmshpgmrx41p",
   "registry+https://github.com/rust-lang/crates.io-index#fdeflate@0.3.7": "130ga18vyxbb5idbgi07njymdaavvk6j08yh1dfarm294ssm6s0y",
   "registry+https://github.com/rust-lang/crates.io-index#field-offset@0.3.6": "0zq5sssaa2ckmcmxxbly8qgz3sxpb8g1lwv90sdh1z74qif2gqiq",
@@ -146,6 +134,7 @@
   "registry+https://github.com/rust-lang/crates.io-index#generic-array@0.11.2": "0a7w8w0rg47nmcinnfzv443lcyb8mplwc251p1jyr5xj2yh6wzv6",
   "registry+https://github.com/rust-lang/crates.io-index#generic-array@0.14.7": "16lyyrzrljfq424c3n8kfwkqihlimmsg5nhshbbp48np3yjrqr45",
   "registry+https://github.com/rust-lang/crates.io-index#getrandom@0.2.15": "1mzlnrb3dgyd1fb84gvw10pyr8wdqdl4ry4sr64i1s8an66pqmn4",
+  "registry+https://github.com/rust-lang/crates.io-index#getrandom@0.3.1": "1y154yzby383p63ndw6zpfm0fz3vf6c0zdwc7df6vkl150wrr923",
   "registry+https://github.com/rust-lang/crates.io-index#gif@0.11.4": "01hbw3isapzpzff8l6aw55jnaqx2bcscrbwyf3rglkbbfp397p9y",
   "registry+https://github.com/rust-lang/crates.io-index#gif@0.13.1": "1whrkvdg26gp1r7f95c6800y6ijqw5y0z8rgj6xihpi136dxdciz",
   "registry+https://github.com/rust-lang/crates.io-index#gimli@0.31.1": "0gvqc0ramx8szv76jhfd4dms0zyamvlg4whhiz11j34hh3dqxqh7",
@@ -170,10 +159,8 @@
   "registry+https://github.com/rust-lang/crates.io-index#gtk4@0.7.3": "0hh8nzglmz94v1m1h6vy8z12m6fr7ia467ry0md5fa4p7sm53sss",
   "registry+https://github.com/rust-lang/crates.io-index#h2@0.3.26": "1s7msnfv7xprzs6xzfj5sg6p8bjcdpcqcmjjbkd345cyi1x55zl1",
   "registry+https://github.com/rust-lang/crates.io-index#half@2.4.1": "123q4zzw1x4309961i69igzd1wb7pj04aaii3kwasrz3599qrl3d",
-  "registry+https://github.com/rust-lang/crates.io-index#hashbrown@0.14.5": "1wa1vy1xs3mp11bn3z9dv0jricgr6a2j0zkf1g19yz3vw4il89z5",
   "registry+https://github.com/rust-lang/crates.io-index#hashbrown@0.15.2": "12dj0yfn59p3kh3679ac0w1fagvzf4z2zp87a13gbbqbzw0185dz",
   "registry+https://github.com/rust-lang/crates.io-index#hashlink@0.10.0": "1h8lzvnl9qxi3zyagivzz2p1hp6shgddfmccyf6jv7s1cdicz0kk",
-  "registry+https://github.com/rust-lang/crates.io-index#hashlink@0.9.1": "1byq4nyrflm5s6wdx5qwp96l1qbp2d0nljvrr5yqrsfy51qzz93b",
   "registry+https://github.com/rust-lang/crates.io-index#headers-core@0.2.0": "0ab469xfpd411mc3dhmjhmzrhqikzyj8a17jn5bkj9zfpy0n9xp7",
   "registry+https://github.com/rust-lang/crates.io-index#headers@0.3.9": "0w62gnwh2p1lml0zqdkrx9dp438881nhz32zrzdy61qa0a9kns06",
   "registry+https://github.com/rust-lang/crates.io-index#heck@0.4.1": "1a7mqsnycv5z4z5vnv1k34548jzmc0ajic7c1j8jsaspnhw5ql4m",
@@ -185,19 +172,15 @@
   "registry+https://github.com/rust-lang/crates.io-index#hkdf@0.12.4": "1xxxzcarz151p1b858yn5skmhyrvn8fs4ivx5km3i1kjmnr8wpvv",
   "registry+https://github.com/rust-lang/crates.io-index#hmac@0.12.1": "0pmbr069sfg76z7wsssfk5ddcqd9ncp79fyz6zcm6yn115yc6jbc",
   "registry+https://github.com/rust-lang/crates.io-index#home@0.5.11": "1kxb4k87a9sayr8jipr7nq9wpgmjk4hk4047hmf9kc24692k75aq",
-  "registry+https://github.com/rust-lang/crates.io-index#http-body-util@0.1.2": "0kslwazg4400qnc2azkrgqqci0fppv12waicnsy5d8hncvbjjd3r",
   "registry+https://github.com/rust-lang/crates.io-index#http-body@0.4.6": "1lmyjfk6bqk6k9gkn1dxq770sb78pqbqshga241hr5p995bb5skw",
-  "registry+https://github.com/rust-lang/crates.io-index#http-body@1.0.1": "111ir5k2b9ihz5nr9cz7cwm7fnydca7dx4hc7vr16scfzghxrzhy",
   "registry+https://github.com/rust-lang/crates.io-index#http@0.2.12": "1w81s4bcbmcj9bjp7mllm8jlz6b31wzvirz8bgpzbqkpwmbvn730",
   "registry+https://github.com/rust-lang/crates.io-index#http@1.2.0": "1skglzdf98j5nzxlii540n11is0w4l80mi5sm3xrj716asps4v7i",
   "registry+https://github.com/rust-lang/crates.io-index#httparse@1.9.5": "0ip9v8m9lvgvq1lznl31wvn0ch1v254na7lhid9p29yx9rbx6wbx",
   "registry+https://github.com/rust-lang/crates.io-index#httpdate@1.0.3": "1aa9rd2sac0zhjqh24c9xvir96g188zldkx0hr6dnnlx5904cfyz",
   "registry+https://github.com/rust-lang/crates.io-index#humantime@2.1.0": "1r55pfkkf5v0ji1x6izrjwdq9v6sc7bv99xj6srywcar37xmnfls",
   "registry+https://github.com/rust-lang/crates.io-index#hyper-tls@0.5.0": "01crgy13102iagakf6q4mb75dprzr7ps1gj0l5hxm1cvm7gks66n",
-  "registry+https://github.com/rust-lang/crates.io-index#hyper-util@0.1.10": "1d1iwrkysjhq63pg54zk3vfby1j7zmxzm9zzyfr4lwvp0szcybfz",
   "registry+https://github.com/rust-lang/crates.io-index#hyper@0.10.16": "0wwjh9p3mzvg3fss2lqz5r7ddcgl1fh9w6my2j69d6k0lbcm41ha",
   "registry+https://github.com/rust-lang/crates.io-index#hyper@0.14.32": "1rvcb0smz8q1i0y6p7rwxr02x5sclfg2hhxf3g0774zczn0cgps1",
-  "registry+https://github.com/rust-lang/crates.io-index#hyper@1.5.2": "1q7akfb443yrjzkmnnbp2vs8zi15hgbk466rr4y144v4ppabhvr5",
   "registry+https://github.com/rust-lang/crates.io-index#iana-time-zone-haiku@0.1.2": "17r6jmj31chn7xs9698r122mapq85mfnv98bb4pg6spm0si2f67k",
   "registry+https://github.com/rust-lang/crates.io-index#iana-time-zone@0.1.61": "085jjsls330yj1fnwykfzmb2f10zp6l7w4fhq81ng81574ghhpi3",
   "registry+https://github.com/rust-lang/crates.io-index#icu_collections@1.5.0": "09j5kskirl59mvqc8kabhy7005yyy7dp88jw9f6f3gkf419a8byv",
@@ -215,8 +198,6 @@
   "registry+https://github.com/rust-lang/crates.io-index#idna_adapter@1.2.0": "0wggnkiivaj5lw0g0384ql2d7zk4ppkn3b1ry4n0ncjpr7qivjns",
   "registry+https://github.com/rust-lang/crates.io-index#image@0.23.14": "18gn2f7xp30pf9aqka877knlq308khxqiwjvsccvzaa4f9zcpzr4",
   "registry+https://github.com/rust-lang/crates.io-index#image@0.24.9": "17gnr6ifnpzvhjf6dwbl9hki8x6bji5mwcqp0048x1jm5yfi742n",
-  "registry+https://github.com/rust-lang/crates.io-index#include_dir@0.7.4": "1pfh3g45z88kwq93skng0n6g3r7zkhq9ldqs9y8rvr7i11s12gcj",
-  "registry+https://github.com/rust-lang/crates.io-index#include_dir_macros@0.7.4": "0x8smnf6knd86g69p19z5lpfsaqp8w0nx14kdpkz1m8bxnkqbavw",
   "registry+https://github.com/rust-lang/crates.io-index#indent_write@2.2.0": "1hqjp80argdskrhd66g9sh542yxy8qi77j6rc69qd0l7l52rdzhc",
   "registry+https://github.com/rust-lang/crates.io-index#indexmap@2.7.0": "07s7jmdymvd0rm4yswp0j3napx57hkjm9gs9n55lvs2g78vj5y32",
   "registry+https://github.com/rust-lang/crates.io-index#intl-memoizer@0.5.2": "1nkvql7c7b76axv4g68di1p2m9bnxq1cbn6mlqcawf72zhhf08py",
@@ -251,7 +232,6 @@
   "registry+https://github.com/rust-lang/crates.io-index#log@0.4.25": "17ydv5zhfv1zzygy458bmg3f3jx1vfziv9d74817w76yhfqgbjq4",
   "registry+https://github.com/rust-lang/crates.io-index#logger@0.4.0": "14xlxvkspcfnspjil0xi63qj5cybxn1hjmr5gq8m4v1g9k5p54bc",
   "registry+https://github.com/rust-lang/crates.io-index#matches@0.1.10": "1994402fq4viys7pjhzisj4wcw894l53g798kkm2y74laxk0jci5",
-  "registry+https://github.com/rust-lang/crates.io-index#matchit@0.7.3": "156bgdmmlv4crib31qhgg49nsjk88dxkdqp80ha2pk2rk6n6ax0f",
   "registry+https://github.com/rust-lang/crates.io-index#md-5@0.10.6": "1kvq5rnpm4fzwmyv5nmnxygdhhb2369888a06gdc9pxyrzh7x7nq",
   "registry+https://github.com/rust-lang/crates.io-index#memchr@2.7.4": "18z32bhxrax0fnjikv475z7ii718hq457qwmaryixfxsl2qrmjkq",
   "registry+https://github.com/rust-lang/crates.io-index#memoffset@0.9.1": "12i17wh9a9plx869g7j4whf62xw68k5zd4k0k5nh6ys5mszid028",
@@ -317,10 +297,9 @@
   "registry+https://github.com/rust-lang/crates.io-index#polling@3.7.4": "0bs4nhwfwsvlzlhah2gbhj3aa9ynvchv2g350wapswh26a65c156",
   "registry+https://github.com/rust-lang/crates.io-index#powerfmt@0.2.0": "14ckj2xdpkhv3h6l5sdmb9f1d57z8hbfpdldjc2vl5givq2y77j3",
   "registry+https://github.com/rust-lang/crates.io-index#ppv-lite86@0.2.20": "017ax9ssdnpww7nrl1hvqh2lzncpv04nnsibmnw9nxjnaqlpp5bp",
-  "registry+https://github.com/rust-lang/crates.io-index#pretty_assertions@1.4.1": "0v8iq35ca4rw3rza5is3wjxwsf88303ivys07anc5yviybi31q9s",
   "registry+https://github.com/rust-lang/crates.io-index#pretty_env_logger@0.5.0": "076w9dnvcpx6d3mdbkqad8nwnsynb7c8haxmscyrz7g3vga28mw6",
   "registry+https://github.com/rust-lang/crates.io-index#proc-macro-crate@1.3.1": "069r1k56bvgk0f58dm5swlssfcp79im230affwk6d9ck20g04k3z",
-  "registry+https://github.com/rust-lang/crates.io-index#proc-macro-crate@2.0.2": "092x5acqnic14cw6vacqap5kgknq3jn4c6jij9zi6j85839jc3xh",
+  "registry+https://github.com/rust-lang/crates.io-index#proc-macro-crate@2.0.0": "1s23imns07vmacn2xjd5hv2h6rr94iqq3fd2frwa6i4h2nk6d0vy",
   "registry+https://github.com/rust-lang/crates.io-index#proc-macro-error-attr@1.0.4": "0sgq6m5jfmasmwwy8x4mjygx5l7kp8s4j60bv25ckv2j1qc41gm1",
   "registry+https://github.com/rust-lang/crates.io-index#proc-macro-error@1.0.4": "1373bhxaf0pagd8zkyd03kkx6bchzf6g0dkwrwzsnal9z47lj9fs",
   "registry+https://github.com/rust-lang/crates.io-index#proc-macro2@1.0.93": "169dw9wch753if1mgyi2nfl1il77gslvh6y2q46qplprwml6m530",
@@ -353,11 +332,7 @@
   "registry+https://github.com/rust-lang/crates.io-index#regex@1.11.1": "148i41mzbx8bmq32hsj1q4karkzzx5m60qza6gdw4pdc9qdyyi5m",
   "registry+https://github.com/rust-lang/crates.io-index#remove_dir_all@0.5.3": "1rzqbsgkmr053bxxl04vmvsd1njyz0nxvly97aip6aa2cmb15k9s",
   "registry+https://github.com/rust-lang/crates.io-index#reqwest@0.11.27": "0qjary4hpplpgdi62d2m0xvbn6lnzckwffm0rgkm2x51023m6ryx",
-  "registry+https://github.com/rust-lang/crates.io-index#reserve-port@2.0.1": "10x21rdb1hjzp6n5flbbw3hfd7brmirckz1q0zsf3a7s5d516f4q",
   "registry+https://github.com/rust-lang/crates.io-index#rsa@0.9.7": "06amqm85raq26v6zg00fbf93lbj3kx559n2lpxc3wrvbbiy5vis7",
-  "registry+https://github.com/rust-lang/crates.io-index#rusqlite@0.32.1": "0vlx040bppl414pbjgbp7qr4jdxwszi9krx0m63zzf2f2whvflvp",
-  "registry+https://github.com/rust-lang/crates.io-index#rusqlite_migration@1.3.1": "076dm65g0sngzrb93r07va4l5zl3gjx9gq5mlsh21p7p0bl44fwj",
-  "registry+https://github.com/rust-lang/crates.io-index#rust-multipart-rfc7578_2@0.6.1": "0mwd3i2mk91n6diaxnkw28vyjbifhrm5ls73pcpfzz8a1i0lidq3",
   "registry+https://github.com/rust-lang/crates.io-index#rustc-demangle@0.1.24": "07zysaafgrkzy2rjgwqdj2a8qdpsm6zv6f5pgpk9x0lm40z9b6vi",
   "registry+https://github.com/rust-lang/crates.io-index#rustc-hash@1.1.0": "1qkc5khrmv5pqi5l5ca9p5nl5hs742cagrndhbrlk3dhlrx3zm08",
   "registry+https://github.com/rust-lang/crates.io-index#rustc_version@0.4.1": "14lvdsmr5si5qbqzrajgb6vfn69k0sfygrvfvr2mps26xwi3mjyg",
@@ -377,10 +352,9 @@
   "registry+https://github.com/rust-lang/crates.io-index#self_cell@1.1.0": "1gmxk5bvnnimcif7v1jk8ai2azfvh9djki545nd86vsnphjgrzf2",
   "registry+https://github.com/rust-lang/crates.io-index#semver@1.0.24": "1fmvjjkd3f64y5fqr1nakkq371mnwzv09fbz5mbmdxril63ypdiw",
   "registry+https://github.com/rust-lang/crates.io-index#serde@0.9.15": "1bsla8l5xr9pp5sirkal6mngxcq6q961km88jvf339j5ff8j7dil",
-  "registry+https://github.com/rust-lang/crates.io-index#serde@1.0.217": "0w2ck1p1ajmrv1cf51qf7igjn2nc51r0izzc00fzmmhkvxjl5z02",
-  "registry+https://github.com/rust-lang/crates.io-index#serde_derive@1.0.217": "180r3rj5gi5s1m23q66cr5wlfgc5jrs6n1mdmql2njnhk37zg6ss",
-  "registry+https://github.com/rust-lang/crates.io-index#serde_json@1.0.136": "1lipcjhh1zazh283i4wsl4l14knh81q2rlkwmag8v8s2rwihqsik",
-  "registry+https://github.com/rust-lang/crates.io-index#serde_path_to_error@0.1.16": "19hlz2359l37ifirskpcds7sxg0gzpqvfilibs7whdys0128i6dg",
+  "registry+https://github.com/rust-lang/crates.io-index#serde@1.0.218": "0q6z4bnrwagnms0bds4886711l6mc68s979i49zd3xnvkg8wkpz8",
+  "registry+https://github.com/rust-lang/crates.io-index#serde_derive@1.0.218": "0azqd74xbpb1v5vf6w1fdbgmwp39ljjfj25cib5rgrzlj7hh75gh",
+  "registry+https://github.com/rust-lang/crates.io-index#serde_json@1.0.139": "19kj3irpa22a7djz1jaf4wambzh7psiqa6zyafqnb76crhx6ry24",
   "registry+https://github.com/rust-lang/crates.io-index#serde_spanned@0.6.8": "1q89g70azwi4ybilz5jb8prfpa575165lmrffd49vmcf76qpqq47",
   "registry+https://github.com/rust-lang/crates.io-index#serde_urlencoded@0.7.1": "1zgklbdaysj3230xivihs30qi5vkhigg323a9m62k8jwf4a1qjfk",
   "registry+https://github.com/rust-lang/crates.io-index#serde_yml@0.0.12": "1p8xwz4znd6fj962y22fdvvv16gb8c0hx4iv5hjplngiidcdvqjr",
@@ -412,14 +386,13 @@
   "registry+https://github.com/rust-lang/crates.io-index#syn@1.0.109": "0ds2if4600bd59wsv7jjgfkayfzy3hnazs394kz6zdkmna8l3dkj",
   "registry+https://github.com/rust-lang/crates.io-index#syn@2.0.96": "102wk3cgawimi3i0q3r3xw3i858zkyingg6y7gsxfy733amsvl6m",
   "registry+https://github.com/rust-lang/crates.io-index#sync_wrapper@0.1.2": "0q01lyj0gr9a93n10nxsn8lwbzq97jqd6b768x17c8f7v7gccir0",
-  "registry+https://github.com/rust-lang/crates.io-index#sync_wrapper@1.0.2": "0qvjyasd6w18mjg5xlaq5jgy84jsjfsvmnn12c13gypxbv75dwhb",
   "registry+https://github.com/rust-lang/crates.io-index#synstructure@0.13.1": "0wc9f002ia2zqcbj0q2id5x6n7g1zjqba7qkg2mr0qvvmdk7dby8",
   "registry+https://github.com/rust-lang/crates.io-index#system-configuration-sys@0.5.0": "1jckxvdr37bay3i9v52izgy52dg690x5xfg3hd394sv2xf4b2px7",
   "registry+https://github.com/rust-lang/crates.io-index#system-configuration@0.5.1": "1rz0r30xn7fiyqay2dvzfy56cvaa3km74hnbz2d72p97bkf3lfms",
   "registry+https://github.com/rust-lang/crates.io-index#system-deps@6.2.2": "0j93ryw031n3h8b0nfpj5xwh3ify636xmv8kxianvlyyipmkbrd3",
   "registry+https://github.com/rust-lang/crates.io-index#target-lexicon@0.12.16": "1cg3bnx1gdkdr5hac1hzxy64fhw4g7dqkd0n3dxy5lfngpr1mi31",
   "registry+https://github.com/rust-lang/crates.io-index#tempdir@0.3.7": "1n5n86zxpgd85y0mswrp5cfdisizq2rv3la906g6ipyc03xvbwhm",
-  "registry+https://github.com/rust-lang/crates.io-index#tempfile@3.15.0": "016pmkbwn3shas44gcwq1kc9lajalb90qafhiip5fvv8h6f5b2ls",
+  "registry+https://github.com/rust-lang/crates.io-index#tempfile@3.17.1": "0c52ggq5vy5mzgk5ly36cgzs1cig3cv6r1jarijmzxgkn6na1r92",
   "registry+https://github.com/rust-lang/crates.io-index#termcolor@1.4.1": "0mappjh3fj3p2nmrg4y7qv94rchwi9mzmgmfflr8p2awdj7lyy86",
   "registry+https://github.com/rust-lang/crates.io-index#thiserror-impl@1.0.69": "1h84fmn2nai41cxbhk6pqf46bxqq1b344v8yz089w1chzi76rvjg",
   "registry+https://github.com/rust-lang/crates.io-index#thiserror-impl@2.0.11": "1hkkn7p2y4cxbffcrprybkj0qy1rl1r6waxmxqvr764axaxc3br6",
@@ -440,14 +413,12 @@
   "registry+https://github.com/rust-lang/crates.io-index#tokio-tungstenite@0.21.0": "0f5wj0crsx74rlll97lhw0wk6y12nhdnqvmnjx002hjn08fmcfy8",
   "registry+https://github.com/rust-lang/crates.io-index#tokio-util@0.7.13": "0y0h10a52c7hrldmr3410bp7j3fadq0jn9nf7awddgd2an6smz6p",
   "registry+https://github.com/rust-lang/crates.io-index#tokio@1.43.0": "17pdm49ihlhfw3rpxix3kdh2ppl1yv7nwp1kxazi5r1xz97zlq9x",
-  "registry+https://github.com/rust-lang/crates.io-index#toml@0.8.2": "0g9ysjaqvm2mv8q85xpqfn7hi710hj24sd56k49wyddvvyq8lp8q",
-  "registry+https://github.com/rust-lang/crates.io-index#toml_datetime@0.6.3": "0jsy7v8bdvmzsci6imj8fzgd255fmy5fzp6zsri14yrry7i77nkw",
+  "registry+https://github.com/rust-lang/crates.io-index#toml@0.8.20": "0j012b37iz1mihksr6a928s6dzszxvblzg3l5wxp7azzsv6sb1yd",
+  "registry+https://github.com/rust-lang/crates.io-index#toml_datetime@0.6.8": "0hgv7v9g35d7y9r2afic58jvlwnf73vgd1mz2k8gihlgrf73bmqd",
   "registry+https://github.com/rust-lang/crates.io-index#toml_edit@0.19.15": "08bl7rp5g6jwmfpad9s8jpw8wjrciadpnbaswgywpr9hv9qbfnqv",
   "registry+https://github.com/rust-lang/crates.io-index#toml_edit@0.20.2": "0f7k5svmxw98fhi28jpcyv7ldr2s3c867pjbji65bdxjpd44svir",
-  "registry+https://github.com/rust-lang/crates.io-index#tower-http@0.6.2": "15wnvhl6cpir9125s73bqjzjsvfb0fmndmsimnl2ddnlhfvs6gs0",
-  "registry+https://github.com/rust-lang/crates.io-index#tower-layer@0.3.3": "03kq92fdzxin51w8iqix06dcfgydyvx7yr6izjq0p626v9n2l70j",
+  "registry+https://github.com/rust-lang/crates.io-index#toml_edit@0.22.24": "0x0lgp70x5cl9nla03xqs5vwwwlrwmd0djkdrp3h3lpdymgpkd0p",
   "registry+https://github.com/rust-lang/crates.io-index#tower-service@0.3.3": "1hzfkvkci33ra94xjx64vv3pp0sq346w06fpkcdwjcid7zhvdycd",
-  "registry+https://github.com/rust-lang/crates.io-index#tower@0.5.2": "1ybmd59nm4abl9bsvy6rx31m4zvzp5rja2slzpn712y9b68ssffh",
   "registry+https://github.com/rust-lang/crates.io-index#tracing-attributes@0.1.28": "0v92l9cxs42rdm4m5hsa8z7ln1xsiw1zc2iil8c6k7lzq0jf2nir",
   "registry+https://github.com/rust-lang/crates.io-index#tracing-core@0.1.33": "170gc7cxyjx824r9kr17zc9gvzx89ypqfdzq259pr56gg5bwjwp6",
   "registry+https://github.com/rust-lang/crates.io-index#tracing@0.1.41": "1l5xrzyjfyayrwhvhldfnwdyligi1mpqm8mzbi2m1d6y6p2hlkkq",
@@ -474,7 +445,6 @@
   "registry+https://github.com/rust-lang/crates.io-index#unsafe-any@0.4.2": "0zwwphsqkw5qaiqmjwngnfpv9ym85qcsyj7adip9qplzjzbn00zk",
   "registry+https://github.com/rust-lang/crates.io-index#url@1.7.2": "0nim1c90mxpi9wgdw2xh8dqd72vlklwlzam436akcrhjac6pqknx",
   "registry+https://github.com/rust-lang/crates.io-index#url@2.5.4": "0q6sgznyy2n4l5lm16zahkisvc9nip9aa5q1pps7656xra3bdy1j",
-  "registry+https://github.com/rust-lang/crates.io-index#urlencoding@2.1.3": "1nj99jp37k47n0hvaz5fvz7z6jd0sb4ppvfy3nphr1zbnyixpy6s",
   "registry+https://github.com/rust-lang/crates.io-index#utf-8@0.7.6": "1a9ns3fvgird0snjkd3wbdhwd3zdpc2h5gpyybrfr6ra5pkqxk09",
   "registry+https://github.com/rust-lang/crates.io-index#utf16_iter@1.0.5": "0ik2krdr73hfgsdzw0218fn35fa09dg2hvbi1xp3bmdfrp9js8y8",
   "registry+https://github.com/rust-lang/crates.io-index#utf8_iter@1.0.4": "1gmna9flnj8dbyd8ba17zigrp9c4c3zclngf5lnb5yvz1ri41hdn",
@@ -492,6 +462,7 @@
   "registry+https://github.com/rust-lang/crates.io-index#warp@0.3.7": "07137zd13lchy5hxpspd0hs6sl19b0fv2zc1chf02nwnzw1d4y23",
   "registry+https://github.com/rust-lang/crates.io-index#wasi@0.10.0+wasi-snapshot-preview1": "07y3l8mzfzzz4cj09c8y90yak4hpsi9g7pllyzpr6xvwrabka50s",
   "registry+https://github.com/rust-lang/crates.io-index#wasi@0.11.0+wasi-snapshot-preview1": "08z4hxwkpdpalxjps1ai9y7ihin26y9f476i53dv98v45gkqg3cw",
+  "registry+https://github.com/rust-lang/crates.io-index#wasi@0.13.3+wasi-0.2.2": "1lnapbvdcvi3kc749wzqvwrpd483win2kicn1faa4dja38p6v096",
   "registry+https://github.com/rust-lang/crates.io-index#wasite@0.1.0": "0nw5h9nmcl4fyf4j5d4mfdjfgvwi1cakpi349wc4zrr59wxxinmq",
   "registry+https://github.com/rust-lang/crates.io-index#wasm-bindgen-backend@0.2.100": "1ihbf1hq3y81c4md9lyh6lcwbx6a5j0fw4fygd423g62lm8hc2ig",
   "registry+https://github.com/rust-lang/crates.io-index#wasm-bindgen-futures@0.4.50": "0q8ymi6i9r3vxly551dhxcyai7nc491mspj0j1wbafxwq074fpam",
@@ -528,11 +499,13 @@
   "registry+https://github.com/rust-lang/crates.io-index#windows_x86_64_msvc@0.48.5": "0f4mdp895kkjh9zv8dxvn4pc10xr7839lf5pa9l0193i2pkgr57d",
   "registry+https://github.com/rust-lang/crates.io-index#windows_x86_64_msvc@0.52.6": "1v7rb5cibyzx8vak29pdrk8nx9hycsjs4w0jgms08qk49jl6v7sq",
   "registry+https://github.com/rust-lang/crates.io-index#winnow@0.5.40": "0xk8maai7gyxda673mmw3pj1hdizy5fpi7287vaywykkk19sk4zm",
+  "registry+https://github.com/rust-lang/crates.io-index#winnow@0.7.3": "1c9bmhpdwbdmll6b4l6skabz0296dchnmnxw84hh2y3ggyllwzqf",
   "registry+https://github.com/rust-lang/crates.io-index#winreg@0.50.0": "1cddmp929k882mdh6i9f2as848f13qqna6czwsqzkh1pqnr5fkjj",
+  "registry+https://github.com/rust-lang/crates.io-index#wit-bindgen-rt@0.33.0": "0g4lwfp9x6a2i1hgjn8k14nr4fsnpd5izxhc75zpi2s5cvcg6s1j",
   "registry+https://github.com/rust-lang/crates.io-index#write16@1.0.0": "0dnryvrrbrnl7vvf5vb1zkmwldhjkf2n5znliviam7bm4900z2fi",
   "registry+https://github.com/rust-lang/crates.io-index#writeable@0.5.5": "0lawr6y0bwqfyayf3z8zmqlhpnzhdx0ahs54isacbhyjwa7g778y",
+  "registry+https://github.com/rust-lang/crates.io-index#xml-rs@0.8.25": "1i73ajf6scni5bi1a51r19xykgrambdx5fkks0fyg5jqqbml1ff5",
   "registry+https://github.com/rust-lang/crates.io-index#yansi-term@0.1.2": "1w8vjlvxba6yvidqdvxddx3crl6z66h39qxj8xi6aqayw2nk0p7y",
-  "registry+https://github.com/rust-lang/crates.io-index#yansi@1.0.1": "0jdh55jyv0dpd38ij4qh60zglbw9aa8wafqai6m0wa7xaxk3mrfg",
   "registry+https://github.com/rust-lang/crates.io-index#yoke-derive@0.7.5": "0m4i4a7gy826bfvnqa9wy6sp90qf0as3wps3wb0smjaamn68g013",
   "registry+https://github.com/rust-lang/crates.io-index#yoke@0.7.5": "0h3znzrdmll0a7sglzf9ji0p5iqml11wrj1dypaf6ad6kbpnl3hj",
   "registry+https://github.com/rust-lang/crates.io-index#zerocopy-derive@0.7.35": "0gnf2ap2y92nwdalzz3x7142f2b83sni66l39vxp2ijd6j080kzs",
diff --git a/flake.lock b/flake.lock
index 7fc0bd8..faf7ce0 100644
--- a/flake.lock
+++ b/flake.lock
@@ -1,5 +1,20 @@
 {
   "nodes": {
+    "crane": {
+      "locked": {
+        "lastModified": 1739936662,
+        "narHash": "sha256-x4syUjNUuRblR07nDPeLDP7DpphaBVbUaSoeZkFbGSk=",
+        "owner": "ipetkov",
+        "repo": "crane",
+        "rev": "19de14aaeb869287647d9461cbd389187d8ecdb7",
+        "type": "github"
+      },
+      "original": {
+        "owner": "ipetkov",
+        "repo": "crane",
+        "type": "github"
+      }
+    },
     "flake-utils": {
       "inputs": {
         "systems": "systems"
@@ -20,16 +35,16 @@
     },
     "nixpkgs": {
       "locked": {
-        "lastModified": 1704732714,
-        "narHash": "sha256-ABqK/HggMYA/jMUXgYyqVAcQ8QjeMyr1jcXfTpSHmps=",
+        "lastModified": 1740339700,
+        "narHash": "sha256-cbrw7EgQhcdFnu6iS3vane53bEagZQy/xyIkDWpCgVE=",
         "owner": "NixOS",
         "repo": "nixpkgs",
-        "rev": "6723fa4e4f1a30d42a633bef5eb01caeb281adc3",
+        "rev": "04ef94c4c1582fd485bbfdb8c4a8ba250e359195",
         "type": "github"
       },
       "original": {
         "id": "nixpkgs",
-        "ref": "nixos-23.11",
+        "ref": "nixos-24.11",
         "type": "indirect"
       }
     },
@@ -50,6 +65,7 @@
     },
     "root": {
       "inputs": {
+        "crane": "crane",
         "nixpkgs": "nixpkgs",
         "typeshare": "typeshare",
         "unstable": "unstable"
diff --git a/flake.nix b/flake.nix
index 375bc5c..9b846f2 100644
--- a/flake.nix
+++ b/flake.nix
@@ -2,12 +2,13 @@
   description = "Lumenescent Dreams Tools";
 
   inputs = {
-    nixpkgs.url = "nixpkgs/nixos-23.11";
+    nixpkgs.url = "nixpkgs/nixos-24.11";
     unstable.url = "nixpkgs/nixos-unstable";
     typeshare.url = "github:1Password/typeshare";
+    crane.url = "github:ipetkov/crane";
   };
 
-  outputs = { self, nixpkgs, unstable, typeshare, ... }:
+  outputs = { self, nixpkgs, unstable, typeshare, crane, ... }:
     let 
       version = builtins.string 0 8 self.lastModifiedDate;
       supportedSystems = [ "x86_64-linux" ];
@@ -57,6 +58,8 @@
       packages."x86_64-linux" =
         let
           pkgs = import nixpkgs { system = "x86_64-linux"; };
+          craneLib = crane.mkLib pkgs;
+          src = craneLib.cleanCargoSource ./.;
 
           gtkNativeInputs = [
             pkgs.pkg-config
@@ -89,6 +92,11 @@
           dashboard = cargo_nix.workspaceMembers.dashboard.build;
           # file-service = cargo_nix.workspaceMembers.file-service.build;
           fitnesstrax = cargo_nix.workspaceMembers.fitnesstrax.build;
+          l10n-db = craneLib.buildPackage {
+            pname = "l10n-db";
+            cargoExtraArgs = "-p l10n-db";
+            src = ./.;
+          };
           otg-gtk = cargo_nix.workspaceMembers.otg-gtk.build;
 
           all = pkgs.symlinkJoin {
@@ -99,6 +107,7 @@
                 dashboard
                 # file-service
                 fitnesstrax
+                l10n-db
                 otg-gtk
             ];
           };
diff --git a/l10n-db/Cargo.toml b/l10n-db/Cargo.toml
new file mode 100644
index 0000000..5a1f64a
--- /dev/null
+++ b/l10n-db/Cargo.toml
@@ -0,0 +1,24 @@
+[package]
+name = "l10n-db"
+version = "0.1.0"
+edition = "2021"
+
+[dependencies]
+chrono = { version = "0.4.39", features = ["serde"] }
+clap = { version = "4.5.30", features = ["derive"] }
+icu_locid = { version = "1.5.0", features = ["serde"] }
+serde = { version = "1.0.218", features = ["derive"] }
+serde_json = "1.0.139"
+tempfile = "3.17.1"
+thiserror = "2.0.11"
+toml = "0.8.20"
+xml-rs = "0.8.25"
+
+# [lib]
+# name = "l10n_db"
+# path = "src/lib.rs"
+# 
+# [[bin]]
+# name = "l10n-db"
+# path = "src/main.rs"
+
diff --git a/l10n-db/config.toml b/l10n-db/config.toml
new file mode 100644
index 0000000..913bdbb
--- /dev/null
+++ b/l10n-db/config.toml
@@ -0,0 +1,8 @@
+db_path = "./i18n"
+base_locale = "en"
+locales = [
+    "en",
+    "eo",
+    "de",
+    "es",
+]
diff --git a/l10n-db/i18n/SaveSettings.toml b/l10n-db/i18n/SaveSettings.toml
new file mode 100644
index 0000000..384705e
--- /dev/null
+++ b/l10n-db/i18n/SaveSettings.toml
@@ -0,0 +1,22 @@
+key = "SaveSettings"
+description = "This is a label on a button which will save the settings when clicked"
+
+[variants.eo]
+locale = "eo"
+content = "Konservi Agordojn"
+modified = "2025-02-24T19:32:11.246639077Z"
+
+[variants.es]
+locale = "es"
+content = "Guardar Configuraciones"
+modified = "2025-02-24T19:33:23.861329923Z"
+
+[variants.en]
+locale = "en"
+content = "Save Settings"
+modified = "2025-02-22T23:44:18.874218939Z"
+
+[variants.de]
+locale = "de"
+content = "Einstellungen Speichern"
+modified = "2025-02-24T19:33:19.516005843Z"
diff --git a/l10n-db/i18n/TimeDistance.toml b/l10n-db/i18n/TimeDistance.toml
new file mode 100644
index 0000000..aa6ffb2
--- /dev/null
+++ b/l10n-db/i18n/TimeDistance.toml
@@ -0,0 +1,22 @@
+key = "TimeDistance"
+description = "A summary of a workout or many workouts that involve a time and a distance"
+
+[variants.es]
+locale = "es"
+content = "{distance} de {activity} en {hours, plural, one {}=1 {{hours} hora} other {{hours} horas}} y {minutes, plural, one {}=1 {{minutes} minuto} other {{minutes} minutos}}"
+modified = "2025-02-24T19:33:23.861604738Z"
+
+[variants.eo]
+locale = "eo"
+content = "{distance} de {activity} en {hours, plural, =1 {{hours} horo} other {{hours} horoj}} {minutes, plural, =1 {{minutes} minuto} other {{minutes} minutoj}}"
+modified = "2025-02-24T19:32:11.246943602Z"
+
+[variants.en]
+locale = "en"
+content = "{distance} of {activity} in {hours, plural, =1 {{hours} hour} other {{hours} hours}} and {minutes, plural, =1 {{minutes} minute} other {{minutes} minutes}}"
+modified = "2025-02-24T14:09:17.361641899Z"
+
+[variants.de]
+locale = "de"
+content = "{distance} von {activity} in {hours, plural, one {}=1 {{hours} Stunde} other {{hours} Stunden}} und {minutes, plural, one {}=1 {{minutes} Minute} other {{minutes} Minuten}}"
+modified = "2025-02-24T19:33:19.516210807Z"
diff --git a/l10n-db/i18n/Welcome.toml b/l10n-db/i18n/Welcome.toml
new file mode 100644
index 0000000..db36ae3
--- /dev/null
+++ b/l10n-db/i18n/Welcome.toml
@@ -0,0 +1,22 @@
+key = "Welcome"
+description = "This is a welcome content that will be shown on first app opening, before configuration."
+
+[variants.en]
+locale = "en"
+content = "Welcome to FitnessTrax, the privacy-centered fitness tracker"
+modified = "2025-02-25T02:12:25.757240004Z"
+
+[variants.eo]
+locale = "eo"
+content = "Bonvenon al FitnessTrax"
+modified = "2025-02-24T19:32:11.246407627Z"
+
+[variants.es]
+locale = "es"
+content = "Bienvenido a FitnessTrax"
+modified = "2025-02-24T19:33:23.861143003Z"
+
+[variants.de]
+locale = "de"
+content = "Willkommen bei FitnessTrax"
+modified = "2025-02-24T19:33:19.515861453Z"
diff --git a/l10n-db/src/bin/l10n-db.rs b/l10n-db/src/bin/l10n-db.rs
new file mode 100644
index 0000000..d02b30e
--- /dev/null
+++ b/l10n-db/src/bin/l10n-db.rs
@@ -0,0 +1,143 @@
+use std::{fmt, path::PathBuf};
+
+use clap::{Parser, Subcommand};
+
+use icu_locid::{langid, LanguageIdentifier};
+use l10n_db::{
+    self, js, read_file,
+    xliff::{self, import_file},
+    Bundle, Editor, ReadError,
+};
+use serde::Deserialize;
+
+#[derive(Parser)]
+#[command(version, about, long_about = None)]
+struct Cli {
+    #[command(subcommand)]
+    command: Option<Commands>,
+}
+
+#[derive(Subcommand)]
+enum Commands {
+    /// Edit, potentially creating, a key
+    EditKey {
+        #[arg(short, long)]
+        key: String,
+        #[arg(short, long)]
+        locale: Option<String>,
+    },
+    /// List al keys in the database
+    ListKeys,
+    // Search the database
+    // Search { },
+    Import {
+        #[arg(short, long)]
+        file: String,
+    },
+    /// Export the database
+    Export {
+        #[arg(short, long)]
+        format: String,
+        #[arg(short, long)]
+        file: String,
+        #[arg(short, long)]
+        locale: Option<String>,
+    },
+    Report,
+}
+
+#[derive(Debug, Deserialize)]
+struct Config {
+    db_path: PathBuf,
+    base_locale: LanguageIdentifier,
+}
+
+fn edit_key(bundle: &mut Bundle, key: String, locale: LanguageIdentifier, editor: &str) {
+    let message = bundle.message(key);
+    Editor::edit(message, locale, editor);
+    bundle.save();
+}
+
+#[derive(Clone, Debug, Default)]
+struct Report {
+    keys: Vec<String>,
+    // source_deleted: Vec<String>,
+    out_of_date: Vec<String>,
+}
+
+impl fmt::Display for Report {
+    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
+        writeln!(f, "{} messages in bundle", self.keys.len())?;
+        writeln!(f, "Out of date messages")?;
+        for key in self.out_of_date.iter() {
+            writeln!(f, "\t{}", key)?;
+        }
+        Ok(())
+    }
+}
+
+fn generate_report(bundle: &Bundle, base_locale: &LanguageIdentifier) -> Report {
+    let mut report: Report = Default::default();
+    for (key, message) in bundle.message_iter() {
+        report.keys.push(key.to_owned());
+        if !message.variants_out_of_date(base_locale).is_empty() {
+            report.out_of_date.push(key.to_owned())
+        }
+    }
+
+    report
+}
+
+fn main() {
+    let editor = std::env::var("EDITOR").expect("Set EDITOR to the path to your favorite editor");
+
+    let config: Config = read_file(&PathBuf::from("./config.toml"))
+        .and_then(|bytes| String::from_utf8(bytes).map_err(|_| ReadError::InvalidFormat))
+        .and_then(|content| toml::from_str(&content).map_err(|_| ReadError::InvalidFormat))
+        .unwrap();
+
+    let cli = Cli::parse();
+
+    let mut bundle = Bundle::load_from_disk(PathBuf::from(&config.db_path));
+
+    match &cli.command {
+        Some(Commands::EditKey { key, locale }) => {
+            let identifier = locale.as_ref()
+                .map(|l| l.parse::<LanguageIdentifier>().unwrap())
+                .unwrap_or(config.base_locale);
+            edit_key(&mut bundle, key.to_owned(), identifier, &editor)
+        }
+        Some(Commands::ListKeys) => {
+            for (key, _) in bundle.message_iter() {
+                println!("{}", key);
+            }
+        }
+        Some(Commands::Import { file }) => {
+            import_file(&mut bundle, &PathBuf::from(file)).unwrap();
+            bundle.save();
+        }
+        Some(Commands::Export {
+            format,
+            file,
+            locale,
+        }) => {
+            let locale = locale
+                .as_ref()
+                .map(|l| l.clone().parse::<LanguageIdentifier>().unwrap())
+                .unwrap_or(langid!("en"));
+
+            match format.as_ref() {
+                "js" => js::export_file(&bundle, locale, &PathBuf::from(file)).unwrap(),
+                "xliff" => {
+                    xliff::export_file(&bundle, locale, &PathBuf::from(file)).unwrap()
+                }
+                _ => todo!(),
+            }
+        }
+        Some(Commands::Report) => {
+            let report = generate_report(&bundle, &config.base_locale);
+            println!("{}", report);
+        }
+        None => {}
+    }
+}
diff --git a/l10n-db/src/bundle.rs b/l10n-db/src/bundle.rs
new file mode 100644
index 0000000..1b55392
--- /dev/null
+++ b/l10n-db/src/bundle.rs
@@ -0,0 +1,143 @@
+use std::{collections::HashMap, fs::File, io::{BufReader, Read, Write}, path::{Path, PathBuf}};
+
+use chrono::{DateTime, Utc};
+use icu_locid::LanguageIdentifier;
+use serde::{Deserialize, Serialize};
+
+pub struct Bundle {
+    path: PathBuf,
+    messages: HashMap<String, Message>,
+}
+
+impl Bundle {
+    pub fn load_from_disk(path: PathBuf) -> Self {
+        let mut messages = HashMap::new();
+        if path.is_dir() {
+            for entry in std::fs::read_dir(&path).unwrap() {
+                let entry = entry.unwrap();
+                let path = entry.path().clone();
+                let key = path.file_stem().unwrap();
+
+                let message = Message::from_file(&entry.path());
+                messages.insert(key.to_str().unwrap().to_owned(), message);
+            }
+        }
+        Self { path, messages }
+    }
+
+    pub fn message_iter(&self) -> impl Iterator<Item = (&String, &Message)> {
+        self.messages.iter()
+    }
+
+    pub fn message(&mut self, name: String) -> &mut Message {
+        self.messages
+            .entry(name.to_owned())
+            .or_insert(Message::new(name.to_owned()))
+    }
+
+    pub fn save(&self) {
+        self.messages.iter().for_each(|(key, value)| {
+            let mut path = self.path.clone();
+            path.push(key);
+            path.set_extension("toml");
+            save_file(&path, toml::to_string(value).unwrap().as_bytes());
+        });
+    }
+}
+
+fn save_file(path: &PathBuf, s: &[u8]) {
+    let mut f = File::create(path).unwrap();
+    let _ = f.write(s).unwrap();
+}
+
+#[derive(Deserialize, Serialize, Debug)]
+pub struct Message {
+    key: String,
+    description: String,
+    variants: HashMap<LanguageIdentifier, Variant>,
+}
+
+impl Message {
+    pub fn new(key: String) -> Self {
+        Self {
+            key,
+            description: "".to_owned(),
+            variants: HashMap::new(),
+        }
+    }
+
+    pub fn from_file(path: &Path) -> Message {
+        let file = std::fs::File::open(path).unwrap();
+        let mut content = Vec::new();
+        let mut reader = BufReader::new(file);
+        let _ = reader.read_to_end(&mut content);
+        toml::from_str(&String::from_utf8(content).unwrap()).unwrap()
+    }
+
+    pub fn set_description(&mut self, desc: String) {
+        self.description = desc;
+    }
+
+    pub fn description(&self) -> &str {
+        &self.description
+    }
+
+    pub fn variant(&self, locale: &LanguageIdentifier) -> Option<&Variant> {
+        self.variants.get(locale)
+    }
+
+    pub fn variant_mut(&mut self, locale: LanguageIdentifier) -> &mut Variant {
+        self.variants.entry(locale.clone()).or_insert(Variant {
+            locale,
+            content: "".to_owned(),
+            modified: Utc::now(),
+        })
+    }
+
+    pub fn variants_out_of_date(
+        &self,
+        base_locale: &LanguageIdentifier,
+    ) -> Vec<LanguageIdentifier> {
+        match self
+            .variants
+            .get(base_locale)
+            .map(|variant| variant.modified())
+        {
+            Some(base_date) => self
+                .variants
+                .iter()
+                .filter(|(_, value)| base_date > value.modified())
+                .map(|(locale, _)| locale.clone())
+                .collect(),
+            None => vec![],
+        }
+    }
+
+    // pub fn missing_variants(&self, locals: Vec<LanguageIdentifier>) -> Vec<LanguageIdentifier> {}
+}
+
+#[derive(Deserialize, Serialize, Debug, Clone)]
+pub struct Variant {
+    locale: LanguageIdentifier,
+    content: String,
+    modified: DateTime<Utc>,
+}
+
+impl Variant {
+    pub fn content(&self) -> &str {
+        &self.content
+    }
+
+    pub fn set_content(&mut self, content: String) {
+        self.content = content;
+        self.modified = Utc::now();
+    }
+
+    pub fn modified(&self) -> DateTime<Utc> {
+        self.modified
+    }
+}
+
+
+#[cfg(test)]
+mod test {}
diff --git a/l10n-db/src/editor.rs b/l10n-db/src/editor.rs
new file mode 100644
index 0000000..b6d9a9f
--- /dev/null
+++ b/l10n-db/src/editor.rs
@@ -0,0 +1,43 @@
+use std::{io::Write, process::Command};
+
+use icu_locid::{langid, LanguageIdentifier};
+use serde::{Deserialize, Serialize};
+
+use crate::{read_fh, Message};
+
+#[derive(Serialize, Deserialize, Debug, Clone)]
+struct EditorMessage {
+    description: String,
+    source: String,
+    content: String,
+}
+
+pub struct Editor {}
+
+impl Editor {
+    pub fn edit(msg: &mut Message, locale: LanguageIdentifier, editor: &str) {
+        let description = msg.description().to_owned();
+        let source_string = msg.variant_mut(langid!("en")).content().to_owned();
+        let variant = msg.variant_mut(locale);
+        // let editable_content = EditorMessage::from_variant(description, &variant);
+        let editable_content = EditorMessage {
+            description,
+            source: source_string,
+            content: variant.content().to_owned(),
+        };
+
+        let mut file = tempfile::NamedTempFile::new().unwrap();
+        let _ = file.write(toml::to_string(&editable_content).unwrap().as_bytes());
+        let _ = file.flush();
+        let mut cmd = Command::new(editor).args([file.path()]).spawn().unwrap();
+        cmd.wait().unwrap();
+        let file = file.reopen().unwrap();
+        let content = read_fh(&file).unwrap();
+
+        let new_variant: EditorMessage =
+            toml::from_str(&String::from_utf8(content).unwrap()).unwrap();
+
+        variant.set_content(new_variant.content);
+        msg.set_description(new_variant.description);
+    }
+}
diff --git a/l10n-db/src/formats/js.rs b/l10n-db/src/formats/js.rs
new file mode 100644
index 0000000..6aa94b2
--- /dev/null
+++ b/l10n-db/src/formats/js.rs
@@ -0,0 +1,22 @@
+use std::{collections::BTreeMap, fs::File, io::Write, path::Path};
+
+use icu_locid::LanguageIdentifier;
+
+use crate::{Bundle, WriteError};
+
+pub fn export_file(bundle: &Bundle, locale: LanguageIdentifier, path: &Path) -> Result<(), WriteError> {
+    let mut file = File::create(path).unwrap();
+    export_fh(bundle, locale, &mut file)
+}
+
+pub fn export_fh(bundle: &Bundle, locale: LanguageIdentifier, fh: &mut File) -> Result<(), WriteError> {
+    let messages = bundle.message_iter().map(|(key, message)| {
+        let content = message.variant(&locale).unwrap().content().to_owned();
+        (key.to_owned(), content)
+    }).collect::<Vec<(String, String)>>();
+
+    let messages: BTreeMap<String, String> = messages.into_iter().collect();
+    let _ = fh.write(serde_json::to_string_pretty(&messages).unwrap().as_bytes()).unwrap();
+
+    Ok(())
+}
diff --git a/l10n-db/src/formats/mod.rs b/l10n-db/src/formats/mod.rs
new file mode 100644
index 0000000..6fbd8ce
--- /dev/null
+++ b/l10n-db/src/formats/mod.rs
@@ -0,0 +1,4 @@
+pub mod js;
+pub mod xliff;
+
+
diff --git a/l10n-db/src/formats/xliff.rs b/l10n-db/src/formats/xliff.rs
new file mode 100644
index 0000000..d05ecaa
--- /dev/null
+++ b/l10n-db/src/formats/xliff.rs
@@ -0,0 +1,146 @@
+use std::{
+    fs::File,
+    io::{BufReader, Read, Write},
+    path::Path,
+};
+
+use icu_locid::{langid, LanguageIdentifier};
+use xml::{attribute::OwnedAttribute, reader, writer, EmitterConfig, EventReader, EventWriter};
+
+use crate::{Bundle, Message, ReadError, WriteError};
+
+pub fn export_file(
+    bundle: &Bundle,
+    locale: LanguageIdentifier,
+    path: &Path,
+) -> Result<(), WriteError> {
+    let mut file = File::create(path).unwrap();
+    export_fh(bundle, locale, &mut file)
+}
+
+pub fn export_fh<W>(bundle: &Bundle, locale: LanguageIdentifier, fh: W) -> Result<(), WriteError>
+where
+    W: Write,
+{
+    let mut writer = EmitterConfig::new().perform_indent(true).create_writer(fh);
+
+    writer
+        .write(
+            writer::XmlEvent::start_element("xliff")
+                .attr("xmlns", "urn:oasis:names:tc:xliff:document:2.0")
+                .attr("version", "2.0")
+                .attr("srcLang", &format!("{}", locale)),
+        )
+        .unwrap();
+    writer
+        .write(writer::XmlEvent::start_element("file").attr("id", "main"))
+        .unwrap();
+
+    for (key, message) in bundle.message_iter() {
+        write_message(&mut writer, key, message, &locale);
+    }
+
+    writer.write(writer::XmlEvent::end_element()).unwrap();
+    writer.write(writer::XmlEvent::end_element()).unwrap();
+    Ok(())
+}
+
+pub fn import_file(bundle: &mut Bundle, path: &Path) -> Result<(), ReadError> {
+    let file = File::open(path).unwrap();
+    let file = BufReader::new(file);
+
+    import_reader(bundle, file)
+}
+
+pub fn import_reader<R>(bundle: &mut Bundle, fh: R) -> Result<(), ReadError>
+where
+    R: Read,
+{
+    let parser = EventReader::new(fh);
+
+    let mut locale: LanguageIdentifier = langid!("en");
+    let mut current_key = None;
+    let mut current_text: Option<String> = None;
+    let mut in_target = false;
+
+    for event in parser {
+        match event {
+            Ok(reader::XmlEvent::StartElement {
+                name, attributes, ..
+            }) => match name.local_name.as_ref() {
+                "xliff" => {
+                    locale = find_attribute(&attributes, "trgLang")
+                        .unwrap()
+                        .parse::<LanguageIdentifier>()
+                        .unwrap();
+                }
+                "unit" => current_key = find_attribute(&attributes, "id"),
+                "target" => in_target = true,
+                _ => println!("name: {}", name),
+            },
+            Ok(reader::XmlEvent::EndElement { name }) => match name.local_name.as_ref() {
+                "unit" => {
+                    if let Some(key) = current_key {
+                        let message = bundle.message(key);
+                        let variant = message.variant_mut(locale.clone());
+                        if let Some(ref text) = current_text {
+                            variant.set_content(text.clone());
+                        }
+                    }
+                    current_key = None;
+                }
+                "target" => in_target = false,
+                _ => {}
+            },
+            Ok(reader::XmlEvent::Characters(data)) => {
+                if in_target {
+                    current_text = Some(data)
+                }
+            }
+            Err(e) => {
+                eprintln!("error: {e}");
+                break;
+            }
+            _ => {}
+        }
+    }
+
+    Ok(())
+}
+
+fn write_message<T>(
+    writer: &mut EventWriter<T>,
+    key: &str,
+    message: &Message,
+    locale: &LanguageIdentifier,
+) where
+    T: std::io::Write,
+{
+    [
+        writer::XmlEvent::start_element("unit")
+            .attr("id", key)
+            .into(),
+        writer::XmlEvent::start_element("notes").into(),
+        writer::XmlEvent::start_element("note").into(),
+        writer::XmlEvent::characters(message.description()),
+        writer::XmlEvent::end_element().into(),
+        writer::XmlEvent::end_element().into(),
+        writer::XmlEvent::start_element("segment").into(),
+        writer::XmlEvent::start_element("source").into(),
+        writer::XmlEvent::characters(message.variant(locale).unwrap().content()),
+        writer::XmlEvent::end_element().into(),
+        writer::XmlEvent::end_element().into(),
+        writer::XmlEvent::end_element().into(),
+    ]
+    .into_iter()
+    .for_each(|elem: writer::XmlEvent| writer.write(elem).unwrap());
+}
+
+fn find_attribute(attrs: &Vec<OwnedAttribute>, name: &str) -> Option<String> {
+    for f in attrs {
+        if name == f.name.local_name {
+            return Some(f.value.clone());
+        }
+    }
+    None
+}
diff --git a/l10n-db/src/lib.rs b/l10n-db/src/lib.rs
new file mode 100644
index 0000000..2268a86
--- /dev/null
+++ b/l10n-db/src/lib.rs
@@ -0,0 +1,11 @@
+mod bundle;
+pub use bundle::{Bundle, Message, Variant};
+
+mod editor;
+pub use editor::Editor;
+
+mod formats;
+pub use formats::{js, xliff};
+
+mod utils;
+pub use utils::*;
diff --git a/l10n-db/src/utils.rs b/l10n-db/src/utils.rs
new file mode 100644
index 0000000..82435a0
--- /dev/null
+++ b/l10n-db/src/utils.rs
@@ -0,0 +1,37 @@
+use std::{fs::File, io::{BufReader, ErrorKind, Read}, path::Path};
+use thiserror::Error;
+
+#[derive(Debug, Error)]
+pub enum ReadError {
+    #[error("file not found")]
+    FileNotFound,
+
+    #[error("invalid file format")]
+    InvalidFormat,
+
+    #[error("unhandled read error")]
+    Unhandled(ErrorKind),
+}
+
+pub fn read_file(path: &Path) -> Result<Vec<u8>, ReadError> {
+    let file = File::open(path).map_err(|err| {
+        match err.kind() {
+            ErrorKind::NotFound => ReadError::FileNotFound,
+            _ => ReadError::Unhandled( err.kind()),
+        }
+    })?;
+    read_fh(&file)
+}
+
+pub fn read_fh(file: &File) -> Result<Vec<u8>, ReadError> {
+    let mut content = Vec::new();
+    let mut reader = BufReader::new(file);
+    reader.read_to_end(&mut content).map_err(|err| ReadError::Unhandled(err.kind()))?;
+    Ok(content)
+}
+
+#[derive(Debug, Error)]
+pub enum WriteError {
+    #[error("unhandled write error")]
+    Unhandled(ErrorKind),
+}
diff --git a/pico-st7789/.cargo/config.toml b/pico-st7789/.cargo/config.toml
new file mode 100644
index 0000000..9585789
--- /dev/null
+++ b/pico-st7789/.cargo/config.toml
@@ -0,0 +1,11 @@
+[build]
+target = "thumbv6m-none-eabi"
+
+[target.thumbv6m-none-eabi]
+rustflags = [
+    "-C", "link-arg=--nmagic",
+    "-C", "link-arg=-Tlink.x",
+    "-C", "no-vectorize-loops",
+]
+
+runner = "elf2uf2-rs -d"
diff --git a/pico-st7789/Cargo.toml b/pico-st7789/Cargo.toml
new file mode 100644
index 0000000..ff98242
--- /dev/null
+++ b/pico-st7789/Cargo.toml
@@ -0,0 +1,13 @@
+[package]
+name = "pico-st7789"
+version = "0.1.0"
+edition = "2021"
+
+[dependencies]
+bitflags = "2.9.0"
+cortex-m-rt = "0.7.3"
+embedded-alloc = "0.6.0"
+embedded-hal = "1.0.0"
+fugit = "0.3.7"
+panic-halt = "1.0.0"
+rp-pico = "0.9.0"
diff --git a/pico-st7789/src/canvas.rs b/pico-st7789/src/canvas.rs
new file mode 100644
index 0000000..7a16821
--- /dev/null
+++ b/pico-st7789/src/canvas.rs
@@ -0,0 +1,62 @@
+//   a a a
+// f       b
+// f       b
+// f       b
+//   g g g
+// e       c
+// e       c
+// e       c
+//   d d d
+
+use crate::font::{Font, Glyph};
+
+pub struct RGB {
+    pub r: u8,
+    pub g: u8,
+    pub b: u8,
+}
+
+pub trait Canvas {
+    fn set_pixel(&mut self, x: usize, y: usize, color: &RGB);
+
+    fn fill(&mut self, x1: usize, y1: usize, x2: usize, y2: usize, color: &RGB) {
+        for x in x1..x2 + 1 {
+            for y in y1..y2 + 1 {
+                self.set_pixel(x, y, color);
+            }
+        }
+    }
+
+    fn square(&mut self, x1: usize, y1: usize, x2: usize, y2: usize, color: &RGB) {
+        for x in x1..x2 + 1 {
+            self.set_pixel(x, y1, color);
+        }
+        for x in x1..x2 + 1 {
+            self.set_pixel(x, y2, color);
+        }
+        for y in y1..y2 + 1 {
+            self.set_pixel(x1, y, color);
+        }
+        for y in y1..y2 + 1 {
+            self.set_pixel(x2, y, color);
+        }
+    }
+}
+
+pub fn print<A>(
+    canvas: &mut impl Canvas,
+    font: &impl Font<A>,
+    sx: usize,
+    sy: usize,
+    text: &str,
+    color: &RGB,
+) where
+    A: Glyph,
+{
+    let mut x = sx;
+    for c in text.chars().map(|c| font.glyph(c)) {
+        c.draw(canvas, x, sy, color);
+        let (dx, _) = c.extents();
+        x = x + dx + 1;
+    }
+}
diff --git a/pico-st7789/src/font/bits_5_8.rs b/pico-st7789/src/font/bits_5_8.rs
new file mode 100644
index 0000000..ecada60
--- /dev/null
+++ b/pico-st7789/src/font/bits_5_8.rs
@@ -0,0 +1,513 @@
+use alloc::collections::btree_map::BTreeMap;
+
+use crate::canvas::{Canvas, RGB};
+
+use super::{Font, Glyph};
+
+pub struct BitmapGlyph([u8; 7]);
+
+pub struct BitmapFont(BTreeMap<char, BitmapGlyph>);
+
+impl BitmapFont {
+    pub fn new() -> Self {
+        let mut font = BTreeMap::new();
+        font.insert(' ', BitmapGlyph([
+            0b00000,
+            0b00000,
+            0b00000,
+            0b00000,
+            0b00000,
+            0b00000,
+            0b00000,
+        ]));
+        font.insert(
+            ':',
+            BitmapGlyph([
+                0b00000,
+                0b00100,
+                0b00100,
+                0b00000,
+                0b00100,
+                0b00100,
+                0b00000,
+            ]),
+        );
+        font.insert(
+            '/',
+            BitmapGlyph([
+                0b00001,
+                0b00010,
+                0b00010,
+                0b00100,
+                0b01000,
+                0b01000,
+                0b10000,
+            ]),
+        );
+        font.insert(
+            '0',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b10001,
+                0b10101,
+                0b10001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            '1',
+            BitmapGlyph([
+                0b00001,
+                0b00011,
+                0b00101,
+                0b00001,
+                0b00001,
+                0b00001,
+                0b00001,
+            ]),
+        );
+        font.insert(
+            '2',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b00001,
+                0b00010,
+                0b00100,
+                0b01000,
+                0b11111,
+            ]),
+        );
+        font.insert(
+            '3',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b00001,
+                0b01110,
+                0b00001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            '4',
+            BitmapGlyph([
+                0b10001,
+                0b10001,
+                0b10001,
+                0b11111,
+                0b00001,
+                0b00001,
+                0b00001,
+            ]),
+        );
+        font.insert(
+            '5',
+            BitmapGlyph([
+                0b11111,
+                0b10000,
+                0b10000,
+                0b11111,
+                0b00001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            '6',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b10000,
+                0b11110,
+                0b10001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            '7',
+            BitmapGlyph([
+                0b11111,
+                0b00001,
+                0b00010,
+                0b00110,
+                0b00100,
+                0b01000,
+                0b01000,
+            ]),
+        );
+        font.insert(
+            '8',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b10001,
+                0b01110,
+                0b10001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            '9',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b10001,
+                0b01111,
+                0b00001,
+                0b00001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            'A',
+            BitmapGlyph([
+                0b00100,
+                0b01010,
+                0b01010,
+                0b10001,
+                0b11111,
+                0b10001,
+                0b10001,
+            ]),
+        );
+        font.insert(
+            'B',
+            BitmapGlyph([
+                0b11110,
+                0b10001,
+                0b10001,
+                0b1111 ,
+                0b10001,
+                0b10001,
+                0b11110,
+            ]),
+        );
+        font.insert(
+            'C',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b10000,
+                0b10000,
+                0b10000,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            'D',
+            BitmapGlyph([
+                0b11110,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b11110,
+            ]),
+        );
+        font.insert(
+            'E',
+            BitmapGlyph([
+                0b11111,
+                0b10000,
+                0b10000,
+                0b11111,
+                0b10000,
+                0b10000,
+                0b11111,
+            ]),
+        );
+        font.insert(
+            'F',
+            BitmapGlyph([
+                0b11111,
+                0b10000,
+                0b10000,
+                0b11110,
+                0b10000,
+                0b10000,
+                0b10000,
+            ]),
+        );
+        font.insert(
+            'G',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b10000,
+                0b10011,
+                0b10001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            'H',
+            BitmapGlyph([
+                0b10001,
+                0b10001,
+                0b10001,
+                0b11111,
+                0b10001,
+                0b10001,
+                0b10001,
+            ]),
+        );
+        font.insert(
+            'I',
+            BitmapGlyph([
+                0b11111,
+                0b00100,
+                0b00100,
+                0b00100,
+                0b00100,
+                0b00100,
+                0b11111,
+            ]),
+        );
+        font.insert(
+            'J',
+            BitmapGlyph([
+                0b00111,
+                0b00001,
+                0b00001,
+                0b00001,
+                0b00001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            'K',
+            BitmapGlyph([
+                0b10001,
+                0b10010,
+                0b10100,
+                0b11000,
+                0b10100,
+                0b10010,
+                0b10001,
+            ]),
+        );
+        font.insert(
+            'L',
+            BitmapGlyph([
+                0b10000,
+                0b10000,
+                0b10000,
+                0b10000,
+                0b10000,
+                0b10000,
+                0b11111,
+            ]),
+        );
+        font.insert(
+            'M',
+            BitmapGlyph([
+                0b10001,
+                0b11011,
+                0b10101,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+            ]),
+        );
+        font.insert(
+            'N',
+            BitmapGlyph([
+                0b10001,
+                0b11001,
+                0b11001,
+                0b10101,
+                0b10011,
+                0b10011,
+                0b10001,
+            ]),
+        );
+        font.insert(
+            'O',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            'P',
+            BitmapGlyph([
+                0b11110,
+                0b10001,
+                0b10001,
+                0b11110,
+                0b10000,
+                0b10000,
+                0b10000,
+            ]),
+        );
+        font.insert(
+            'Q',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10101,
+                0b10011,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            'R',
+            BitmapGlyph([
+                0b11110,
+                0b10001,
+                0b10001,
+                0b11110,
+                0b10100,
+                0b10010,
+                0b10001,
+            ]),
+        );
+        font.insert(
+            'S',
+            BitmapGlyph([
+                0b01110,
+                0b10001,
+                0b10000,
+                0b01110,
+                0b00001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            'T',
+            BitmapGlyph([
+                0b11111,
+                0b00100,
+                0b00100,
+                0b00100,
+                0b00100,
+                0b00100,
+                0b00100,
+            ]),
+        );
+        font.insert(
+            'U',
+            BitmapGlyph([
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b01110,
+            ]),
+        );
+        font.insert(
+            'V',
+            BitmapGlyph([
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b01010,
+                0b01010,
+                0b00100,
+            ]),
+        );
+        font.insert(
+            'W',
+            BitmapGlyph([
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10001,
+                0b10101,
+                0b10101,
+                0b01010,
+            ]),
+        );
+        font.insert(
+            'X',
+            BitmapGlyph([
+                0b10001,
+                0b10001,
+                0b01010,
+                0b00100,
+                0b01010,
+                0b10001,
+                0b10001,
+            ]),
+        );
+        font.insert(
+            'Y',
+            BitmapGlyph([
+                0b10001,
+                0b10001,
+                0b01010,
+                0b00100,
+                0b00100,
+                0b00100,
+                0b00100,
+            ]),
+        );
+        font.insert(
+            'Z',
+            BitmapGlyph([
+                0b11111,
+                0b00001,
+                0b00010,
+                0b00100,
+                0b01000,
+                0b10000,
+                0b11111,
+            ]),
+        );
+        Self(font)
+    }
+}
+
+impl Font<BitmapGlyph> for BitmapFont {
+    fn glyph(&self, c: char) -> &BitmapGlyph {
+        self.0.get(&c).unwrap_or(self.0.get(&' ').unwrap())
+    }
+}
+
+impl Glyph for BitmapGlyph {
+    fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB) {
+        for row in 0..7 {
+            if self.0[row] & (1 << 4) > 0 {
+                canvas.set_pixel(x, y + row, color);
+            }
+            if self.0[row] & (1 << 3) > 0 {
+                canvas.set_pixel(x + 1, y + row, color);
+            }
+            if self.0[row] & (1 << 2) > 0 {
+                canvas.set_pixel(x + 2, y + row, color);
+            }
+            if self.0[row] & (1 << 1) > 0 {
+                canvas.set_pixel(x + 3, y + row, color);
+            }
+            if self.0[row] & 1 > 0 {
+                canvas.set_pixel(x + 4, y + row, color);
+            }
+        }
+    }
+
+    fn extents(&self) -> (usize, usize) {
+        (5, 7)
+    }
+}
diff --git a/pico-st7789/src/font/mod.rs b/pico-st7789/src/font/mod.rs
new file mode 100644
index 0000000..f20065b
--- /dev/null
+++ b/pico-st7789/src/font/mod.rs
@@ -0,0 +1,20 @@
+use crate::canvas::{Canvas, RGB};
+
+mod bits_5_8;
+pub use bits_5_8::BitmapFont;
+
+mod seven_segment;
+pub use seven_segment::SevenSegmentFont;
+
+mod sixteen_segment;
+pub use sixteen_segment::SixteenSegmentFont;
+
+pub trait Font<A> {
+    fn glyph(&self, c: char) -> &A;
+}
+
+pub trait Glyph {
+    fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB);
+    fn extents(&self) -> (usize, usize);
+}
+
diff --git a/pico-st7789/src/font/seven_segment.rs b/pico-st7789/src/font/seven_segment.rs
new file mode 100644
index 0000000..94dd8cd
--- /dev/null
+++ b/pico-st7789/src/font/seven_segment.rs
@@ -0,0 +1,211 @@
+use alloc::collections::btree_map::BTreeMap;
+use bitflags::bitflags;
+
+use crate::canvas::{Canvas, RGB};
+
+use super::{Font, Glyph};
+
+// Seven Segments
+//
+//   a a a
+// f       b
+// f       b
+// f       b
+//   g g g
+// e       c
+// e       c
+// e       c
+//   d d d
+
+bitflags! {
+    pub struct SevenSegmentGlyph: u8 {
+        const NONE = 0;
+        const A = 0x01;
+        const B = 0x01 << 1;
+        const C = 0x01 << 2;
+        const D = 0x01 << 3;
+        const E = 0x01 << 4;
+        const F = 0x01 << 5;
+        const G = 0x01 << 6;
+        const DOT = 0x01 << 7;
+    }
+}
+
+pub struct SevenSegmentFont(BTreeMap<char, SevenSegmentGlyph>);
+
+impl SevenSegmentFont {
+    pub fn new() -> Self {
+        let mut font = BTreeMap::new();
+        font.insert(' ', SevenSegmentGlyph::NONE);
+        font.insert(
+            '0',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::B
+                | SevenSegmentGlyph::C
+                | SevenSegmentGlyph::D
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::F,
+        );
+        font.insert('1', SevenSegmentGlyph::B | SevenSegmentGlyph::C);
+        font.insert(
+            '2',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::B
+                | SevenSegmentGlyph::D
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::G,
+        );
+        font.insert(
+            '3',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::B
+                | SevenSegmentGlyph::C
+                | SevenSegmentGlyph::G
+                | SevenSegmentGlyph::D,
+        );
+        font.insert(
+            '4',
+            SevenSegmentGlyph::B
+                | SevenSegmentGlyph::C
+                | SevenSegmentGlyph::F
+                | SevenSegmentGlyph::G,
+        );
+        font.insert(
+            '5',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::C
+                | SevenSegmentGlyph::D
+                | SevenSegmentGlyph::F
+                | SevenSegmentGlyph::G,
+        );
+        font.insert(
+            '6',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::F
+                | SevenSegmentGlyph::G
+                | SevenSegmentGlyph::C
+                | SevenSegmentGlyph::D,
+        );
+        font.insert(
+            '7',
+            SevenSegmentGlyph::A | SevenSegmentGlyph::B | SevenSegmentGlyph::C,
+        );
+        font.insert(
+            '8',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::B
+                | SevenSegmentGlyph::C
+                | SevenSegmentGlyph::D
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::F
+                | SevenSegmentGlyph::G,
+        );
+        font.insert(
+            '9',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::B
+                | SevenSegmentGlyph::C
+                | SevenSegmentGlyph::F
+                | SevenSegmentGlyph::G,
+        );
+        font.insert(
+            'A',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::B
+                | SevenSegmentGlyph::C
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::F
+                | SevenSegmentGlyph::G,
+        );
+        font.insert(
+            'B',
+            SevenSegmentGlyph::C
+                | SevenSegmentGlyph::D
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::F
+                | SevenSegmentGlyph::G,
+        );
+        font.insert(
+            'C',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::D
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::F,
+        );
+        font.insert(
+            'D',
+            SevenSegmentGlyph::B
+                | SevenSegmentGlyph::C
+                | SevenSegmentGlyph::D
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::G,
+        );
+        font.insert(
+            'E',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::D
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::F
+                | SevenSegmentGlyph::G,
+        );
+        font.insert(
+            'F',
+            SevenSegmentGlyph::A
+                | SevenSegmentGlyph::E
+                | SevenSegmentGlyph::F
+                | SevenSegmentGlyph::G,
+        );
+        Self(font)
+    }
+}
+
+impl Font<SevenSegmentGlyph> for SevenSegmentFont {
+    fn glyph(&self, c: char) -> &SevenSegmentGlyph {
+        self.0.get(&c).unwrap()
+    }
+}
+
+impl Glyph for SevenSegmentGlyph {
+    fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB) {
+        if self.contains(SevenSegmentGlyph::A) {
+            canvas.set_pixel(x + 1, y, color);
+            canvas.set_pixel(x + 2, y, color);
+            canvas.set_pixel(x + 3, y, color);
+        }
+        if self.contains(SevenSegmentGlyph::B) {
+            canvas.set_pixel(x + 4, y + 1, color);
+            canvas.set_pixel(x + 4, y + 2, color);
+            canvas.set_pixel(x + 4, y + 3, color);
+        }
+        if self.contains(SevenSegmentGlyph::C) {
+            canvas.set_pixel(x + 4, y + 5, color);
+            canvas.set_pixel(x + 4, y + 6, color);
+            canvas.set_pixel(x + 4, y + 7, color);
+        }
+        if self.contains(SevenSegmentGlyph::D) {
+            canvas.set_pixel(x + 1, y + 8, color);
+            canvas.set_pixel(x + 2, y + 8, color);
+            canvas.set_pixel(x + 3, y + 8, color);
+        }
+        if self.contains(SevenSegmentGlyph::E) {
+            canvas.set_pixel(x, y + 5, color);
+            canvas.set_pixel(x, y + 6, color);
+            canvas.set_pixel(x, y + 7, color);
+        }
+        if self.contains(SevenSegmentGlyph::F) {
+            canvas.set_pixel(x, y + 1, color);
+            canvas.set_pixel(x, y + 2, color);
+            canvas.set_pixel(x, y + 3, color);
+        }
+        if self.contains(SevenSegmentGlyph::G) {
+            canvas.set_pixel(x + 1, y + 4, color);
+            canvas.set_pixel(x + 2, y + 4, color);
+            canvas.set_pixel(x + 3, y + 4, color);
+        }
+    }
+
+    fn extents(&self) -> (usize, usize) {
+        (5, 8)
+    }
+}
diff --git a/pico-st7789/src/font/sixteen_segment.rs b/pico-st7789/src/font/sixteen_segment.rs
new file mode 100644
index 0000000..eb8cb29
--- /dev/null
+++ b/pico-st7789/src/font/sixteen_segment.rs
@@ -0,0 +1,689 @@
+use alloc::collections::btree_map::BTreeMap;
+use bitflags::bitflags;
+
+use crate::canvas::{Canvas, RGB};
+
+use super::{Font, Glyph};
+
+// Sixteen Segments
+// https://www.partsnotincluded.com/segmented-led-display-ascii-library/
+//
+//   a1 a2
+// f h i  jb
+// f   i j b
+// f  hij  b
+//   g1 g2
+// e  klm  c
+// e k l m c
+// ek  l  mc
+//   d1 d2
+
+bitflags! {
+    pub struct SixteenSegmentGlyph: u32 {
+        const NONE = 0;
+        const A1 = 0x0001;
+        const A2 = 0x0001 << 1;
+        const B = 0x0001 << 2;
+        const C = 0x0001 << 3;
+        const D1 = 0x0001 << 4;
+        const D2 = 0x0001 << 5;
+        const E = 0x0001 << 6;
+        const F = 0x0001 << 7;
+        const G1 = 0x0001 << 8;
+        const G2 = 0x0001 << 9;
+        const H = 0x0001 << 10;
+        const I = 0x0001 << 11;
+        const J = 0x0001 << 12;
+        const K = 0x0001 << 13;
+        const L = 0x0001 << 14;
+        const M = 0x0001 << 15;
+        const DOT = 0x0001 << 16;
+    }
+}
+
+pub struct SixteenSegmentFont(BTreeMap<char, SixteenSegmentGlyph>);
+
+impl SixteenSegmentFont {
+    pub fn new() -> Self {
+        let mut font = BTreeMap::new();
+        font.insert(' ', SixteenSegmentGlyph::NONE);
+        font.insert(
+            '!',
+            SixteenSegmentGlyph::B | SixteenSegmentGlyph::C | SixteenSegmentGlyph::DOT,
+        );
+        font.insert('"', SixteenSegmentGlyph::B | SixteenSegmentGlyph::I);
+        font.insert(
+            '0',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::J
+                | SixteenSegmentGlyph::K,
+        );
+        font.insert(
+            '1',
+            SixteenSegmentGlyph::B | SixteenSegmentGlyph::C | SixteenSegmentGlyph::J,
+        );
+        font.insert(
+            '2',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            '3',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            '4',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            '5',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert(
+            '6',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            '7',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C,
+        );
+        font.insert(
+            '8',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            '9',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            'A',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            'B',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::G2
+                | SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'C',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F,
+        );
+        font.insert(
+            'D',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'E',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1,
+        );
+        font.insert(
+            'F',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1,
+        );
+        font.insert(
+            'G',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            'H',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            'I',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'J',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E,
+        );
+        font.insert(
+            'K',
+            SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::J
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert(
+            'L',
+            SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F,
+        );
+        font.insert(
+            'M',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::H
+                | SixteenSegmentGlyph::J,
+        );
+        font.insert(
+            'N',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::H
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert(
+            'O',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F,
+        );
+        font.insert(
+            'P',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            'Q',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert(
+            'R',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert(
+            'S',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            'T',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'U',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F,
+        );
+        font.insert(
+            'V',
+            SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::J
+                | SixteenSegmentGlyph::K,
+        );
+        font.insert(
+            'W',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::K
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert(
+            'X',
+            SixteenSegmentGlyph::H
+                | SixteenSegmentGlyph::J
+                | SixteenSegmentGlyph::K
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert(
+            'Y',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            'Z',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::J
+                | SixteenSegmentGlyph::K,
+        );
+
+        font.insert(
+            'a',
+            SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'b',
+            SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'c',
+            SixteenSegmentGlyph::D1 | SixteenSegmentGlyph::E | SixteenSegmentGlyph::G1,
+        );
+        font.insert(
+            'd',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::G2
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'e',
+            SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::K,
+        );
+        font.insert(
+            'f',
+            SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2
+                | SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'g',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'h',
+            SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert('i', SixteenSegmentGlyph::L);
+        font.insert(
+            'j',
+            SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'k',
+            SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::J
+                | SixteenSegmentGlyph::L
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert('l', SixteenSegmentGlyph::E | SixteenSegmentGlyph::F);
+        font.insert(
+            'm',
+            SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::L
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            'n',
+            SixteenSegmentGlyph::E | SixteenSegmentGlyph::L | SixteenSegmentGlyph::G1,
+        );
+        font.insert(
+            'o',
+            SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            'p',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::I,
+        );
+        font.insert(
+            'q',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert('r', SixteenSegmentGlyph::E | SixteenSegmentGlyph::G1);
+        font.insert(
+            's',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1
+                | SixteenSegmentGlyph::L,
+        );
+        font.insert(
+            't',
+            SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G1,
+        );
+        font.insert(
+            'u',
+            SixteenSegmentGlyph::D1 | SixteenSegmentGlyph::E | SixteenSegmentGlyph::L,
+        );
+        font.insert('v', SixteenSegmentGlyph::E | SixteenSegmentGlyph::K);
+        font.insert(
+            'w',
+            SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::K
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert(
+            'x',
+            SixteenSegmentGlyph::H
+                | SixteenSegmentGlyph::J
+                | SixteenSegmentGlyph::K
+                | SixteenSegmentGlyph::M,
+        );
+        font.insert(
+            'y',
+            SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::C
+                | SixteenSegmentGlyph::I
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::G2,
+        );
+        font.insert(
+            'z',
+            SixteenSegmentGlyph::D1 | SixteenSegmentGlyph::G1 | SixteenSegmentGlyph::K,
+        );
+        font.insert(
+            '@',
+            SixteenSegmentGlyph::A1
+                | SixteenSegmentGlyph::A2
+                | SixteenSegmentGlyph::B
+                | SixteenSegmentGlyph::D1
+                | SixteenSegmentGlyph::D2
+                | SixteenSegmentGlyph::E
+                | SixteenSegmentGlyph::F
+                | SixteenSegmentGlyph::G2,
+        );
+        Self(font)
+    }
+}
+
+impl Font<SixteenSegmentGlyph> for SixteenSegmentFont {
+    fn glyph(&self, c: char) -> &SixteenSegmentGlyph {
+        self.0.get(&c).unwrap_or(&SixteenSegmentGlyph::NONE)
+    }
+}
+
+impl Glyph for SixteenSegmentGlyph {
+    fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB) {
+        if self.contains(SixteenSegmentGlyph::A1) {
+            canvas.set_pixel(x + 1, y, color);
+            canvas.set_pixel(x + 2, y, color);
+            canvas.set_pixel(x + 3, y, color);
+        }
+        if self.contains(SixteenSegmentGlyph::A2) {
+            canvas.set_pixel(x + 5, y, color);
+            canvas.set_pixel(x + 6, y, color);
+            canvas.set_pixel(x + 7, y, color);
+        }
+        if self.contains(SixteenSegmentGlyph::B) {
+            canvas.set_pixel(x + 8, y + 1, color);
+            canvas.set_pixel(x + 8, y + 2, color);
+            canvas.set_pixel(x + 8, y + 3, color);
+            canvas.set_pixel(x + 8, y + 4, color);
+            canvas.set_pixel(x + 8, y + 5, color);
+        }
+        if self.contains(SixteenSegmentGlyph::C) {
+            canvas.set_pixel(x + 8, y + 7, color);
+            canvas.set_pixel(x + 8, y + 8, color);
+            canvas.set_pixel(x + 8, y + 9, color);
+            canvas.set_pixel(x + 8, y + 10, color);
+            canvas.set_pixel(x + 8, y + 11, color);
+        }
+
+        if self.contains(SixteenSegmentGlyph::D1) {
+            canvas.set_pixel(x + 1, y + 12, color);
+            canvas.set_pixel(x + 2, y + 12, color);
+            canvas.set_pixel(x + 3, y + 12, color);
+        }
+        if self.contains(SixteenSegmentGlyph::D2) {
+            canvas.set_pixel(x + 5, y + 12, color);
+            canvas.set_pixel(x + 6, y + 12, color);
+            canvas.set_pixel(x + 7, y + 12, color);
+        }
+
+        if self.contains(SixteenSegmentGlyph::E) {
+            canvas.set_pixel(x, y + 7, color);
+            canvas.set_pixel(x, y + 8, color);
+            canvas.set_pixel(x, y + 9, color);
+            canvas.set_pixel(x, y + 10, color);
+            canvas.set_pixel(x, y + 11, color);
+        }
+        if self.contains(SixteenSegmentGlyph::F) {
+            canvas.set_pixel(x, y + 1, color);
+            canvas.set_pixel(x, y + 2, color);
+            canvas.set_pixel(x, y + 3, color);
+            canvas.set_pixel(x, y + 4, color);
+            canvas.set_pixel(x, y + 5, color);
+        }
+
+        if self.contains(SixteenSegmentGlyph::G1) {
+            canvas.set_pixel(x + 1, y + 6, color);
+            canvas.set_pixel(x + 2, y + 6, color);
+            canvas.set_pixel(x + 3, y + 6, color);
+        }
+        if self.contains(SixteenSegmentGlyph::G2) {
+            canvas.set_pixel(x + 5, y + 6, color);
+            canvas.set_pixel(x + 6, y + 6, color);
+            canvas.set_pixel(x + 7, y + 6, color);
+        }
+
+        if self.contains(SixteenSegmentGlyph::H) {
+            canvas.set_pixel(x + 1, y + 1, color);
+            canvas.set_pixel(x + 1, y + 2, color);
+            canvas.set_pixel(x + 2, y + 3, color);
+            canvas.set_pixel(x + 3, y + 4, color);
+            canvas.set_pixel(x + 3, y + 5, color);
+        }
+        if self.contains(SixteenSegmentGlyph::I) {
+            canvas.set_pixel(x + 4, y + 1, color);
+            canvas.set_pixel(x + 4, y + 2, color);
+            canvas.set_pixel(x + 4, y + 3, color);
+            canvas.set_pixel(x + 4, y + 4, color);
+            canvas.set_pixel(x + 4, y + 5, color);
+        }
+        if self.contains(SixteenSegmentGlyph::J) {
+            canvas.set_pixel(x + 7, y + 1, color);
+            canvas.set_pixel(x + 7, y + 2, color);
+            canvas.set_pixel(x + 6, y + 3, color);
+            canvas.set_pixel(x + 5, y + 4, color);
+            canvas.set_pixel(x + 5, y + 5, color);
+        }
+        if self.contains(SixteenSegmentGlyph::K) {
+            canvas.set_pixel(x + 3, y + 7, color);
+            canvas.set_pixel(x + 3, y + 8, color);
+            canvas.set_pixel(x + 2, y + 9, color);
+            canvas.set_pixel(x + 1, y + 10, color);
+            canvas.set_pixel(x + 1, y + 11, color);
+        }
+        if self.contains(SixteenSegmentGlyph::L) {
+            canvas.set_pixel(x + 4, y + 7, color);
+            canvas.set_pixel(x + 4, y + 8, color);
+            canvas.set_pixel(x + 4, y + 9, color);
+            canvas.set_pixel(x + 4, y + 10, color);
+            canvas.set_pixel(x + 4, y + 11, color);
+        }
+        if self.contains(SixteenSegmentGlyph::M) {
+            canvas.set_pixel(x + 5, y + 7, color);
+            canvas.set_pixel(x + 5, y + 8, color);
+            canvas.set_pixel(x + 6, y + 9, color);
+            canvas.set_pixel(x + 7, y + 10, color);
+            canvas.set_pixel(x + 7, y + 11, color);
+        }
+    }
+
+    fn extents(&self) -> (usize, usize) {
+        (9, 13)
+    }
+}
diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs
new file mode 100644
index 0000000..ad3a90f
--- /dev/null
+++ b/pico-st7789/src/main.rs
@@ -0,0 +1,184 @@
+#![no_main]
+#![no_std]
+
+extern crate alloc;
+
+use alloc::fmt::format;
+use embedded_alloc::LlffHeap as Heap;
+use embedded_hal::{delay::DelayNs, digital::OutputPin};
+use fugit::RateExtU32;
+use panic_halt as _;
+use rp_pico::{
+    entry,
+    hal::{clocks::init_clocks_and_plls, spi::Spi, Clock, Sio, Timer, Watchdog},
+    pac, Pins,
+};
+
+mod canvas;
+use canvas::{Canvas, RGB, print};
+
+mod font;
+use font::{BitmapFont, Font, Glyph, SevenSegmentFont, SixteenSegmentFont};
+
+mod st7789;
+use st7789::{ST7789Display, SETUP_PROGRAM};
+
+const XOSC_CRYSTAL_FREQ: u32 = 12_000_000; // MHz, https://forums.raspberrypi.com/viewtopic.php?t=356764
+
+const ROWS: usize = 320;
+const COLUMNS: usize = 170;
+const FRAMEBUF: usize = ROWS * COLUMNS * 3;
+
+#[global_allocator]
+static HEAP: Heap = Heap::empty();
+
+static mut BUF: [u8; 163200] = [0; 163200];
+
+pub struct FrameBuf {
+    pub buf: &'static mut [u8; 163200],
+    pub width: usize,
+}
+
+impl FrameBuf {
+    pub fn new() -> Self {
+        Self {
+            buf: unsafe { &mut BUF },
+            width: 170,
+        }
+    }
+}
+
+impl Canvas for FrameBuf {
+    fn set_pixel(&mut self, x: usize, y: usize, color: &RGB) {
+        self.buf[(y * self.width + x) * 3 + 0] = color.r << 2;
+        self.buf[(y * self.width + x) * 3 + 1] = color.g << 2;
+        self.buf[(y * self.width + x) * 3 + 2] = color.b << 2;
+    }
+}
+
+#[entry]
+unsafe fn main() -> ! {
+    {
+        use core::mem::MaybeUninit;
+        const HEAP_SIZE: usize = 64 * 1024;
+        static mut HEAP_MEM: [MaybeUninit<u8>; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE];
+        unsafe { HEAP.init(HEAP_MEM.as_ptr() as usize, HEAP_SIZE) }
+    }
+
+    let font_bitmap = BitmapFont::new();
+
+    // rp_pico::pac::Peripherals is a reference to physical hardware defined on the Pico.
+    let mut peripherals = pac::Peripherals::take().unwrap();
+
+    // SIO inidcates "Single Cycle IO". I don't know what this means, but it could mean that this
+    // is a class of IO operations that can be run in a single clock cycle, such as switching a
+    // GPIO pin on or off.
+    let sio = Sio::new(peripherals.SIO);
+
+    // Many of the following systems require a watchdog. I do not know what this does, either, but
+    // it may be some failsafe software that will reset operations if the watchdog detects a lack
+    // of activity.
+    let mut watchdog = Watchdog::new(peripherals.WATCHDOG);
+
+    // Here we grab the GPIO pins in bank 0.
+    let pins = Pins::new(
+        peripherals.IO_BANK0,
+        peripherals.PADS_BANK0,
+        sio.gpio_bank0,
+        &mut peripherals.RESETS,
+    );
+
+    // Initialize an abstraction of the clock system with a batch of standard hardware clocks.
+    let clocks = init_clocks_and_plls(
+        XOSC_CRYSTAL_FREQ,
+        peripherals.XOSC,
+        peripherals.CLOCKS,
+        peripherals.PLL_SYS,
+        peripherals.PLL_USB,
+        &mut peripherals.RESETS,
+        &mut watchdog,
+    )
+    .ok()
+    .unwrap();
+
+    // An abstraction for a timer which we can use to delay the code.
+    let mut timer = Timer::new(peripherals.TIMER, &mut peripherals.RESETS, &clocks);
+
+    let mut led = pins.led.into_function();
+
+    // Grab the clock and data pins for SPI1. For Clock pins and for Data pins, there are only two
+    // pins each on the Pico which can function for SPI1.
+    let spi_clk = pins.gpio2.into_function();
+    let spi_sdo = pins.gpio3.into_function();
+    // let spi_sdi = pins.gpio4.into_function();
+    // Chip select 1 means the chip is not enabled
+    let mut board_select = pins.gpio13.into_function();
+    let mut data_command = pins.gpio15.into_function();
+    let mut reset = pins.gpio14.into_function();
+
+    let _ = reset.set_low();
+    let _ = board_select.set_high();
+    let _ = data_command.set_high();
+
+    // Now, create the SPI function abstraction for SPI1 with spi_clk and spi_sdo.
+    let mut spi = Spi::<_, _, _, 8>::new(peripherals.SPI0, (spi_sdo, spi_clk)).init(
+        &mut peripherals.RESETS,
+        // The SPI system uses the peripheral clock
+        clocks.peripheral_clock.freq(),
+        // Transmit data at a rate of 32Mbit.
+        32_u32.MHz(),
+        // Run with SPI Mode 1. This means that the clock line should start high and that data will
+        // be sampled starting at the first falling edge.
+        embedded_hal::spi::MODE_3,
+    );
+
+    let mut display = ST7789Display::new(board_select, data_command, spi);
+
+    let _ = reset.set_high();
+    timer.delay_ms(10);
+    for step in SETUP_PROGRAM {
+        let mut display = display.acquire();
+        display.send_command(&step, &mut timer);
+    }
+
+    timer.delay_ms(1000);
+
+    /*
+    let mut framebuf = [0; FRAMEBUF];
+    let mut canvas = Canvas::new(&mut framebuf, COLUMNS);
+    */
+    let mut canvas = FrameBuf::new();
+    let white = RGB { r: 63, g: 63, b: 63 };
+    let mut count = 0;
+
+    loop {
+        canvas.fill(0, 10, 170, 20, &RGB{r: 0, g: 0, b: 0 });
+        print(&mut canvas, &font_bitmap, 1, 10, &format(format_args!("COUNT: {:03}", count)), &RGB{ r: 32, g: 32, b: 63 });
+        print(&mut canvas, &font_bitmap, 1, 200, " !\"#$%&'<>*+,-./", &RGB{ r: 63, g: 63, b: 63 });
+        print(&mut canvas, &font_bitmap, 1, 220, "0123456789|: = ?", &RGB{ r: 63, g: 63, b: 63 });
+        print(&mut canvas, &font_bitmap, 1, 240, "@ABCDEFGHIJKLMNO", &RGB{ r: 63, g: 63, b: 63 });
+        print(&mut canvas, &font_bitmap, 1, 260, "PQRSTUVWXYZ[\\]^_", &RGB{ r: 63, g: 63, b: 63 });
+        print(&mut canvas, &font_bitmap, 1, 280, "`abcdefghijklmno", &RGB{ r: 63, g: 63, b: 63 });
+        print(&mut canvas, &font_bitmap, 1, 300, "pqrstuvwxyz{|}", &RGB{ r: 63, g: 63, b: 63 });
+
+        // canvas.square(10, 70, 160, 310, &white);
+
+        {
+            let display = display.acquire();
+            let _ = led.set_high();
+            timer.delay_ms(100);
+            display.send_buf(canvas.buf);
+            let _ = led.set_low();
+        }
+        count = count + 1;
+        /*
+        for x in 80..90 {
+            for y in 155..165 {
+                draw_pixel(&mut frame, x, y, (0, 0, 63));
+            }
+        }
+        */
+
+        timer.delay_ms(1000);
+    }
+}
diff --git a/pico-st7789/src/st7789.rs b/pico-st7789/src/st7789.rs
new file mode 100644
index 0000000..c2bb0e2
--- /dev/null
+++ b/pico-st7789/src/st7789.rs
@@ -0,0 +1,204 @@
+use embedded_hal::{delay::DelayNs, digital::OutputPin, spi::SpiBus};
+use rp_pico::hal::{
+    gpio::{FunctionSio, Pin, PinId, PullDown, SioOutput},
+    spi::{Enabled, SpiDevice, ValidSpiPinout},
+    Spi, Timer,
+};
+
+pub struct Step {
+    param_cnt: usize,
+    command: u8,
+    params: [u8; 4],
+    delay: Option<u32>,
+}
+
+impl Step {
+    pub fn send_command<D, Pinout, P>(
+        &self,
+        spi: &mut Spi<Enabled, D, Pinout, 8>,
+        data_command: &mut Pin<P, FunctionSio<SioOutput>, PullDown>,
+    ) where
+        D: SpiDevice,
+        Pinout: ValidSpiPinout<D>,
+        P: PinId,
+    {
+        let _ = data_command.set_low();
+        let _ = spi.write(&[self.command]);
+        if self.param_cnt > 0 {
+            let _ = data_command.set_high();
+            let _ = spi.write(&self.params[0..self.param_cnt]);
+        }
+    }
+}
+
+const NOP: u8 = 0x00;
+
+const SWRESET: Step = Step {
+    param_cnt: 0,
+    command: 0x01,
+    params: [0, 0, 0, 0],
+    delay: Some(150),
+};
+const SLPOUT: Step = Step {
+    param_cnt: 0,
+    command: 0x11,
+    params: [0, 0, 0, 0],
+    delay: Some(10),
+};
+const COLMOD: u8 = 0x3a;
+const MADCTL: Step = Step {
+    param_cnt: 1,
+    command: 0x36,
+    params: [0x00, 0, 0, 0],
+    delay: None,
+};
+const CASET: u8 = 0x2a;
+const RASET: u8 = 0x2b;
+const INVON: Step = Step {
+    param_cnt: 0,
+    command: 0x21,
+    params: [0, 0, 0, 0],
+    delay: Some(10),
+};
+const NORON: Step = Step {
+    param_cnt: 0,
+    command: 0x13,
+    params: [0, 0, 0, 0],
+    delay: Some(10),
+};
+const DISPOFF: Step = Step {
+    param_cnt: 0,
+    command: 0x28,
+    params: [0, 0, 0, 0],
+    delay: Some(10),
+};
+const DISPON: Step = Step {
+    param_cnt: 0,
+    command: 0x29,
+    params: [0, 0, 0, 0],
+    delay: Some(10),
+};
+const RAMWR: u8 = 0x2c;
+
+// Adafruit setup instructions
+// SWRESET (0x01), 150ms delay
+// SLPOUT (0x11), 10ms delay
+// COLMOD (0x3a) 0x55 (65K RGB, 16bit/pixel), 10ms delay
+// MADCTL (0x36) 0x00,
+//  memory data access control, RGB
+// CASET 0x00, 0, 0, 170,
+//  column address set, 4 parameters
+//  0x00, 0x00 indicates xstart is 0
+//  0x00, 170 indicates xend is 170
+// RASET 0x00, 0, 320 >> 8, 320 & 0xFF,
+//  row address set, 4 parameters
+//  0x00, 0x00 indicates ystart is 0
+//  3230 >> 8, 320 & 0xff indicates that 320 is the last y address
+// INVON, 10ms delay
+//  invert the display
+// NORON, 10ms delay
+//  normal display mode
+// DISPON, 10ms delay
+//  turn the display on
+
+pub const SETUP_PROGRAM: [Step; 8] = [
+    SWRESET,
+    SLPOUT,
+    Step {
+        param_cnt: 1,
+        command: COLMOD,
+        params: [0x66, 0, 0, 0],
+        delay: Some(10),
+    },
+    MADCTL,
+    Step {
+        param_cnt: 4,
+        command: CASET,
+        params: [0, 35, 0, 204],
+        delay: None,
+    },
+    /*
+    Step {
+        param_cnt: 4,
+        command: RASET,
+        params: [0, 0, (320 >> 8) as u8, (320 & 0xff) as u8],
+        delay: None,
+    },
+    */
+    INVON,
+    NORON,
+    DISPON,
+];
+
+pub struct ST7789Display<
+    BoardSelectId: PinId,
+    DataCommandId: PinId,
+    D: SpiDevice,
+    Pinout: ValidSpiPinout<D>,
+> {
+    inner: ST7789DisplayEnabled<BoardSelectId, DataCommandId, D, Pinout>,
+}
+
+impl<BoardSelectId: PinId, DataCommandId: PinId, D: SpiDevice, Pinout: ValidSpiPinout<D>>
+    ST7789Display<BoardSelectId, DataCommandId, D, Pinout>
+{
+    pub fn new(
+        board_select: Pin<BoardSelectId, FunctionSio<SioOutput>, PullDown>,
+        data_command: Pin<DataCommandId, FunctionSio<SioOutput>, PullDown>,
+        spi: Spi<Enabled, D, Pinout, 8>,
+    ) -> Self {
+        Self {
+            inner: ST7789DisplayEnabled {
+                board_select,
+                data_command,
+                spi,
+            },
+        }
+    }
+
+    pub fn acquire(
+        &mut self,
+    ) -> &mut ST7789DisplayEnabled<BoardSelectId, DataCommandId, D, Pinout> {
+        self.inner.board_select.set_low();
+        &mut self.inner
+    }
+}
+
+pub struct ST7789DisplayEnabled<
+    BoardSelectId: PinId,
+    DataCommandId: PinId,
+    D: SpiDevice,
+    Pinout: ValidSpiPinout<D>,
+> {
+    board_select: Pin<BoardSelectId, FunctionSio<SioOutput>, PullDown>,
+    data_command: Pin<DataCommandId, FunctionSio<SioOutput>, PullDown>,
+    spi: Spi<Enabled, D, Pinout, 8>,
+}
+
+impl<BoardSelectId: PinId, DataCommandId: PinId, D: SpiDevice, Pinout: ValidSpiPinout<D>>
+    ST7789DisplayEnabled<BoardSelectId, DataCommandId, D, Pinout>
+{
+    pub fn send_command(&mut self, step: &Step, timer: &mut Timer) {
+        step.send_command(&mut self.spi, &mut self.data_command);
+        if let Some(delay) = step.delay {
+            timer.delay_ms(delay);
+        }
+    }
+
+    pub fn send_buf(&mut self, frame: &[u8]) {
+        // let _ = DISPOFF.send_command(&mut self.spi, &mut self.data_command);
+        let _ = self.data_command.set_low();
+        let _ = self.spi.write(&[RAMWR]);
+        let _ = self.data_command.set_high();
+        let _ = self.spi.write(&frame);
+        // let _ = DISPON.send_command(&mut self.spi, &mut self.data_command);
+    }
+}
+
+impl<BoardSelectId: PinId, DataCommandId: PinId, D: SpiDevice, Pinout: ValidSpiPinout<D>> Drop
+    for ST7789DisplayEnabled<BoardSelectId, DataCommandId, D, Pinout>
+{
+    fn drop(&mut self) {
+        self.board_select.set_high();
+    }
+}
diff --git a/rust-toolchain b/rust-toolchain
index b96bf13..f5d4290 100644
--- a/rust-toolchain
+++ b/rust-toolchain
@@ -1,4 +1,4 @@
 [toolchain]
-channel = "1.81.0"
+channel = "1.85.0"
 targets = [ "wasm32-unknown-unknown", "thumbv6m-none-eabi" ]
 components = [ "rustfmt", "rust-analyzer", "clippy" ]