Jelajahi Sumber

Remove copy from ids

Cesar Rodas 1 tahun lalu
induk
melakukan
f29403f42f

+ 3464 - 0
Cargo.lock

@@ -0,0 +1,3464 @@
+# This file is automatically @generated by Cargo.
+# It is not intended for manual editing.
+version = 3
+
+[[package]]
+name = "actix-codec"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "78d1833b3838dbe990df0f1f87baf640cf6146e898166afe401839d1b001e570"
+dependencies = [
+ "bitflags 1.3.2",
+ "bytes 0.5.6",
+ "futures-core",
+ "futures-sink",
+ "log",
+ "pin-project 0.4.30",
+ "tokio 0.2.25",
+ "tokio-util",
+]
+
+[[package]]
+name = "actix-connect"
+version = "2.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "177837a10863f15ba8d3ae3ec12fac1099099529ed20083a27fdfe247381d0dc"
+dependencies = [
+ "actix-codec",
+ "actix-rt",
+ "actix-service",
+ "actix-utils",
+ "derive_more",
+ "either",
+ "futures-util",
+ "http",
+ "log",
+ "trust-dns-proto",
+ "trust-dns-resolver",
+]
+
+[[package]]
+name = "actix-http"
+version = "2.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2be6b66b62a794a8e6d366ac9415bb7d475ffd1e9f4671f38c1d8a8a5df950b3"
+dependencies = [
+ "actix-codec",
+ "actix-connect",
+ "actix-rt",
+ "actix-service",
+ "actix-threadpool",
+ "actix-utils",
+ "base64 0.13.1",
+ "bitflags 1.3.2",
+ "brotli",
+ "bytes 0.5.6",
+ "cookie",
+ "copyless",
+ "derive_more",
+ "either",
+ "encoding_rs",
+ "flate2",
+ "futures-channel",
+ "futures-core",
+ "futures-util",
+ "fxhash",
+ "h2",
+ "http",
+ "httparse",
+ "indexmap 1.9.3",
+ "itoa 0.4.8",
+ "language-tags",
+ "lazy_static",
+ "log",
+ "mime",
+ "percent-encoding",
+ "pin-project 1.1.3",
+ "rand 0.7.3",
+ "regex",
+ "serde",
+ "serde_json",
+ "serde_urlencoded",
+ "sha-1",
+ "slab",
+ "time",
+]
+
+[[package]]
+name = "actix-macros"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b4ca8ce00b267af8ccebbd647de0d61e0674b6e61185cc7a592ff88772bed655"
+dependencies = [
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "actix-router"
+version = "0.2.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2ad299af73649e1fc893e333ccf86f377751eb95ff875d095131574c6f43452c"
+dependencies = [
+ "bytestring",
+ "http",
+ "log",
+ "regex",
+ "serde",
+]
+
+[[package]]
+name = "actix-rt"
+version = "1.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "143fcc2912e0d1de2bcf4e2f720d2a60c28652ab4179685a1ee159e0fb3db227"
+dependencies = [
+ "actix-macros",
+ "actix-threadpool",
+ "copyless",
+ "futures-channel",
+ "futures-util",
+ "smallvec",
+ "tokio 0.2.25",
+]
+
+[[package]]
+name = "actix-server"
+version = "1.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "45407e6e672ca24784baa667c5d32ef109ccdd8d5e0b5ebb9ef8a67f4dfb708e"
+dependencies = [
+ "actix-codec",
+ "actix-rt",
+ "actix-service",
+ "actix-utils",
+ "futures-channel",
+ "futures-util",
+ "log",
+ "mio 0.6.23",
+ "mio-uds",
+ "num_cpus",
+ "slab",
+ "socket2 0.3.19",
+]
+
+[[package]]
+name = "actix-service"
+version = "1.0.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0052435d581b5be835d11f4eb3bce417c8af18d87ddf8ace99f8e67e595882bb"
+dependencies = [
+ "futures-util",
+ "pin-project 0.4.30",
+]
+
+[[package]]
+name = "actix-testing"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "47239ca38799ab74ee6a8a94d1ce857014b2ac36f242f70f3f75a66f691e791c"
+dependencies = [
+ "actix-macros",
+ "actix-rt",
+ "actix-server",
+ "actix-service",
+ "log",
+ "socket2 0.3.19",
+]
+
+[[package]]
+name = "actix-threadpool"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d209f04d002854b9afd3743032a27b066158817965bf5d036824d19ac2cc0e30"
+dependencies = [
+ "derive_more",
+ "futures-channel",
+ "lazy_static",
+ "log",
+ "num_cpus",
+ "parking_lot 0.11.2",
+ "threadpool",
+]
+
+[[package]]
+name = "actix-tls"
+version = "2.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "24789b7d7361cf5503a504ebe1c10806896f61e96eca9a7350e23001aca715fb"
+dependencies = [
+ "actix-codec",
+ "actix-service",
+ "actix-utils",
+ "futures-util",
+]
+
+[[package]]
+name = "actix-utils"
+version = "2.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2e9022dec56632d1d7979e59af14f0597a28a830a9c1c7fec8b2327eb9f16b5a"
+dependencies = [
+ "actix-codec",
+ "actix-rt",
+ "actix-service",
+ "bitflags 1.3.2",
+ "bytes 0.5.6",
+ "either",
+ "futures-channel",
+ "futures-sink",
+ "futures-util",
+ "log",
+ "pin-project 0.4.30",
+ "slab",
+]
+
+[[package]]
+name = "actix-web"
+version = "3.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b6534a126df581caf443ba2751cab42092c89b3f1d06a9d829b1e17edfe3e277"
+dependencies = [
+ "actix-codec",
+ "actix-http",
+ "actix-macros",
+ "actix-router",
+ "actix-rt",
+ "actix-server",
+ "actix-service",
+ "actix-testing",
+ "actix-threadpool",
+ "actix-tls",
+ "actix-utils",
+ "actix-web-codegen",
+ "awc",
+ "bytes 0.5.6",
+ "derive_more",
+ "encoding_rs",
+ "futures-channel",
+ "futures-core",
+ "futures-util",
+ "fxhash",
+ "log",
+ "mime",
+ "pin-project 1.1.3",
+ "regex",
+ "serde",
+ "serde_json",
+ "serde_urlencoded",
+ "socket2 0.3.19",
+ "time",
+ "tinyvec",
+ "url",
+]
+
+[[package]]
+name = "actix-web-codegen"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ad26f77093333e0e7c6ffe54ebe3582d908a104e448723eec6d43d08b07143fb"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "addr2line"
+version = "0.21.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8a30b2e23b9e17a9f90641c7ab1549cd9b44f296d3ccbf309d2863cfe398a0cb"
+dependencies = [
+ "gimli",
+]
+
+[[package]]
+name = "adler"
+version = "1.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe"
+
+[[package]]
+name = "ahash"
+version = "0.8.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2c99f64d1e06488f620f932677e24bc6e2897582980441ae90a671415bd7ec2f"
+dependencies = [
+ "cfg-if 1.0.0",
+ "getrandom 0.2.10",
+ "once_cell",
+ "version_check",
+]
+
+[[package]]
+name = "aho-corasick"
+version = "1.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ea5d730647d4fadd988536d06fecce94b7b4f2a7efdae548f1cf4b63205518ab"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "alloc-no-stdlib"
+version = "2.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cc7bb162ec39d46ab1ca8c77bf72e890535becd1751bb45f64c597edb4c8c6b3"
+
+[[package]]
+name = "alloc-stdlib"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94fb8275041c72129eb51b7d0322c29b8387a0386127718b096429201a5d6ece"
+dependencies = [
+ "alloc-no-stdlib",
+]
+
+[[package]]
+name = "allocator-api2"
+version = "0.2.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0942ffc6dcaadf03badf6e6a2d0228460359d5e34b57ccdc720b7382dfbd5ec5"
+
+[[package]]
+name = "android-tzdata"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e999941b234f3131b00bc13c22d06e8c5ff726d1b6318ac7eb276997bbb4fef0"
+
+[[package]]
+name = "android_system_properties"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "819e7219dbd41043ac279b19830f2efc897156490d7fd6ea916720117ee66311"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "async-channel"
+version = "1.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "81953c529336010edd6d8e358f886d9581267795c61b19475b71314bffa46d35"
+dependencies = [
+ "concurrent-queue",
+ "event-listener",
+ "futures-core",
+]
+
+[[package]]
+name = "async-executor"
+version = "1.5.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2c1da3ae8dabd9c00f453a329dfe1fb28da3c0a72e2478cdcd93171740c20499"
+dependencies = [
+ "async-lock",
+ "async-task",
+ "concurrent-queue",
+ "fastrand 2.0.1",
+ "futures-lite",
+ "slab",
+]
+
+[[package]]
+name = "async-global-executor"
+version = "2.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f1b6f5d7df27bd294849f8eec66ecfc63d11814df7a4f5d74168a2394467b776"
+dependencies = [
+ "async-channel",
+ "async-executor",
+ "async-io",
+ "async-lock",
+ "blocking",
+ "futures-lite",
+ "once_cell",
+]
+
+[[package]]
+name = "async-io"
+version = "1.13.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0fc5b45d93ef0529756f812ca52e44c221b35341892d3dcc34132ac02f3dd2af"
+dependencies = [
+ "async-lock",
+ "autocfg",
+ "cfg-if 1.0.0",
+ "concurrent-queue",
+ "futures-lite",
+ "log",
+ "parking",
+ "polling",
+ "rustix 0.37.24",
+ "slab",
+ "socket2 0.4.9",
+ "waker-fn",
+]
+
+[[package]]
+name = "async-lock"
+version = "2.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "287272293e9d8c41773cec55e365490fe034813a2f172f502d6ddcf75b2f582b"
+dependencies = [
+ "event-listener",
+]
+
+[[package]]
+name = "async-std"
+version = "1.12.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "62565bb4402e926b29953c785397c6dc0391b7b446e45008b0049eb43cec6f5d"
+dependencies = [
+ "async-channel",
+ "async-global-executor",
+ "async-io",
+ "async-lock",
+ "crossbeam-utils",
+ "futures-channel",
+ "futures-core",
+ "futures-io",
+ "futures-lite",
+ "gloo-timers",
+ "kv-log-macro",
+ "log",
+ "memchr",
+ "once_cell",
+ "pin-project-lite 0.2.13",
+ "pin-utils",
+ "slab",
+ "wasm-bindgen-futures",
+]
+
+[[package]]
+name = "async-task"
+version = "4.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ecc7ab41815b3c653ccd2978ec3255c81349336702dfdf62ee6f7069b12a3aae"
+
+[[package]]
+name = "async-trait"
+version = "0.1.73"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bc00ceb34980c03614e35a3a4e218276a0a824e911d07651cd0d858a51e8c0f0"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+]
+
+[[package]]
+name = "atoi"
+version = "2.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f28d99ec8bfea296261ca1af174f24225171fea9664ba9003cbebee704810528"
+dependencies = [
+ "num-traits",
+]
+
+[[package]]
+name = "atomic-waker"
+version = "1.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0"
+
+[[package]]
+name = "autocfg"
+version = "1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa"
+
+[[package]]
+name = "awc"
+version = "2.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b381e490e7b0cfc37ebc54079b0413d8093ef43d14a4e4747083f7fa47a9e691"
+dependencies = [
+ "actix-codec",
+ "actix-http",
+ "actix-rt",
+ "actix-service",
+ "base64 0.13.1",
+ "bytes 0.5.6",
+ "cfg-if 1.0.0",
+ "derive_more",
+ "futures-core",
+ "log",
+ "mime",
+ "percent-encoding",
+ "rand 0.7.3",
+ "serde",
+ "serde_json",
+ "serde_urlencoded",
+]
+
+[[package]]
+name = "backtrace"
+version = "0.3.69"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2089b7e3f35b9dd2d0ed921ead4f6d318c27680d4a5bd167b3ee120edb105837"
+dependencies = [
+ "addr2line",
+ "cc",
+ "cfg-if 1.0.0",
+ "libc",
+ "miniz_oxide",
+ "object",
+ "rustc-demangle",
+]
+
+[[package]]
+name = "base-x"
+version = "0.2.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4cbbc9d0964165b47557570cce6c952866c2678457aca742aafc9fb771d30270"
+
+[[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.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9ba43ea6f343b788c8764558649e08df62f86c6ef251fdaeb1ffd010a9ae50a2"
+
+[[package]]
+name = "base64ct"
+version = "1.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8c3c1a368f70d6cf7302d78f8f7093da241fb8e8807c05cc9e51a125895a6d5b"
+
+[[package]]
+name = "bincode"
+version = "1.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b1f45e9417d87227c7a56d22e471c6206462cba514c7590c09aff4cf6d1ddcad"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "bitflags"
+version = "1.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
+
+[[package]]
+name = "bitflags"
+version = "2.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b4682ae6287fcf752ecaabbfcc7b6f9b72aa33933dc23a554d853aea8eea8635"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "block-buffer"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "block-buffer"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "blocking"
+version = "1.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94c4ef1f913d78636d78d538eec1f18de81e481f44b1be0a81060090530846e1"
+dependencies = [
+ "async-channel",
+ "async-lock",
+ "async-task",
+ "fastrand 2.0.1",
+ "futures-io",
+ "futures-lite",
+ "piper",
+ "tracing",
+]
+
+[[package]]
+name = "brotli"
+version = "3.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "516074a47ef4bce09577a3b379392300159ce5b1ba2e501ff1c819950066100f"
+dependencies = [
+ "alloc-no-stdlib",
+ "alloc-stdlib",
+ "brotli-decompressor",
+]
+
+[[package]]
+name = "brotli-decompressor"
+version = "2.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "da74e2b81409b1b743f8f0c62cc6254afefb8b8e50bbfe3735550f7aeefa3448"
+dependencies = [
+ "alloc-no-stdlib",
+ "alloc-stdlib",
+]
+
+[[package]]
+name = "bumpalo"
+version = "3.14.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7f30e7476521f6f8af1a1c4c0b8cc94f0bee37d91763d0ca2665f299b6cd8aec"
+
+[[package]]
+name = "byteorder"
+version = "1.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610"
+
+[[package]]
+name = "bytes"
+version = "0.5.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0e4cec68f03f32e44924783795810fa50a7035d8c8ebe78580ad7e6c703fba38"
+
+[[package]]
+name = "bytes"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a2bd12c1caf447e69cd4528f47f94d203fd2582878ecb9e9465484c4148a8223"
+
+[[package]]
+name = "bytestring"
+version = "1.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "238e4886760d98c4f899360c834fa93e62cf7f721ac3c2da375cbdf4b8679aae"
+dependencies = [
+ "bytes 1.5.0",
+]
+
+[[package]]
+name = "cc"
+version = "1.0.83"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f1174fb0b6ec23863f8b971027804a42614e347eafb0a95bf0b12cdae21fc4d0"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "cfg-if"
+version = "0.1.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4785bdd1c96b2a846b2bd7cc02e86b6b3dbf14e7e53446c4f54c92a361040822"
+
+[[package]]
+name = "cfg-if"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
+
+[[package]]
+name = "chrono"
+version = "0.4.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7f2c685bad3eb3d45a01354cedb7d5faa66194d1d58ba6e267a8de788f79db38"
+dependencies = [
+ "android-tzdata",
+ "iana-time-zone",
+ "js-sys",
+ "num-traits",
+ "serde",
+ "wasm-bindgen",
+ "windows-targets",
+]
+
+[[package]]
+name = "concurrent-queue"
+version = "2.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f057a694a54f12365049b0958a1685bb52d567f5593b355fbf685838e873d400"
+dependencies = [
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "const-oid"
+version = "0.9.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "28c122c3980598d243d63d9a704629a2d748d101f278052ff068be5a4423ab6f"
+
+[[package]]
+name = "const_fn"
+version = "0.4.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fbdcdcb6d86f71c5e97409ad45898af11cbc995b4ee8112d59095a28d376c935"
+
+[[package]]
+name = "convert_case"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e"
+
+[[package]]
+name = "cookie"
+version = "0.14.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "03a5d7b21829bc7b4bf4754a978a241ae54ea55a40f92bb20216e54096f4b951"
+dependencies = [
+ "percent-encoding",
+ "time",
+ "version_check",
+]
+
+[[package]]
+name = "copyless"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a2df960f5d869b2dd8532793fde43eb5427cceb126c929747a26823ab0eeb536"
+
+[[package]]
+name = "core-foundation"
+version = "0.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "194a7a9e6de53fa55116934067c844d9d749312f75c6f6d0980e8c252f8c2146"
+dependencies = [
+ "core-foundation-sys",
+ "libc",
+]
+
+[[package]]
+name = "core-foundation-sys"
+version = "0.8.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e496a50fda8aacccc86d7529e2c1e0892dbd0f898a6b5645b5561b89c3210efa"
+
+[[package]]
+name = "cpufeatures"
+version = "0.2.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a17b76ff3a4162b0b27f354a0c87015ddad39d35f9c0c36607a3bdd175dde1f1"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "crc"
+version = "3.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "86ec7a15cbe22e59248fc7eadb1907dab5ba09372595da4d73dd805ed4417dfe"
+dependencies = [
+ "crc-catalog",
+]
+
+[[package]]
+name = "crc-catalog"
+version = "2.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9cace84e55f07e7301bae1c519df89cdad8cc3cd868413d3fdbdeca9ff3db484"
+
+[[package]]
+name = "crc32fast"
+version = "1.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b540bd8bc810d3885c6ea91e2018302f68baba2129ab3e88f32389ee9370880d"
+dependencies = [
+ "cfg-if 1.0.0",
+]
+
+[[package]]
+name = "crossbeam-queue"
+version = "0.3.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d1cfb3ea8a53f37c40dea2c7bedcbd88bdfae54f5e2175d6ecaff1c988353add"
+dependencies = [
+ "cfg-if 1.0.0",
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "crossbeam-utils"
+version = "0.8.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5a22b2d63d4d1dc0b7f1b6b2747dd0088008a9be28b6ddf0b1e7d335e3037294"
+dependencies = [
+ "cfg-if 1.0.0",
+]
+
+[[package]]
+name = "crypto-common"
+version = "0.1.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3"
+dependencies = [
+ "generic-array",
+ "typenum",
+]
+
+[[package]]
+name = "der"
+version = "0.7.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fffa369a668c8af7dbf8b5e56c9f744fbd399949ed171606040001947de40b1c"
+dependencies = [
+ "const-oid",
+ "pem-rfc7468",
+ "zeroize",
+]
+
+[[package]]
+name = "derive_more"
+version = "0.99.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4fb810d30a7c1953f91334de7244731fc3f3c10d7fe163338a35b9f640960321"
+dependencies = [
+ "convert_case",
+ "proc-macro2",
+ "quote",
+ "rustc_version 0.4.0",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "digest"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "digest"
+version = "0.10.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292"
+dependencies = [
+ "block-buffer 0.10.4",
+ "const-oid",
+ "crypto-common",
+ "subtle",
+]
+
+[[package]]
+name = "discard"
+version = "1.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "212d0f5754cb6769937f4501cc0e67f4f4483c8d2c3e1e922ee9edbe4ab4c7c0"
+
+[[package]]
+name = "dotenvy"
+version = "0.15.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1aaf95b3e5c8f23aa320147307562d361db0ae0d51242340f558153b4eb2439b"
+
+[[package]]
+name = "either"
+version = "1.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a26ae43d7bcc3b814de94796a5e736d4029efb0ee900c12e2d54c993ad1a1e07"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "encoding_rs"
+version = "0.8.33"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7268b386296a025e474d5140678f75d6de9493ae55a5d709eeb9dd08149945e1"
+dependencies = [
+ "cfg-if 1.0.0",
+]
+
+[[package]]
+name = "enum-as-inner"
+version = "0.3.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "570d109b813e904becc80d8d5da38376818a143348413f7149f1340fe04754d4"
+dependencies = [
+ "heck",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "env_logger"
+version = "0.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "85cdab6a89accf66733ad5a1693a4dcced6aeff64602b634530dd73c1f3ee9f0"
+dependencies = [
+ "humantime",
+ "is-terminal",
+ "log",
+ "regex",
+ "termcolor",
+]
+
+[[package]]
+name = "equivalent"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5"
+
+[[package]]
+name = "errno"
+version = "0.3.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "add4f07d43996f76ef320709726a556a9d4f965d9410d8d0271132d2f8293480"
+dependencies = [
+ "errno-dragonfly",
+ "libc",
+ "windows-sys",
+]
+
+[[package]]
+name = "errno-dragonfly"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "aa68f1b12764fab894d2755d2518754e71b4fd80ecfb822714a1206c2aab39bf"
+dependencies = [
+ "cc",
+ "libc",
+]
+
+[[package]]
+name = "etcetera"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "136d1b5283a1ab77bd9257427ffd09d8667ced0570b6f938942bc7568ed5b943"
+dependencies = [
+ "cfg-if 1.0.0",
+ "home",
+ "windows-sys",
+]
+
+[[package]]
+name = "event-listener"
+version = "2.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0206175f82b8d6bf6652ff7d71a1e27fd2e4efde587fd368662814d6ec1d9ce0"
+
+[[package]]
+name = "fastrand"
+version = "1.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e51093e27b0797c359783294ca4f0a911c270184cb10f85783b118614a1501be"
+dependencies = [
+ "instant",
+]
+
+[[package]]
+name = "fastrand"
+version = "2.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "25cbce373ec4653f1a01a31e8a5e5ec0c622dc27ff9c4e6606eefef5cbbed4a5"
+
+[[package]]
+name = "finl_unicode"
+version = "1.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8fcfdc7a0362c9f4444381a9e697c79d435fe65b52a37466fc2c1184cee9edc6"
+
+[[package]]
+name = "flate2"
+version = "1.0.27"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c6c98ee8095e9d1dcbf2fcc6d95acccb90d1c81db1e44725c6a984b1dbdfb010"
+dependencies = [
+ "crc32fast",
+ "miniz_oxide",
+]
+
+[[package]]
+name = "flume"
+version = "0.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "55ac459de2512911e4b674ce33cf20befaba382d05b62b008afc1c8b57cbf181"
+dependencies = [
+ "futures-core",
+ "futures-sink",
+ "spin 0.9.8",
+]
+
+[[package]]
+name = "fnv"
+version = "1.0.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1"
+
+[[package]]
+name = "foreign-types"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f6f339eb8adc052cd2ca78910fda869aefa38d22d5cb648e6485e4d3fc06f3b1"
+dependencies = [
+ "foreign-types-shared",
+]
+
+[[package]]
+name = "foreign-types-shared"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b"
+
+[[package]]
+name = "form_urlencoded"
+version = "1.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a62bc1cf6f830c2ec14a513a9fb124d0a213a629668a4186f329db21fe045652"
+dependencies = [
+ "percent-encoding",
+]
+
+[[package]]
+name = "fuchsia-zircon"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2e9763c69ebaae630ba35f74888db465e49e259ba1bc0eda7d06f4a067615d82"
+dependencies = [
+ "bitflags 1.3.2",
+ "fuchsia-zircon-sys",
+]
+
+[[package]]
+name = "fuchsia-zircon-sys"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3dcaa9ae7725d12cdb85b3ad99a434db70b468c09ded17e012d86b5c1010f7a7"
+
+[[package]]
+name = "futures"
+version = "0.3.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "23342abe12aba583913b2e62f22225ff9c950774065e4bfb61a19cd9770fec40"
+dependencies = [
+ "futures-channel",
+ "futures-core",
+ "futures-executor",
+ "futures-io",
+ "futures-sink",
+ "futures-task",
+ "futures-util",
+]
+
+[[package]]
+name = "futures-channel"
+version = "0.3.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "955518d47e09b25bbebc7a18df10b81f0c766eaf4c4f1cccef2fca5f2a4fb5f2"
+dependencies = [
+ "futures-core",
+ "futures-sink",
+]
+
+[[package]]
+name = "futures-core"
+version = "0.3.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4bca583b7e26f571124fe5b7561d49cb2868d79116cfa0eefce955557c6fee8c"
+
+[[package]]
+name = "futures-executor"
+version = "0.3.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ccecee823288125bd88b4d7f565c9e58e41858e47ab72e8ea2d64e93624386e0"
+dependencies = [
+ "futures-core",
+ "futures-task",
+ "futures-util",
+]
+
+[[package]]
+name = "futures-intrusive"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1d930c203dd0b6ff06e0201a4a2fe9149b43c684fd4420555b26d21b1a02956f"
+dependencies = [
+ "futures-core",
+ "lock_api",
+ "parking_lot 0.12.1",
+]
+
+[[package]]
+name = "futures-io"
+version = "0.3.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4fff74096e71ed47f8e023204cfd0aa1289cd54ae5430a9523be060cdb849964"
+
+[[package]]
+name = "futures-lite"
+version = "1.13.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "49a9d51ce47660b1e808d3c990b4709f2f415d928835a17dfd16991515c46bce"
+dependencies = [
+ "fastrand 1.9.0",
+ "futures-core",
+ "futures-io",
+ "memchr",
+ "parking",
+ "pin-project-lite 0.2.13",
+ "waker-fn",
+]
+
+[[package]]
+name = "futures-macro"
+version = "0.3.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "89ca545a94061b6365f2c7355b4b32bd20df3ff95f02da9329b34ccc3bd6ee72"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+]
+
+[[package]]
+name = "futures-sink"
+version = "0.3.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f43be4fe21a13b9781a69afa4985b0f6ee0e1afab2c6f454a8cf30e2b2237b6e"
+
+[[package]]
+name = "futures-task"
+version = "0.3.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "76d3d132be6c0e6aa1534069c705a74a5997a356c0dc2f86a47765e5617c5b65"
+
+[[package]]
+name = "futures-util"
+version = "0.3.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "26b01e40b772d54cf6c6d721c1d1abd0647a0106a12ecaa1c186273392a69533"
+dependencies = [
+ "futures-channel",
+ "futures-core",
+ "futures-io",
+ "futures-macro",
+ "futures-sink",
+ "futures-task",
+ "memchr",
+ "pin-project-lite 0.2.13",
+ "pin-utils",
+ "slab",
+]
+
+[[package]]
+name = "fxhash"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c31b6d751ae2c7f11320402d34e41349dd1016f8d5d45e48c4312bc8625af50c"
+dependencies = [
+ "byteorder",
+]
+
+[[package]]
+name = "generic-array"
+version = "0.14.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a"
+dependencies = [
+ "typenum",
+ "version_check",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.1.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce"
+dependencies = [
+ "cfg-if 1.0.0",
+ "libc",
+ "wasi 0.9.0+wasi-snapshot-preview1",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.2.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "be4136b2a15dd319360be1c07d9933517ccf0be8f16bf62a3bee4f0d618df427"
+dependencies = [
+ "cfg-if 1.0.0",
+ "libc",
+ "wasi 0.11.0+wasi-snapshot-preview1",
+]
+
+[[package]]
+name = "gimli"
+version = "0.28.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6fb8d784f27acf97159b40fc4db5ecd8aa23b9ad5ef69cdd136d3bc80665f0c0"
+
+[[package]]
+name = "gloo-timers"
+version = "0.2.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9b995a66bb87bebce9a0f4a95aed01daca4872c050bfcb21653361c03bc35e5c"
+dependencies = [
+ "futures-channel",
+ "futures-core",
+ "js-sys",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "h2"
+version = "0.2.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5e4728fd124914ad25e99e3d15a9361a879f6620f63cb56bbb08f95abb97a535"
+dependencies = [
+ "bytes 0.5.6",
+ "fnv",
+ "futures-core",
+ "futures-sink",
+ "futures-util",
+ "http",
+ "indexmap 1.9.3",
+ "slab",
+ "tokio 0.2.25",
+ "tokio-util",
+ "tracing",
+ "tracing-futures",
+]
+
+[[package]]
+name = "hashbrown"
+version = "0.12.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888"
+
+[[package]]
+name = "hashbrown"
+version = "0.14.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7dfda62a12f55daeae5015f81b0baea145391cb4520f86c248fc615d72640d12"
+dependencies = [
+ "ahash",
+ "allocator-api2",
+]
+
+[[package]]
+name = "hashlink"
+version = "0.8.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e8094feaf31ff591f651a2664fb9cfd92bba7a60ce3197265e9482ebe753c8f7"
+dependencies = [
+ "hashbrown 0.14.1",
+]
+
+[[package]]
+name = "heck"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8"
+dependencies = [
+ "unicode-segmentation",
+]
+
+[[package]]
+name = "hermit-abi"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d77f7ec81a6d05a3abb01ab6eb7590f6083d08449fe5a1c8b1e620283546ccb7"
+
+[[package]]
+name = "hex"
+version = "0.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70"
+
+[[package]]
+name = "hkdf"
+version = "0.12.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "791a029f6b9fc27657f6f188ec6e5e43f6911f6f878e0dc5501396e09809d437"
+dependencies = [
+ "hmac",
+]
+
+[[package]]
+name = "hmac"
+version = "0.12.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e"
+dependencies = [
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "home"
+version = "0.5.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5444c27eef6923071f7ebcc33e3444508466a76f7a2b93da00ed6e19f30c1ddb"
+dependencies = [
+ "windows-sys",
+]
+
+[[package]]
+name = "hostname"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3c731c3e10504cc8ed35cfe2f1db4c9274c3d35fa486e3b31df46f068ef3e867"
+dependencies = [
+ "libc",
+ "match_cfg",
+ "winapi 0.3.9",
+]
+
+[[package]]
+name = "http"
+version = "0.2.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bd6effc99afb63425aff9b05836f029929e345a6148a14b7ecd5ab67af944482"
+dependencies = [
+ "bytes 1.5.0",
+ "fnv",
+ "itoa 1.0.9",
+]
+
+[[package]]
+name = "httparse"
+version = "1.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d897f394bad6a705d5f4104762e116a75639e470d80901eed05a860a95cb1904"
+
+[[package]]
+name = "humantime"
+version = "2.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4"
+
+[[package]]
+name = "iana-time-zone"
+version = "0.1.57"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2fad5b825842d2b38bd206f3e81d6957625fd7f0a361e345c30e01a0ae2dd613"
+dependencies = [
+ "android_system_properties",
+ "core-foundation-sys",
+ "iana-time-zone-haiku",
+ "js-sys",
+ "wasm-bindgen",
+ "windows",
+]
+
+[[package]]
+name = "iana-time-zone-haiku"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f"
+dependencies = [
+ "cc",
+]
+
+[[package]]
+name = "idna"
+version = "0.2.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "418a0a6fab821475f634efe3ccc45c013f742efe03d853e8d3355d5cb850ecf8"
+dependencies = [
+ "matches",
+ "unicode-bidi",
+ "unicode-normalization",
+]
+
+[[package]]
+name = "idna"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7d20d6b07bfbc108882d88ed8e37d39636dcc260e15e30c45e6ba089610b917c"
+dependencies = [
+ "unicode-bidi",
+ "unicode-normalization",
+]
+
+[[package]]
+name = "indexmap"
+version = "1.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bd070e393353796e801d209ad339e89596eb4c8d430d18ede6a1cced8fafbd99"
+dependencies = [
+ "autocfg",
+ "hashbrown 0.12.3",
+]
+
+[[package]]
+name = "indexmap"
+version = "2.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8adf3ddd720272c6ea8bf59463c04e0f93d0bbf7c5439b691bca2987e0270897"
+dependencies = [
+ "equivalent",
+ "hashbrown 0.14.1",
+]
+
+[[package]]
+name = "instant"
+version = "0.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7a5bbe824c507c5da5956355e86a746d82e0e1464f65d862cc5e71da70e94b2c"
+dependencies = [
+ "cfg-if 1.0.0",
+]
+
+[[package]]
+name = "io-lifetimes"
+version = "1.0.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "eae7b9aee968036d54dce06cebaefd919e4472e753296daccd6d344e3e2df0c2"
+dependencies = [
+ "hermit-abi",
+ "libc",
+ "windows-sys",
+]
+
+[[package]]
+name = "iovec"
+version = "0.1.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b2b3ea6ff95e175473f8ffe6a7eb7c00d054240321b84c57051175fe3c1e075e"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "ipconfig"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f7e2f18aece9709094573a9f24f483c4f65caa4298e2f7ae1b71cc65d853fad7"
+dependencies = [
+ "socket2 0.3.19",
+ "widestring",
+ "winapi 0.3.9",
+ "winreg",
+]
+
+[[package]]
+name = "is-terminal"
+version = "0.4.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cb0889898416213fab133e1d33a0e5858a48177452750691bde3666d0fdbaf8b"
+dependencies = [
+ "hermit-abi",
+ "rustix 0.38.15",
+ "windows-sys",
+]
+
+[[package]]
+name = "itertools"
+version = "0.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b1c173a5686ce8bfa551b3563d0c2170bf24ca44da99c7ca4bfdab5418c3fe57"
+dependencies = [
+ "either",
+]
+
+[[package]]
+name = "itoa"
+version = "0.4.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b71991ff56294aa922b450139ee08b3bfc70982c6b2c7562771375cf73542dd4"
+
+[[package]]
+name = "itoa"
+version = "1.0.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "af150ab688ff2122fcef229be89cb50dd66af9e01a4ff320cc137eecc9bacc38"
+
+[[package]]
+name = "js-sys"
+version = "0.3.64"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c5f195fe497f702db0f318b07fdd68edb16955aed830df8363d837542f8f935a"
+dependencies = [
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "kernel32-sys"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7507624b29483431c0ba2d82aece8ca6cdba9382bff4ddd0f7490560c056098d"
+dependencies = [
+ "winapi 0.2.8",
+ "winapi-build",
+]
+
+[[package]]
+name = "kv-log-macro"
+version = "1.0.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0de8b303297635ad57c9f5059fd9cee7a47f8e8daa09df0fcd07dd39fb22977f"
+dependencies = [
+ "log",
+]
+
+[[package]]
+name = "language-tags"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a91d884b6667cd606bb5a69aa0c99ba811a115fc68915e7056ec08a46e93199a"
+
+[[package]]
+name = "lazy_static"
+version = "1.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646"
+dependencies = [
+ "spin 0.5.2",
+]
+
+[[package]]
+name = "ledger"
+version = "0.1.0"
+dependencies = [
+ "actix-web",
+ "env_logger",
+ "ledger-utxo",
+ "serde",
+ "serde_json",
+ "tokio 1.32.0",
+]
+
+[[package]]
+name = "ledger-utxo"
+version = "0.1.0"
+dependencies = [
+ "async-trait",
+ "bincode",
+ "chrono",
+ "futures",
+ "hex",
+ "serde",
+ "sha2",
+ "sqlx",
+ "strum",
+ "strum_macros",
+ "thiserror",
+ "tokio 1.32.0",
+]
+
+[[package]]
+name = "libc"
+version = "0.2.148"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9cdc71e17332e86d2e1d38c1f99edcb6288ee11b815fb1a4b049eaa2114d369b"
+
+[[package]]
+name = "libm"
+version = "0.2.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f7012b1bbb0719e1097c47611d3898568c546d597c2e74d66f6087edd5233ff4"
+
+[[package]]
+name = "libsqlite3-sys"
+version = "0.26.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "afc22eff61b133b115c6e8c74e818c628d6d5e7a502afea6f64dee076dd94326"
+dependencies = [
+ "cc",
+ "pkg-config",
+ "vcpkg",
+]
+
+[[package]]
+name = "linked-hash-map"
+version = "0.5.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0717cef1bc8b636c6e1c1bbdefc09e6322da8a9321966e8928ef80d20f7f770f"
+
+[[package]]
+name = "linux-raw-sys"
+version = "0.3.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ef53942eb7bf7ff43a617b3e2c1c4a5ecf5944a7c1bc12d7ee39bbb15e5c1519"
+
+[[package]]
+name = "linux-raw-sys"
+version = "0.4.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3852614a3bd9ca9804678ba6be5e3b8ce76dfc902cae004e3e0c44051b6e88db"
+
+[[package]]
+name = "lock_api"
+version = "0.4.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c1cc9717a20b1bb222f333e6a92fd32f7d8a18ddc5a3191a11af45dcbf4dcd16"
+dependencies = [
+ "autocfg",
+ "scopeguard",
+]
+
+[[package]]
+name = "log"
+version = "0.4.20"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b5e6163cb8c49088c2c36f57875e58ccd8c87c7427f7fbd50ea6710b2f3f2e8f"
+dependencies = [
+ "value-bag",
+]
+
+[[package]]
+name = "lru-cache"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "31e24f1ad8321ca0e8a1e0ac13f23cb668e6f5466c2c57319f6a5cf1cc8e3b1c"
+dependencies = [
+ "linked-hash-map",
+]
+
+[[package]]
+name = "match_cfg"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ffbee8634e0d45d258acb448e7eaab3fce7a0a467395d4d9f228e3c1f01fb2e4"
+
+[[package]]
+name = "matches"
+version = "0.1.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2532096657941c2fea9c289d370a250971c689d4f143798ff67113ec042024a5"
+
+[[package]]
+name = "md-5"
+version = "0.10.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d89e7ee0cfbedfc4da3340218492196241d89eefb6dab27de5df917a6d2e78cf"
+dependencies = [
+ "cfg-if 1.0.0",
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "memchr"
+version = "2.6.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f665ee40bc4a3c5590afb1e9677db74a508659dfd71e126420da8274909a0167"
+
+[[package]]
+name = "mime"
+version = "0.3.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a"
+
+[[package]]
+name = "minimal-lexical"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "68354c5c6bd36d73ff3feceb05efa59b6acb7626617f4962be322a825e61f79a"
+
+[[package]]
+name = "miniz_oxide"
+version = "0.7.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e7810e0be55b428ada41041c41f32c9f1a42817901b4ccf45fa3d4b6561e74c7"
+dependencies = [
+ "adler",
+]
+
+[[package]]
+name = "mio"
+version = "0.6.23"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4afd66f5b91bf2a3bc13fad0e21caedac168ca4c707504e75585648ae80e4cc4"
+dependencies = [
+ "cfg-if 0.1.10",
+ "fuchsia-zircon",
+ "fuchsia-zircon-sys",
+ "iovec",
+ "kernel32-sys",
+ "libc",
+ "log",
+ "miow",
+ "net2",
+ "slab",
+ "winapi 0.2.8",
+]
+
+[[package]]
+name = "mio"
+version = "0.8.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "927a765cd3fc26206e66b296465fa9d3e5ab003e651c1b3c060e7956d96b19d2"
+dependencies = [
+ "libc",
+ "wasi 0.11.0+wasi-snapshot-preview1",
+ "windows-sys",
+]
+
+[[package]]
+name = "mio-uds"
+version = "0.6.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "afcb699eb26d4332647cc848492bbc15eafb26f08d0304550d5aa1f612e066f0"
+dependencies = [
+ "iovec",
+ "libc",
+ "mio 0.6.23",
+]
+
+[[package]]
+name = "miow"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ebd808424166322d4a38da87083bfddd3ac4c131334ed55856112eb06d46944d"
+dependencies = [
+ "kernel32-sys",
+ "net2",
+ "winapi 0.2.8",
+ "ws2_32-sys",
+]
+
+[[package]]
+name = "native-tls"
+version = "0.2.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "07226173c32f2926027b63cce4bcd8076c3552846cbe7925f3aaffeac0a3b92e"
+dependencies = [
+ "lazy_static",
+ "libc",
+ "log",
+ "openssl",
+ "openssl-probe",
+ "openssl-sys",
+ "schannel",
+ "security-framework",
+ "security-framework-sys",
+ "tempfile",
+]
+
+[[package]]
+name = "net2"
+version = "0.2.39"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b13b648036a2339d06de780866fbdfda0dde886de7b3af2ddeba8b14f4ee34ac"
+dependencies = [
+ "cfg-if 0.1.10",
+ "libc",
+ "winapi 0.3.9",
+]
+
+[[package]]
+name = "nom"
+version = "7.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d273983c5a657a70a3e8f2a01329822f3b8c8172b73826411a55751e404a0a4a"
+dependencies = [
+ "memchr",
+ "minimal-lexical",
+]
+
+[[package]]
+name = "num-bigint-dig"
+version = "0.8.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dc84195820f291c7697304f3cbdadd1cb7199c0efc917ff5eafd71225c136151"
+dependencies = [
+ "byteorder",
+ "lazy_static",
+ "libm",
+ "num-integer",
+ "num-iter",
+ "num-traits",
+ "rand 0.8.5",
+ "smallvec",
+ "zeroize",
+]
+
+[[package]]
+name = "num-integer"
+version = "0.1.45"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "225d3389fb3509a24c93f5c29eb6bde2586b98d9f016636dff58d7c6f7569cd9"
+dependencies = [
+ "autocfg",
+ "num-traits",
+]
+
+[[package]]
+name = "num-iter"
+version = "0.1.43"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7d03e6c028c5dc5cac6e2dec0efda81fc887605bb3d884578bb6d6bf7514e252"
+dependencies = [
+ "autocfg",
+ "num-integer",
+ "num-traits",
+]
+
+[[package]]
+name = "num-traits"
+version = "0.2.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f30b0abd723be7e2ffca1272140fac1a2f084c77ec3e123c192b66af1ee9e6c2"
+dependencies = [
+ "autocfg",
+ "libm",
+]
+
+[[package]]
+name = "num_cpus"
+version = "1.16.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43"
+dependencies = [
+ "hermit-abi",
+ "libc",
+]
+
+[[package]]
+name = "object"
+version = "0.32.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9cf5f9dd3933bd50a9e1f149ec995f39ae2c496d31fd772c1fd45ebc27e902b0"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "once_cell"
+version = "1.18.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dd8b5dd2ae5ed71462c540258bedcb51965123ad7e7ccf4b9a8cafaa4a63576d"
+
+[[package]]
+name = "opaque-debug"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5"
+
+[[package]]
+name = "openssl"
+version = "0.10.57"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bac25ee399abb46215765b1cb35bc0212377e58a061560d8b29b024fd0430e7c"
+dependencies = [
+ "bitflags 2.4.0",
+ "cfg-if 1.0.0",
+ "foreign-types",
+ "libc",
+ "once_cell",
+ "openssl-macros",
+ "openssl-sys",
+]
+
+[[package]]
+name = "openssl-macros"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+]
+
+[[package]]
+name = "openssl-probe"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf"
+
+[[package]]
+name = "openssl-sys"
+version = "0.9.93"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "db4d56a4c0478783083cfafcc42493dd4a981d41669da64b4572a2a089b51b1d"
+dependencies = [
+ "cc",
+ "libc",
+ "pkg-config",
+ "vcpkg",
+]
+
+[[package]]
+name = "parking"
+version = "2.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e52c774a4c39359c1d1c52e43f73dd91a75a614652c825408eec30c95a9b2067"
+
+[[package]]
+name = "parking_lot"
+version = "0.11.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7d17b78036a60663b797adeaee46f5c9dfebb86948d1255007a1d6be0271ff99"
+dependencies = [
+ "instant",
+ "lock_api",
+ "parking_lot_core 0.8.6",
+]
+
+[[package]]
+name = "parking_lot"
+version = "0.12.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f"
+dependencies = [
+ "lock_api",
+ "parking_lot_core 0.9.8",
+]
+
+[[package]]
+name = "parking_lot_core"
+version = "0.8.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "60a2cfe6f0ad2bfc16aefa463b497d5c7a5ecd44a23efa72aa342d90177356dc"
+dependencies = [
+ "cfg-if 1.0.0",
+ "instant",
+ "libc",
+ "redox_syscall 0.2.16",
+ "smallvec",
+ "winapi 0.3.9",
+]
+
+[[package]]
+name = "parking_lot_core"
+version = "0.9.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "93f00c865fe7cabf650081affecd3871070f26767e7b2070a3ffae14c654b447"
+dependencies = [
+ "cfg-if 1.0.0",
+ "libc",
+ "redox_syscall 0.3.5",
+ "smallvec",
+ "windows-targets",
+]
+
+[[package]]
+name = "paste"
+version = "1.0.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "de3145af08024dea9fa9914f381a17b8fc6034dfb00f3a84013f7ff43f29ed4c"
+
+[[package]]
+name = "pem-rfc7468"
+version = "0.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "88b39c9bfcfc231068454382784bb460aae594343fb030d46e9f50a645418412"
+dependencies = [
+ "base64ct",
+]
+
+[[package]]
+name = "percent-encoding"
+version = "2.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9b2a4787296e9989611394c33f193f676704af1686e70b8f8033ab5ba9a35a94"
+
+[[package]]
+name = "pin-project"
+version = "0.4.30"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3ef0f924a5ee7ea9cbcea77529dba45f8a9ba9f622419fe3386ca581a3ae9d5a"
+dependencies = [
+ "pin-project-internal 0.4.30",
+]
+
+[[package]]
+name = "pin-project"
+version = "1.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fda4ed1c6c173e3fc7a83629421152e01d7b1f9b7f65fb301e490e8cfc656422"
+dependencies = [
+ "pin-project-internal 1.1.3",
+]
+
+[[package]]
+name = "pin-project-internal"
+version = "0.4.30"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "851c8d0ce9bebe43790dedfc86614c23494ac9f423dd618d3a61fc693eafe61e"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "pin-project-internal"
+version = "1.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4359fd9c9171ec6e8c62926d6faaf553a8dc3f64e1507e76da7911b4f6a04405"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+]
+
+[[package]]
+name = "pin-project-lite"
+version = "0.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "257b64915a082f7811703966789728173279bdebb956b143dbcd23f6f970a777"
+
+[[package]]
+name = "pin-project-lite"
+version = "0.2.13"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8afb450f006bf6385ca15ef45d71d2288452bc3683ce2e2cacc0d18e4be60b58"
+
+[[package]]
+name = "pin-utils"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184"
+
+[[package]]
+name = "piper"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "668d31b1c4eba19242f2088b2bf3316b82ca31082a8335764db4e083db7485d4"
+dependencies = [
+ "atomic-waker",
+ "fastrand 2.0.1",
+ "futures-io",
+]
+
+[[package]]
+name = "pkcs1"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c8ffb9f10fa047879315e6625af03c164b16962a5368d724ed16323b68ace47f"
+dependencies = [
+ "der",
+ "pkcs8",
+ "spki",
+]
+
+[[package]]
+name = "pkcs8"
+version = "0.10.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f950b2377845cebe5cf8b5165cb3cc1a5e0fa5cfa3e1f7f55707d8fd82e0a7b7"
+dependencies = [
+ "der",
+ "spki",
+]
+
+[[package]]
+name = "pkg-config"
+version = "0.3.27"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "26072860ba924cbfa98ea39c8c19b4dd6a4a25423dbdf219c1eca91aa0cf6964"
+
+[[package]]
+name = "polling"
+version = "2.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4b2d323e8ca7996b3e23126511a523f7e62924d93ecd5ae73b333815b0eb3dce"
+dependencies = [
+ "autocfg",
+ "bitflags 1.3.2",
+ "cfg-if 1.0.0",
+ "concurrent-queue",
+ "libc",
+ "log",
+ "pin-project-lite 0.2.13",
+ "windows-sys",
+]
+
+[[package]]
+name = "ppv-lite86"
+version = "0.2.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5b40af805b3121feab8a3c29f04d8ad262fa8e0561883e7653e024ae4479e6de"
+
+[[package]]
+name = "proc-macro-hack"
+version = "0.5.20+deprecated"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dc375e1527247fe1a97d8b7156678dfe7c1af2fc075c9a4db3690ecd2a148068"
+
+[[package]]
+name = "proc-macro2"
+version = "1.0.67"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3d433d9f1a3e8c1263d9456598b16fec66f4acc9a74dacffd35c7bb09b3a1328"
+dependencies = [
+ "unicode-ident",
+]
+
+[[package]]
+name = "quick-error"
+version = "1.2.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a1d01941d82fa2ab50be1e79e6714289dd7cde78eba4c074bc5a4374f650dfe0"
+
+[[package]]
+name = "quote"
+version = "1.0.33"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5267fca4496028628a95160fc423a33e8b2e6af8a5302579e322e4b520293cae"
+dependencies = [
+ "proc-macro2",
+]
+
+[[package]]
+name = "rand"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03"
+dependencies = [
+ "getrandom 0.1.16",
+ "libc",
+ "rand_chacha 0.2.2",
+ "rand_core 0.5.1",
+ "rand_hc",
+]
+
+[[package]]
+name = "rand"
+version = "0.8.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
+dependencies = [
+ "libc",
+ "rand_chacha 0.3.1",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_chacha"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.5.1",
+]
+
+[[package]]
+name = "rand_chacha"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19"
+dependencies = [
+ "getrandom 0.1.16",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.6.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c"
+dependencies = [
+ "getrandom 0.2.10",
+]
+
+[[package]]
+name = "rand_hc"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c"
+dependencies = [
+ "rand_core 0.5.1",
+]
+
+[[package]]
+name = "redox_syscall"
+version = "0.2.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a"
+dependencies = [
+ "bitflags 1.3.2",
+]
+
+[[package]]
+name = "redox_syscall"
+version = "0.3.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "567664f262709473930a4bf9e51bf2ebf3348f2e748ccc50dea20646858f8f29"
+dependencies = [
+ "bitflags 1.3.2",
+]
+
+[[package]]
+name = "regex"
+version = "1.9.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ebee201405406dbf528b8b672104ae6d6d63e6d118cb10e4d51abbc7b58044ff"
+dependencies = [
+ "aho-corasick",
+ "memchr",
+ "regex-automata",
+ "regex-syntax",
+]
+
+[[package]]
+name = "regex-automata"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "59b23e92ee4318893fa3fe3e6fb365258efbfe6ac6ab30f090cdcbb7aa37efa9"
+dependencies = [
+ "aho-corasick",
+ "memchr",
+ "regex-syntax",
+]
+
+[[package]]
+name = "regex-syntax"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dbb5fb1acd8a1a18b3dd5be62d25485eb770e05afb408a9627d14d451bae12da"
+
+[[package]]
+name = "resolv-conf"
+version = "0.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "52e44394d2086d010551b14b53b1f24e31647570cd1deb0379e2c21b329aba00"
+dependencies = [
+ "hostname",
+ "quick-error",
+]
+
+[[package]]
+name = "rsa"
+version = "0.9.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6ab43bb47d23c1a631b4b680199a45255dce26fa9ab2fa902581f624ff13e6a8"
+dependencies = [
+ "byteorder",
+ "const-oid",
+ "digest 0.10.7",
+ "num-bigint-dig",
+ "num-integer",
+ "num-iter",
+ "num-traits",
+ "pkcs1",
+ "pkcs8",
+ "rand_core 0.6.4",
+ "signature",
+ "spki",
+ "subtle",
+ "zeroize",
+]
+
+[[package]]
+name = "rustc-demangle"
+version = "0.1.23"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d626bb9dae77e28219937af045c257c28bfd3f69333c512553507f5f9798cb76"
+
+[[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.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bfa0f585226d2e68097d4f95d113b15b83a82e819ab25717ec0590d9584ef366"
+dependencies = [
+ "semver 1.0.19",
+]
+
+[[package]]
+name = "rustix"
+version = "0.37.24"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4279d76516df406a8bd37e7dff53fd37d1a093f997a3c34a5c21658c126db06d"
+dependencies = [
+ "bitflags 1.3.2",
+ "errno",
+ "io-lifetimes",
+ "libc",
+ "linux-raw-sys 0.3.8",
+ "windows-sys",
+]
+
+[[package]]
+name = "rustix"
+version = "0.38.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d2f9da0cbd88f9f09e7814e388301c8414c51c62aa6ce1e4b5c551d49d96e531"
+dependencies = [
+ "bitflags 2.4.0",
+ "errno",
+ "libc",
+ "linux-raw-sys 0.4.8",
+ "windows-sys",
+]
+
+[[package]]
+name = "rustversion"
+version = "1.0.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7ffc183a10b4478d04cbbbfc96d0873219d962dd5accaff2ffbd4ceb7df837f4"
+
+[[package]]
+name = "ryu"
+version = "1.0.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1ad4cc8da4ef723ed60bced201181d83791ad433213d8c24efffda1eec85d741"
+
+[[package]]
+name = "schannel"
+version = "0.1.22"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0c3733bf4cf7ea0880754e19cb5a462007c4a8c1914bff372ccc95b464f1df88"
+dependencies = [
+ "windows-sys",
+]
+
+[[package]]
+name = "scopeguard"
+version = "1.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49"
+
+[[package]]
+name = "security-framework"
+version = "2.9.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "05b64fb303737d99b81884b2c63433e9ae28abebe5eb5045dcdd175dc2ecf4de"
+dependencies = [
+ "bitflags 1.3.2",
+ "core-foundation",
+ "core-foundation-sys",
+ "libc",
+ "security-framework-sys",
+]
+
+[[package]]
+name = "security-framework-sys"
+version = "2.9.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e932934257d3b408ed8f30db49d85ea163bfe74961f017f405b025af298f0c7a"
+dependencies = [
+ "core-foundation-sys",
+ "libc",
+]
+
+[[package]]
+name = "semver"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1d7eb9ef2c18661902cc47e535f9bc51b78acd254da71d375c2f6720d9a40403"
+dependencies = [
+ "semver-parser",
+]
+
+[[package]]
+name = "semver"
+version = "1.0.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ad977052201c6de01a8ef2aa3378c4bd23217a056337d1d6da40468d267a4fb0"
+
+[[package]]
+name = "semver-parser"
+version = "0.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "388a1df253eca08550bef6c72392cfe7c30914bf41df5269b68cbd6ff8f570a3"
+
+[[package]]
+name = "serde"
+version = "1.0.188"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cf9e0fcba69a370eed61bcf2b728575f726b50b55cba78064753d708ddc7549e"
+dependencies = [
+ "serde_derive",
+]
+
+[[package]]
+name = "serde_derive"
+version = "1.0.188"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4eca7ac642d82aa35b60049a6eccb4be6be75e599bd2e9adb5f875a737654af2"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+]
+
+[[package]]
+name = "serde_json"
+version = "1.0.107"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6b420ce6e3d8bd882e9b243c6eed35dbc9a6110c9769e74b584e0d68d1f20c65"
+dependencies = [
+ "itoa 1.0.9",
+ "ryu",
+ "serde",
+]
+
+[[package]]
+name = "serde_urlencoded"
+version = "0.7.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd"
+dependencies = [
+ "form_urlencoded",
+ "itoa 1.0.9",
+ "ryu",
+ "serde",
+]
+
+[[package]]
+name = "sha-1"
+version = "0.9.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "99cd6713db3cf16b6c84e06321e049a9b9f699826e16096d23bbcc44d15d51a6"
+dependencies = [
+ "block-buffer 0.9.0",
+ "cfg-if 1.0.0",
+ "cpufeatures",
+ "digest 0.9.0",
+ "opaque-debug",
+]
+
+[[package]]
+name = "sha1"
+version = "0.6.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c1da05c97445caa12d05e848c4a4fcbbea29e748ac28f7e80e9b010392063770"
+dependencies = [
+ "sha1_smol",
+]
+
+[[package]]
+name = "sha1"
+version = "0.10.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e3bf829a2d51ab4a5ddf1352d8470c140cadc8301b2ae1789db023f01cedd6ba"
+dependencies = [
+ "cfg-if 1.0.0",
+ "cpufeatures",
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "sha1_smol"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ae1a47186c03a32177042e55dbc5fd5aee900b8e0069a8d70fba96a9375cd012"
+
+[[package]]
+name = "sha2"
+version = "0.10.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8"
+dependencies = [
+ "cfg-if 1.0.0",
+ "cpufeatures",
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "signal-hook-registry"
+version = "1.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d8229b473baa5980ac72ef434c4415e70c4b5e71b423043adb4ba059f89c99a1"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "signature"
+version = "2.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5e1788eed21689f9cf370582dfc467ef36ed9c707f073528ddafa8d83e3b8500"
+dependencies = [
+ "digest 0.10.7",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "slab"
+version = "0.4.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8f92a496fb766b417c996b9c5e57daf2f7ad3b0bebe1ccfca4856390e3d3bb67"
+dependencies = [
+ "autocfg",
+]
+
+[[package]]
+name = "smallvec"
+version = "1.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "942b4a808e05215192e39f4ab80813e599068285906cc91aa64f923db842bd5a"
+
+[[package]]
+name = "socket2"
+version = "0.3.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "122e570113d28d773067fab24266b66753f6ea915758651696b6e35e49f88d6e"
+dependencies = [
+ "cfg-if 1.0.0",
+ "libc",
+ "winapi 0.3.9",
+]
+
+[[package]]
+name = "socket2"
+version = "0.4.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "64a4a911eed85daf18834cfaa86a79b7d266ff93ff5ba14005426219480ed662"
+dependencies = [
+ "libc",
+ "winapi 0.3.9",
+]
+
+[[package]]
+name = "socket2"
+version = "0.5.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4031e820eb552adee9295814c0ced9e5cf38ddf1e8b7d566d6de8e2538ea989e"
+dependencies = [
+ "libc",
+ "windows-sys",
+]
+
+[[package]]
+name = "spin"
+version = "0.5.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6e63cff320ae2c57904679ba7cb63280a3dc4613885beafb148ee7bf9aa9042d"
+
+[[package]]
+name = "spin"
+version = "0.9.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6980e8d7511241f8acf4aebddbb1ff938df5eebe98691418c4468d0b72a96a67"
+dependencies = [
+ "lock_api",
+]
+
+[[package]]
+name = "spki"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d1e996ef02c474957d681f1b05213dfb0abab947b446a62d37770b23500184a"
+dependencies = [
+ "base64ct",
+ "der",
+]
+
+[[package]]
+name = "sqlformat"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6b7b278788e7be4d0d29c0f39497a0eef3fba6bbc8e70d8bf7fde46edeaa9e85"
+dependencies = [
+ "itertools",
+ "nom",
+ "unicode_categories",
+]
+
+[[package]]
+name = "sqlx"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0e50c216e3624ec8e7ecd14c6a6a6370aad6ee5d8cfc3ab30b5162eeeef2ed33"
+dependencies = [
+ "sqlx-core",
+ "sqlx-macros",
+ "sqlx-mysql",
+ "sqlx-postgres",
+ "sqlx-sqlite",
+]
+
+[[package]]
+name = "sqlx-core"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8d6753e460c998bbd4cd8c6f0ed9a64346fcca0723d6e75e52fdc351c5d2169d"
+dependencies = [
+ "ahash",
+ "async-io",
+ "async-std",
+ "atoi",
+ "byteorder",
+ "bytes 1.5.0",
+ "chrono",
+ "crc",
+ "crossbeam-queue",
+ "dotenvy",
+ "either",
+ "event-listener",
+ "futures-channel",
+ "futures-core",
+ "futures-intrusive",
+ "futures-io",
+ "futures-util",
+ "hashlink",
+ "hex",
+ "indexmap 2.0.2",
+ "log",
+ "memchr",
+ "native-tls",
+ "once_cell",
+ "paste",
+ "percent-encoding",
+ "serde",
+ "serde_json",
+ "sha2",
+ "smallvec",
+ "sqlformat",
+ "thiserror",
+ "tokio 1.32.0",
+ "tokio-stream",
+ "tracing",
+ "url",
+]
+
+[[package]]
+name = "sqlx-macros"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9a793bb3ba331ec8359c1853bd39eed32cdd7baaf22c35ccf5c92a7e8d1189ec"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "sqlx-core",
+ "sqlx-macros-core",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "sqlx-macros-core"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0a4ee1e104e00dedb6aa5ffdd1343107b0a4702e862a84320ee7cc74782d96fc"
+dependencies = [
+ "async-std",
+ "dotenvy",
+ "either",
+ "heck",
+ "hex",
+ "once_cell",
+ "proc-macro2",
+ "quote",
+ "serde",
+ "serde_json",
+ "sha2",
+ "sqlx-core",
+ "sqlx-mysql",
+ "sqlx-postgres",
+ "sqlx-sqlite",
+ "syn 1.0.109",
+ "tempfile",
+ "tokio 1.32.0",
+ "url",
+]
+
+[[package]]
+name = "sqlx-mysql"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "864b869fdf56263f4c95c45483191ea0af340f9f3e3e7b4d57a61c7c87a970db"
+dependencies = [
+ "atoi",
+ "base64 0.21.4",
+ "bitflags 2.4.0",
+ "byteorder",
+ "bytes 1.5.0",
+ "chrono",
+ "crc",
+ "digest 0.10.7",
+ "dotenvy",
+ "either",
+ "futures-channel",
+ "futures-core",
+ "futures-io",
+ "futures-util",
+ "generic-array",
+ "hex",
+ "hkdf",
+ "hmac",
+ "itoa 1.0.9",
+ "log",
+ "md-5",
+ "memchr",
+ "once_cell",
+ "percent-encoding",
+ "rand 0.8.5",
+ "rsa",
+ "serde",
+ "sha1 0.10.6",
+ "sha2",
+ "smallvec",
+ "sqlx-core",
+ "stringprep",
+ "thiserror",
+ "tracing",
+ "whoami",
+]
+
+[[package]]
+name = "sqlx-postgres"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "eb7ae0e6a97fb3ba33b23ac2671a5ce6e3cabe003f451abd5a56e7951d975624"
+dependencies = [
+ "atoi",
+ "base64 0.21.4",
+ "bitflags 2.4.0",
+ "byteorder",
+ "chrono",
+ "crc",
+ "dotenvy",
+ "etcetera",
+ "futures-channel",
+ "futures-core",
+ "futures-io",
+ "futures-util",
+ "hex",
+ "hkdf",
+ "hmac",
+ "home",
+ "itoa 1.0.9",
+ "log",
+ "md-5",
+ "memchr",
+ "once_cell",
+ "rand 0.8.5",
+ "serde",
+ "serde_json",
+ "sha1 0.10.6",
+ "sha2",
+ "smallvec",
+ "sqlx-core",
+ "stringprep",
+ "thiserror",
+ "tracing",
+ "whoami",
+]
+
+[[package]]
+name = "sqlx-sqlite"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d59dc83cf45d89c555a577694534fcd1b55c545a816c816ce51f20bbe56a4f3f"
+dependencies = [
+ "atoi",
+ "chrono",
+ "flume",
+ "futures-channel",
+ "futures-core",
+ "futures-executor",
+ "futures-intrusive",
+ "futures-util",
+ "libsqlite3-sys",
+ "log",
+ "percent-encoding",
+ "serde",
+ "sqlx-core",
+ "tracing",
+ "url",
+]
+
+[[package]]
+name = "standback"
+version = "0.2.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e113fb6f3de07a243d434a56ec6f186dfd51cb08448239fe7bcae73f87ff28ff"
+dependencies = [
+ "version_check",
+]
+
+[[package]]
+name = "stdweb"
+version = "0.4.20"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d022496b16281348b52d0e30ae99e01a73d737b2f45d38fed4edf79f9325a1d5"
+dependencies = [
+ "discard",
+ "rustc_version 0.2.3",
+ "stdweb-derive",
+ "stdweb-internal-macros",
+ "stdweb-internal-runtime",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "stdweb-derive"
+version = "0.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c87a60a40fccc84bef0652345bbbbbe20a605bf5d0ce81719fc476f5c03b50ef"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "serde",
+ "serde_derive",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "stdweb-internal-macros"
+version = "0.2.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "58fa5ff6ad0d98d1ffa8cb115892b6e69d67799f6763e162a1c9db421dc22e11"
+dependencies = [
+ "base-x",
+ "proc-macro2",
+ "quote",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "sha1 0.6.1",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "stdweb-internal-runtime"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "213701ba3370744dcd1a12960caa4843b3d68b4d1c0a5d575e0d65b2ee9d16c0"
+
+[[package]]
+name = "stringprep"
+version = "0.1.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bb41d74e231a107a1b4ee36bd1214b11285b77768d2e3824aedafa988fd36ee6"
+dependencies = [
+ "finl_unicode",
+ "unicode-bidi",
+ "unicode-normalization",
+]
+
+[[package]]
+name = "strum"
+version = "0.25.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "290d54ea6f91c969195bdbcd7442c8c2a2ba87da8bf60a7ee86a235d4bc1e125"
+
+[[package]]
+name = "strum_macros"
+version = "0.25.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ad8d03b598d3d0fff69bf533ee3ef19b8eeb342729596df84bcc7e1f96ec4059"
+dependencies = [
+ "heck",
+ "proc-macro2",
+ "quote",
+ "rustversion",
+ "syn 2.0.37",
+]
+
+[[package]]
+name = "subtle"
+version = "2.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "81cdd64d312baedb58e21336b31bc043b77e01cc99033ce76ef539f78e965ebc"
+
+[[package]]
+name = "syn"
+version = "1.0.109"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "unicode-ident",
+]
+
+[[package]]
+name = "syn"
+version = "2.0.37"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7303ef2c05cd654186cb250d29049a24840ca25d2747c25c0381c8d9e2f582e8"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "unicode-ident",
+]
+
+[[package]]
+name = "tempfile"
+version = "3.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cb94d2f3cc536af71caac6b6fcebf65860b347e7ce0cc9ebe8f70d3e521054ef"
+dependencies = [
+ "cfg-if 1.0.0",
+ "fastrand 2.0.1",
+ "redox_syscall 0.3.5",
+ "rustix 0.38.15",
+ "windows-sys",
+]
+
+[[package]]
+name = "termcolor"
+version = "1.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6093bad37da69aab9d123a8091e4be0aa4a03e4d601ec641c327398315f62b64"
+dependencies = [
+ "winapi-util",
+]
+
+[[package]]
+name = "thiserror"
+version = "1.0.49"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1177e8c6d7ede7afde3585fd2513e611227efd6481bd78d2e82ba1ce16557ed4"
+dependencies = [
+ "thiserror-impl",
+]
+
+[[package]]
+name = "thiserror-impl"
+version = "1.0.49"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "10712f02019e9288794769fba95cd6847df9874d49d871d062172f9dd41bc4cc"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+]
+
+[[package]]
+name = "threadpool"
+version = "1.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d050e60b33d41c19108b32cea32164033a9013fe3b46cbd4457559bfbf77afaa"
+dependencies = [
+ "num_cpus",
+]
+
+[[package]]
+name = "time"
+version = "0.2.27"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4752a97f8eebd6854ff91f1c1824cd6160626ac4bd44287f7f4ea2035a02a242"
+dependencies = [
+ "const_fn",
+ "libc",
+ "standback",
+ "stdweb",
+ "time-macros",
+ "version_check",
+ "winapi 0.3.9",
+]
+
+[[package]]
+name = "time-macros"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "957e9c6e26f12cb6d0dd7fc776bb67a706312e7299aed74c8dd5b17ebb27e2f1"
+dependencies = [
+ "proc-macro-hack",
+ "time-macros-impl",
+]
+
+[[package]]
+name = "time-macros-impl"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fd3c141a1b43194f3f56a1411225df8646c55781d5f26db825b3d98507eb482f"
+dependencies = [
+ "proc-macro-hack",
+ "proc-macro2",
+ "quote",
+ "standback",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "tinyvec"
+version = "1.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "87cc5ceb3875bb20c2890005a4e226a4651264a5c75edb2421b52861a0a0cb50"
+dependencies = [
+ "tinyvec_macros",
+]
+
+[[package]]
+name = "tinyvec_macros"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20"
+
+[[package]]
+name = "tokio"
+version = "0.2.25"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6703a273949a90131b290be1fe7b039d0fc884aa1935860dfcbe056f28cd8092"
+dependencies = [
+ "bytes 0.5.6",
+ "futures-core",
+ "iovec",
+ "lazy_static",
+ "libc",
+ "memchr",
+ "mio 0.6.23",
+ "mio-uds",
+ "pin-project-lite 0.1.12",
+ "signal-hook-registry",
+ "slab",
+ "winapi 0.3.9",
+]
+
+[[package]]
+name = "tokio"
+version = "1.32.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "17ed6077ed6cd6c74735e21f37eb16dc3935f96878b1fe961074089cc80893f9"
+dependencies = [
+ "backtrace",
+ "bytes 1.5.0",
+ "libc",
+ "mio 0.8.8",
+ "num_cpus",
+ "parking_lot 0.12.1",
+ "pin-project-lite 0.2.13",
+ "signal-hook-registry",
+ "socket2 0.5.4",
+ "tokio-macros",
+ "windows-sys",
+]
+
+[[package]]
+name = "tokio-macros"
+version = "2.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "630bdcf245f78637c13ec01ffae6187cca34625e8c63150d424b59e55af2675e"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+]
+
+[[package]]
+name = "tokio-stream"
+version = "0.1.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "397c988d37662c7dda6d2208364a706264bf3d6138b11d436cbac0ad38832842"
+dependencies = [
+ "futures-core",
+ "pin-project-lite 0.2.13",
+ "tokio 1.32.0",
+]
+
+[[package]]
+name = "tokio-util"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "be8242891f2b6cbef26a2d7e8605133c2c554cd35b3e4948ea892d6d68436499"
+dependencies = [
+ "bytes 0.5.6",
+ "futures-core",
+ "futures-sink",
+ "log",
+ "pin-project-lite 0.1.12",
+ "tokio 0.2.25",
+]
+
+[[package]]
+name = "tracing"
+version = "0.1.37"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8ce8c33a8d48bd45d624a6e523445fd21ec13d3653cd51f681abf67418f54eb8"
+dependencies = [
+ "cfg-if 1.0.0",
+ "log",
+ "pin-project-lite 0.2.13",
+ "tracing-attributes",
+ "tracing-core",
+]
+
+[[package]]
+name = "tracing-attributes"
+version = "0.1.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5f4f31f56159e98206da9efd823404b79b6ef3143b4a7ab76e67b1751b25a4ab"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+]
+
+[[package]]
+name = "tracing-core"
+version = "0.1.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0955b8137a1df6f1a2e9a37d8a6656291ff0297c1a97c24e0d8425fe2312f79a"
+dependencies = [
+ "once_cell",
+]
+
+[[package]]
+name = "tracing-futures"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "97d095ae15e245a057c8e8451bab9b3ee1e1f68e9ba2b4fbc18d0ac5237835f2"
+dependencies = [
+ "pin-project 1.1.3",
+ "tracing",
+]
+
+[[package]]
+name = "trust-dns-proto"
+version = "0.19.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1cad71a0c0d68ab9941d2fb6e82f8fb2e86d9945b94e1661dd0aaea2b88215a9"
+dependencies = [
+ "async-trait",
+ "cfg-if 1.0.0",
+ "enum-as-inner",
+ "futures",
+ "idna 0.2.3",
+ "lazy_static",
+ "log",
+ "rand 0.7.3",
+ "smallvec",
+ "thiserror",
+ "tokio 0.2.25",
+ "url",
+]
+
+[[package]]
+name = "trust-dns-resolver"
+version = "0.19.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "710f593b371175db53a26d0b38ed2978fafb9e9e8d3868b1acd753ea18df0ceb"
+dependencies = [
+ "cfg-if 0.1.10",
+ "futures",
+ "ipconfig",
+ "lazy_static",
+ "log",
+ "lru-cache",
+ "resolv-conf",
+ "smallvec",
+ "thiserror",
+ "tokio 0.2.25",
+ "trust-dns-proto",
+]
+
+[[package]]
+name = "typenum"
+version = "1.17.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825"
+
+[[package]]
+name = "unicode-bidi"
+version = "0.3.13"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "92888ba5573ff080736b3648696b70cafad7d250551175acbaa4e0385b3e1460"
+
+[[package]]
+name = "unicode-ident"
+version = "1.0.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b"
+
+[[package]]
+name = "unicode-normalization"
+version = "0.1.22"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5c5713f0fc4b5db668a2ac63cdb7bb4469d8c9fed047b1d0292cc7b0ce2ba921"
+dependencies = [
+ "tinyvec",
+]
+
+[[package]]
+name = "unicode-segmentation"
+version = "1.10.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1dd624098567895118886609431a7c3b8f516e41d30e0643f03d94592a147e36"
+
+[[package]]
+name = "unicode_categories"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "39ec24b3121d976906ece63c9daad25b85969647682eee313cb5779fdd69e14e"
+
+[[package]]
+name = "url"
+version = "2.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "143b538f18257fac9cad154828a57c6bf5157e1aa604d4816b5995bf6de87ae5"
+dependencies = [
+ "form_urlencoded",
+ "idna 0.4.0",
+ "percent-encoding",
+]
+
+[[package]]
+name = "value-bag"
+version = "1.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d92ccd67fb88503048c01b59152a04effd0782d035a83a6d256ce6085f08f4a3"
+
+[[package]]
+name = "vcpkg"
+version = "0.2.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426"
+
+[[package]]
+name = "version_check"
+version = "0.9.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f"
+
+[[package]]
+name = "waker-fn"
+version = "1.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f3c4517f54858c779bbcbf228f4fca63d121bf85fbecb2dc578cdf4a39395690"
+
+[[package]]
+name = "wasi"
+version = "0.9.0+wasi-snapshot-preview1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519"
+
+[[package]]
+name = "wasi"
+version = "0.11.0+wasi-snapshot-preview1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423"
+
+[[package]]
+name = "wasm-bindgen"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7706a72ab36d8cb1f80ffbf0e071533974a60d0a308d01a5d0375bf60499a342"
+dependencies = [
+ "cfg-if 1.0.0",
+ "wasm-bindgen-macro",
+]
+
+[[package]]
+name = "wasm-bindgen-backend"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5ef2b6d3c510e9625e5fe6f509ab07d66a760f0885d858736483c32ed7809abd"
+dependencies = [
+ "bumpalo",
+ "log",
+ "once_cell",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-futures"
+version = "0.4.37"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c02dbc21516f9f1f04f187958890d7e6026df8d16540b7ad9492bc34a67cea03"
+dependencies = [
+ "cfg-if 1.0.0",
+ "js-sys",
+ "wasm-bindgen",
+ "web-sys",
+]
+
+[[package]]
+name = "wasm-bindgen-macro"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dee495e55982a3bd48105a7b947fd2a9b4a8ae3010041b9e0faab3f9cd028f1d"
+dependencies = [
+ "quote",
+ "wasm-bindgen-macro-support",
+]
+
+[[package]]
+name = "wasm-bindgen-macro-support"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "54681b18a46765f095758388f2d0cf16eb8d4169b639ab575a8f5693af210c7b"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.37",
+ "wasm-bindgen-backend",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-shared"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ca6ad05a4870b2bf5fe995117d3728437bd27d7cd5f06f13c17443ef369775a1"
+
+[[package]]
+name = "web-sys"
+version = "0.3.64"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9b85cbef8c220a6abc02aefd892dfc0fc23afb1c6a426316ec33253a3877249b"
+dependencies = [
+ "js-sys",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "whoami"
+version = "1.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "22fc3756b8a9133049b26c7f61ab35416c130e8c09b660f5b3958b446f52cc50"
+
+[[package]]
+name = "widestring"
+version = "0.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c168940144dd21fd8046987c16a46a33d5fc84eec29ef9dcddc2ac9e31526b7c"
+
+[[package]]
+name = "winapi"
+version = "0.2.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "167dc9d6949a9b857f3451275e911c3f44255842c1f7a76f33c55103a909087a"
+
+[[package]]
+name = "winapi"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419"
+dependencies = [
+ "winapi-i686-pc-windows-gnu",
+ "winapi-x86_64-pc-windows-gnu",
+]
+
+[[package]]
+name = "winapi-build"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2d315eee3b34aca4797b2da6b13ed88266e6d612562a0c46390af8299fc699bc"
+
+[[package]]
+name = "winapi-i686-pc-windows-gnu"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
+
+[[package]]
+name = "winapi-util"
+version = "0.1.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f29e6f9198ba0d26b4c9f07dbe6f9ed633e1f3d5b8b414090084349e46a52596"
+dependencies = [
+ "winapi 0.3.9",
+]
+
+[[package]]
+name = "winapi-x86_64-pc-windows-gnu"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
+
+[[package]]
+name = "windows"
+version = "0.48.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e686886bc078bc1b0b600cac0147aadb815089b6e4da64016cbd754b6342700f"
+dependencies = [
+ "windows-targets",
+]
+
+[[package]]
+name = "windows-sys"
+version = "0.48.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9"
+dependencies = [
+ "windows-targets",
+]
+
+[[package]]
+name = "windows-targets"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c"
+dependencies = [
+ "windows_aarch64_gnullvm",
+ "windows_aarch64_msvc",
+ "windows_i686_gnu",
+ "windows_i686_msvc",
+ "windows_x86_64_gnu",
+ "windows_x86_64_gnullvm",
+ "windows_x86_64_msvc",
+]
+
+[[package]]
+name = "windows_aarch64_gnullvm"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8"
+
+[[package]]
+name = "windows_aarch64_msvc"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc"
+
+[[package]]
+name = "windows_i686_gnu"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e"
+
+[[package]]
+name = "windows_i686_msvc"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406"
+
+[[package]]
+name = "windows_x86_64_gnu"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e"
+
+[[package]]
+name = "windows_x86_64_gnullvm"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc"
+
+[[package]]
+name = "windows_x86_64_msvc"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538"
+
+[[package]]
+name = "winreg"
+version = "0.6.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b2986deb581c4fe11b621998a5e53361efe6b48a151178d0cd9eeffa4dc6acc9"
+dependencies = [
+ "winapi 0.3.9",
+]
+
+[[package]]
+name = "ws2_32-sys"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d59cefebd0c892fa2dd6de581e937301d8552cb44489cdff035c6187cb63fa5e"
+dependencies = [
+ "winapi 0.2.8",
+ "winapi-build",
+]
+
+[[package]]
+name = "zeroize"
+version = "1.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2a0956f1ba7c7909bfb66c2e9e4124ab6f6482560f6628b5aaeba39207c9aad9"

+ 16 - 0
Cargo.toml

@@ -0,0 +1,16 @@
+[package]
+name = "ledger"
+version = "0.1.0"
+edition = "2021"
+
+# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
+[workspace]
+members = ["utxo"]
+
+[dependencies]
+ledger-utxo = { path = "utxo", features = ["sqlite"] }
+actix-web = "3"
+serde = { version = "1", features = ["derive"] }
+serde_json = "1"
+tokio = { version = "1.32.0", features = ["full"] }
+env_logger = "0.10.0"

+ 7 - 0
TODO.md

@@ -0,0 +1,7 @@
+* [ ] Add caching layer: This cache layer can built on top of the utxo::ledger,
+  because all operations can be safely cached until a new transaction
+  referencing their account is issued, by that point, all the caches related to
+  anaccount can be evicted
+* [ ] Build admin interface
+* [ ] ADd memo to changes. Build append only table with all movements as
+  inserts. Wraps the objects to all their changes

+ 245 - 0
src/main.rs

@@ -0,0 +1,245 @@
+use std::sync::Arc;
+
+use actix_web::{
+    error::InternalError, get, middleware::Logger, post, web, App, HttpResponse, HttpServer,
+    Responder,
+};
+use ledger_utxo::{AccountId, AnyId, AssetDefinition, AssetManager, Status, TransactionId};
+use serde::{Deserialize, Serialize};
+use serde_json::json;
+
+#[derive(Deserialize)]
+pub struct Movement {
+    pub account: AccountId,
+    pub amount: String,
+    pub asset: String,
+}
+
+#[derive(Deserialize)]
+pub struct Deposit {
+    pub account: AccountId,
+    pub amount: String,
+    pub asset: String,
+    pub memo: String,
+    pub status: Status,
+}
+
+impl Deposit {
+    pub async fn to_ledger_transaction(
+        self,
+        ledger: &Ledger,
+    ) -> Result<ledger_utxo::Transaction, ledger_utxo::Error> {
+        let amount = ledger._inner.parse_amount(&self.asset, &self.amount)?;
+        ledger
+            ._inner
+            .deposit(&self.account, amount, self.status, self.memo)
+            .await
+    }
+}
+
+#[derive(Deserialize)]
+pub struct Transaction {
+    pub debit: Vec<Movement>,
+    pub credit: Vec<Movement>,
+    pub memo: String,
+    pub status: Status,
+}
+
+#[derive(Deserialize)]
+pub struct UpdateTransaction {
+    pub status: Status,
+    pub memo: String,
+}
+
+impl UpdateTransaction {
+    pub async fn to_ledger_transaction(
+        self,
+        id: &TransactionId,
+        ledger: &Ledger,
+    ) -> Result<ledger_utxo::Transaction, ledger_utxo::Error> {
+        ledger._inner.change_status(id, self.status).await
+    }
+}
+
+impl Transaction {
+    pub async fn to_ledger_transaction(
+        self,
+        ledger: &Ledger,
+    ) -> Result<ledger_utxo::Transaction, ledger_utxo::Error> {
+        let from = self
+            .debit
+            .into_iter()
+            .map(|x| {
+                ledger
+                    ._inner
+                    .parse_amount(&x.asset, &x.amount)
+                    .map(|amount| (x.account, amount))
+            })
+            .collect::<Result<Vec<_>, _>>()?;
+
+        let to = self
+            .credit
+            .into_iter()
+            .map(|x| {
+                ledger
+                    ._inner
+                    .parse_amount(&x.asset, &x.amount)
+                    .map(|amount| (x.account, amount))
+            })
+            .collect::<Result<Vec<_>, _>>()?;
+
+        ledger
+            ._inner
+            .new_transaction(self.memo, self.status, from, to)
+            .await
+    }
+}
+
+#[derive(Serialize, Deserialize)]
+struct Item {
+    id: i32,
+    name: String,
+}
+
+#[derive(Serialize)]
+struct AccountResponse {
+    amount: String,
+    asset: Arc<str>,
+}
+
+#[get("/balance/{id}")]
+async fn get_balance(info: web::Path<AccountId>, ledger: web::Data<Ledger>) -> impl Responder {
+    let asset_manager = ledger._inner.asset_manager();
+    match ledger._inner.get_balance(&info.0).await {
+        Ok(balances) => {
+            match balances
+                .into_iter()
+                .map(|amount| {
+                    asset_manager
+                        .get_name(amount.asset().id)
+                        .map(|asset| AccountResponse {
+                            amount: amount.to_string(),
+                            asset: asset,
+                        })
+                })
+                .collect::<Result<Vec<_>, _>>()
+            {
+                Ok(balances) => HttpResponse::Ok().json(balances),
+                Err(e) => HttpResponse::BadRequest().json(e),
+            }
+        }
+        Err(e) => HttpResponse::BadRequest().json(e),
+    }
+}
+
+#[get("/{id}")]
+async fn get_info(info: web::Path<AnyId>, ledger: web::Data<Ledger>) -> impl Responder {
+    let result = match info.0 {
+        AnyId::Account(account_id) => ledger
+            ._inner
+            .get_transactions(&account_id, None)
+            .await
+            .map(|transactions| HttpResponse::Ok().json(transactions)),
+        AnyId::Transaction(transaction_id) => ledger
+            ._inner
+            .get_transaction(&transaction_id)
+            .await
+            .map(|tx| HttpResponse::Ok().json(tx)),
+    };
+
+    match result {
+        Ok(x) => x,
+        Err(err) => HttpResponse::InternalServerError().json(err),
+    }
+}
+
+#[post("/deposit")]
+async fn deposit(item: web::Json<Deposit>, ledger: web::Data<Ledger>) -> impl Responder {
+    // Insert the item into a database or another data source.
+    // For this example, we'll just echo the received item.
+    match item.into_inner().to_ledger_transaction(&ledger).await {
+        Ok(tx) => {
+            // Insert the item into a database or another data source.
+            // For this example, we'll just echo the received item.
+            HttpResponse::Created().json(tx)
+        }
+        Err(e) => HttpResponse::Created().json(e),
+    }
+}
+
+#[post("/tx")]
+async fn create_transaction(
+    item: web::Json<Transaction>,
+    ledger: web::Data<Ledger>,
+) -> impl Responder {
+    match item.into_inner().to_ledger_transaction(&ledger).await {
+        Ok(tx) => HttpResponse::Accepted().json(tx),
+        Err(err) => HttpResponse::Created().json(err),
+    }
+}
+
+#[post("/{id}")]
+async fn update_status(
+    info: web::Path<TransactionId>,
+    item: web::Json<UpdateTransaction>,
+    ledger: web::Data<Ledger>,
+) -> impl Responder {
+    match item
+        .into_inner()
+        .to_ledger_transaction(&info.0, &ledger)
+        .await
+    {
+        Ok(tx) => HttpResponse::Accepted().json(tx),
+        Err(err) => HttpResponse::Created().json(err),
+    }
+}
+
+pub struct Ledger {
+    _inner: ledger_utxo::Ledger<'static, ledger_utxo::Batch<'static>, ledger_utxo::Sqlite<'static>>,
+}
+
+#[actix_web::main]
+async fn main() -> std::io::Result<()> {
+    if std::env::var_os("RUST_LOG").is_none() {
+        std::env::set_var("RUST_LOG", "actix_web=info");
+    }
+    env_logger::init_from_env(env_logger::Env::new().default_filter_or("info"));
+
+    let asset_manager = AssetManager::new(vec![
+        AssetDefinition::new(1, "BTC", 8),
+        AssetDefinition::new(2, "USD", 4),
+    ]);
+
+    let pool = ledger_utxo::sqlite::SqlitePoolOptions::new()
+        .connect("sqlite:./test.db")
+        .await
+        .expect("pool");
+
+    let storage = ledger_utxo::Sqlite::new(pool.clone(), asset_manager.clone());
+    storage.setup().await.expect("setup");
+
+    HttpServer::new(move || {
+        let storage = ledger_utxo::Sqlite::new(pool.clone(), asset_manager.clone());
+        let ledger = ledger_utxo::Ledger::new(storage, asset_manager.clone());
+        App::new()
+            .wrap(Logger::default())
+            .app_data(web::Data::new(Ledger { _inner: ledger }))
+            .app_data(web::JsonConfig::default().error_handler(|err, _req| {
+                InternalError::from_response(
+                    "",
+                    HttpResponse::BadRequest()
+                        .content_type("application/json")
+                        .body(format!(r#"{{"error":"{}"}}"#, err)),
+                )
+                .into()
+            }))
+            .service(get_balance)
+            .service(get_info)
+            .service(deposit)
+            .service(create_transaction)
+            .service(update_status)
+    })
+    .bind("127.0.0.1:8080")?
+    .run()
+    .await
+}

+ 136 - 0
utxo/Cargo.lock

@@ -36,6 +36,21 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "0942ffc6dcaadf03badf6e6a2d0228460359d5e34b57ccdc720b7382dfbd5ec5"
 
 [[package]]
+name = "android-tzdata"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e999941b234f3131b00bc13c22d06e8c5ff726d1b6318ac7eb276997bbb4fef0"
+
+[[package]]
+name = "android_system_properties"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "819e7219dbd41043ac279b19830f2efc897156490d7fd6ea916720117ee66311"
+dependencies = [
+ "libc",
+]
+
+[[package]]
 name = "async-trait"
 version = "0.1.73"
 source = "registry+https://github.com/rust-lang/crates.io-index"
@@ -122,6 +137,12 @@ dependencies = [
 ]
 
 [[package]]
+name = "bumpalo"
+version = "3.14.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7f30e7476521f6f8af1a1c4c0b8cc94f0bee37d91763d0ca2665f299b6cd8aec"
+
+[[package]]
 name = "byteorder"
 version = "1.4.3"
 source = "registry+https://github.com/rust-lang/crates.io-index"
@@ -149,6 +170,20 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
 
 [[package]]
+name = "chrono"
+version = "0.4.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7f2c685bad3eb3d45a01354cedb7d5faa66194d1d58ba6e267a8de788f79db38"
+dependencies = [
+ "android-tzdata",
+ "iana-time-zone",
+ "js-sys",
+ "num-traits",
+ "wasm-bindgen",
+ "windows-targets",
+]
+
+[[package]]
 name = "const-oid"
 version = "0.9.5"
 source = "registry+https://github.com/rust-lang/crates.io-index"
@@ -548,6 +583,29 @@ dependencies = [
 ]
 
 [[package]]
+name = "iana-time-zone"
+version = "0.1.57"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2fad5b825842d2b38bd206f3e81d6957625fd7f0a361e345c30e01a0ae2dd613"
+dependencies = [
+ "android_system_properties",
+ "core-foundation-sys",
+ "iana-time-zone-haiku",
+ "js-sys",
+ "wasm-bindgen",
+ "windows",
+]
+
+[[package]]
+name = "iana-time-zone-haiku"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f"
+dependencies = [
+ "cc",
+]
+
+[[package]]
 name = "idna"
 version = "0.4.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
@@ -583,6 +641,15 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "af150ab688ff2122fcef229be89cb50dd66af9e01a4ff320cc137eecc9bacc38"
 
 [[package]]
+name = "js-sys"
+version = "0.3.64"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c5f195fe497f702db0f318b07fdd68edb16955aed830df8363d837542f8f935a"
+dependencies = [
+ "wasm-bindgen",
+]
+
+[[package]]
 name = "lazy_static"
 version = "1.4.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
@@ -597,6 +664,7 @@ version = "0.1.0"
 dependencies = [
  "async-trait",
  "bincode",
+ "chrono",
  "futures",
  "hex",
  "serde",
@@ -1252,6 +1320,7 @@ dependencies = [
  "atoi",
  "byteorder",
  "bytes",
+ "chrono",
  "crc",
  "crossbeam-queue",
  "dotenvy",
@@ -1314,6 +1383,7 @@ dependencies = [
  "sha2",
  "sqlx-core",
  "sqlx-mysql",
+ "sqlx-postgres",
  "sqlx-sqlite",
  "syn 1.0.109",
  "tempfile",
@@ -1332,6 +1402,7 @@ dependencies = [
  "bitflags 2.4.0",
  "byteorder",
  "bytes",
+ "chrono",
  "crc",
  "digest",
  "dotenvy",
@@ -1373,6 +1444,7 @@ dependencies = [
  "base64",
  "bitflags 2.4.0",
  "byteorder",
+ "chrono",
  "crc",
  "dotenvy",
  "etcetera",
@@ -1409,6 +1481,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "be4c21bf34c7cae5b283efb3ac1bcc7670df7561124dc2f8bdc0b59be40f79a2"
 dependencies = [
  "atoi",
+ "chrono",
  "flume",
  "futures-channel",
  "futures-core",
@@ -1673,12 +1746,75 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423"
 
 [[package]]
+name = "wasm-bindgen"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7706a72ab36d8cb1f80ffbf0e071533974a60d0a308d01a5d0375bf60499a342"
+dependencies = [
+ "cfg-if",
+ "wasm-bindgen-macro",
+]
+
+[[package]]
+name = "wasm-bindgen-backend"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5ef2b6d3c510e9625e5fe6f509ab07d66a760f0885d858736483c32ed7809abd"
+dependencies = [
+ "bumpalo",
+ "log",
+ "once_cell",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.31",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-macro"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dee495e55982a3bd48105a7b947fd2a9b4a8ae3010041b9e0faab3f9cd028f1d"
+dependencies = [
+ "quote",
+ "wasm-bindgen-macro-support",
+]
+
+[[package]]
+name = "wasm-bindgen-macro-support"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "54681b18a46765f095758388f2d0cf16eb8d4169b639ab575a8f5693af210c7b"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.31",
+ "wasm-bindgen-backend",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-shared"
+version = "0.2.87"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ca6ad05a4870b2bf5fe995117d3728437bd27d7cd5f06f13c17443ef369775a1"
+
+[[package]]
 name = "whoami"
 version = "1.4.1"
 source = "registry+https://github.com/rust-lang/crates.io-index"
 checksum = "22fc3756b8a9133049b26c7f61ab35416c130e8c09b660f5b3958b446f52cc50"
 
 [[package]]
+name = "windows"
+version = "0.48.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e686886bc078bc1b0b600cac0147aadb815089b6e4da64016cbd754b6342700f"
+dependencies = [
+ "windows-targets",
+]
+
+[[package]]
 name = "windows-sys"
 version = "0.48.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"

+ 9 - 2
utxo/Cargo.toml

@@ -7,11 +7,18 @@ edition = "2021"
 [dependencies]
 async-trait = "0.1.73"
 bincode = { version = "1.3.3", features = ["i128"] }
+chrono = { version = "0.4.31", features = ["serde"] }
 futures = "0.3.28"
 hex = "0.4.3"
 serde = { version = "1.0.188", features = ["derive"] }
 sha2 = "0.10.7"
-sqlx = { version = "0.7.1", features = ["runtime-tokio", "tls-native-tls", "sqlite"] }
+sqlx = { version = "0.7.1", features = [
+    "runtime-tokio",
+    "runtime-async-std-native-tls",
+    "tls-native-tls",
+    "sqlite",
+    "chrono",
+], optional = true }
 strum = "0.25.0"
 strum_macros = "0.25.2"
 thiserror = "1.0.48"
@@ -22,4 +29,4 @@ sqlx = { version = "0.7.1", features = ["sqlite"] }
 
 [features]
 default = []
-sqlite = ["sqlx/sqlite"]
+sqlite = ["sqlx"]

TEMPAT SAMPAH
utxo/src/.DS_Store


+ 136 - 3
utxo/src/amount.rs

@@ -1,8 +1,15 @@
 use crate::Asset;
-use serde::Serialize;
+use serde::{Serialize, Serializer};
 
+/// The raw storage for cents, the more the better
 pub type AmountCents = i128;
 
+#[derive(Clone, Debug, Eq, PartialEq, thiserror::Error)]
+pub enum Error {
+    #[error("{0} is not a valid number")]
+    NoANumber(String),
+}
+
 /// Amount
 ///
 /// The cents are stored in their lowest denomination, or their "cents".  For
@@ -17,19 +24,67 @@ pub type AmountCents = i128;
 /// operation.
 ///
 ///
-/// The `cents` and `Asset.id` must be used to store amounts in the storge
+/// The `cents` and `Asset.id` must be used to store amounts in the storage
 /// layer. Float or string representations should be used to display
-#[derive(Copy, Clone, Debug, Eq, PartialEq, Serialize)]
+#[derive(Copy, Clone, Debug, Eq, PartialEq)]
 pub struct Amount {
     asset: Asset,
     cents: AmountCents,
 }
 
+impl Serialize for Amount {
+    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
+    where
+        S: Serializer,
+    {
+        let serialized = self.to_string();
+        serializer.serialize_str(&serialized)
+    }
+}
+
 impl Amount {
     pub fn new(asset: Asset, cents: AmountCents) -> Self {
         Self { asset, cents }
     }
 
+    pub fn from_human(asset: Asset, human_amount: &str) -> Result<Self, Error> {
+        let mut dot_at = None;
+        for (pos, i) in human_amount.chars().enumerate() {
+            match i {
+                '-' => {
+                    if pos != 0 {
+                        return Err(Error::NoANumber(human_amount.to_owned()));
+                    }
+                }
+                '.' => {
+                    if dot_at.is_some() {
+                        return Err(Error::NoANumber(human_amount.to_owned()));
+                    }
+                    dot_at = Some(pos);
+                }
+                '0'..='9' => {}
+                _ => {
+                    return Err(Error::NoANumber(human_amount.to_owned()));
+                }
+            }
+        }
+
+        let (whole, fractional_part) = if let Some(dot_at) = dot_at {
+            let (whole, fractional_part) = human_amount.split_at(dot_at);
+            (whole, fractional_part[1..].to_owned())
+        } else {
+            (human_amount, "".to_owned())
+        };
+
+        let fractional_part = fractional_part + &"0".repeat(asset.precision.into());
+
+        let cents = (whole.to_owned() + &fractional_part[..asset.precision.into()])
+            .parse::<AmountCents>()
+            .map_err(|_| Error::NoANumber(format!("{}.{}", whole, fractional_part)))?;
+
+        Ok(Self { asset, cents })
+    }
+
     #[inline]
     pub fn asset(&self) -> &Asset {
         &self.asset
@@ -51,3 +106,81 @@ impl Amount {
         })
     }
 }
+
+impl ToString for Amount {
+    fn to_string(&self) -> String {
+        let str = self.cents.abs().to_string();
+        let precision: usize = self.asset.precision.into();
+        let str = if str.len() < precision + 1 {
+            format!("{}{}", "0".repeat(precision - str.len() + 1), str)
+        } else {
+            str
+        };
+
+        let (left, right) = str.split_at(str.len() - precision);
+        format!(
+            "{}{}.{}",
+            if self.cents.is_negative() { "-" } else { "" },
+            left,
+            right
+        )
+        .trim_end_matches("0")
+        .trim_end_matches(".")
+        .to_owned()
+    }
+}
+
+#[cfg(test)]
+mod test {
+    use super::*;
+
+    #[test]
+    fn dollar() {
+        let usd = Asset {
+            id: 1,
+            precision: 4,
+        };
+        let amount = usd.new_amount(1022100);
+        assert_eq!(amount.to_string(), "102.21");
+    }
+
+    #[test]
+    fn bitcoin() {
+        let btc = Asset {
+            id: 1,
+            precision: 8,
+        };
+        assert_eq!(btc.new_amount(1022100).to_string(), "0.010221");
+        assert_eq!(btc.new_amount(10).to_string(), "0.0000001");
+        assert_eq!(btc.new_amount(10000000).to_string(), "0.1");
+        assert_eq!(btc.new_amount(100000000).to_string(), "1");
+        assert_eq!(
+            btc.new_amount(100000000)
+                .checked_add(&btc.new_amount(100000000))
+                .unwrap()
+                .to_string(),
+            "2",
+        );
+        assert_eq!(btc.new_amount(1000000000).to_string(), "10");
+    }
+
+    #[test]
+    fn from_human() {
+        let btc = Asset {
+            id: 1,
+            precision: 8,
+        };
+        let parsed_amount = Amount::from_human(btc, "0.1").expect("valid amount");
+        assert_eq!(parsed_amount.to_string(), "0.1");
+        let parsed_amount = Amount::from_human(btc, "-0.1").expect("valid amount");
+        assert_eq!(parsed_amount.to_string(), "-0.1");
+        let parsed_amount = Amount::from_human(btc, "-0.000001").expect("valid amount");
+        assert_eq!(parsed_amount.to_string(), "-0.000001");
+        let parsed_amount = Amount::from_human(btc, "-0.000000001").expect("valid amount");
+        assert_eq!(parsed_amount.to_string(), "0");
+        let parsed_amount = Amount::from_human(btc, "0.000001").expect("valid amount");
+        assert_eq!(parsed_amount.to_string(), "0.000001");
+        let parsed_amount = Amount::from_human(btc, "-0.000000100001").expect("valid amount");
+        assert_eq!(parsed_amount.to_string(), "-0.0000001");
+    }
+}

+ 23 - 8
utxo/src/asset_manager.rs

@@ -4,6 +4,7 @@ use std::{collections::HashMap, sync::Arc};
 #[derive(Clone, Debug, Eq, PartialEq)]
 pub struct AssetManager {
     assets: Arc<HashMap<AssetId, AssetDefinition>>,
+    asset_names: Arc<HashMap<Arc<str>, AssetDefinition>>,
 }
 
 #[derive(Clone, Debug, Eq, PartialEq, serde::Deserialize)]
@@ -18,28 +19,42 @@ impl AssetManager {
         Self {
             assets: Arc::new(
                 assets
+                    .iter()
+                    .map(|asset| (asset.asset.id, asset.clone()))
+                    .collect(),
+            ),
+            asset_names: Arc::new(
+                assets
                     .into_iter()
-                    .map(|asset| (asset.asset.id, asset))
+                    .map(|asset| (asset.name.clone(), asset))
                     .collect(),
             ),
         }
     }
 
+    pub fn get_name(&self, id: AssetId) -> Result<Arc<str>, Error> {
+        self.assets
+            .get(&id)
+            .map(|asset| asset.name.clone())
+            .ok_or(Error::AssetIdNotFound(id))
+    }
+
     pub fn asset(&self, id: AssetId) -> Result<Asset, Error> {
         self.assets
             .get(&id)
             .map(|asset| asset.asset)
-            .ok_or(Error::AssetNotFound(id))
+            .ok_or(Error::AssetIdNotFound(id))
     }
 
-    pub fn asset_name(&self, asset: &Asset) -> Result<Arc<str>, Error> {
-        self.assets
-            .get(&asset.id)
-            .map(|asset| asset.name.clone())
-            .ok_or(Error::AssetNotFound(asset.id))
+    pub fn human_amount_by_name(&self, name: &str, human_amount: &str) -> Result<Amount, Error> {
+        Ok(self
+            .asset_names
+            .get(name)
+            .map(|asset| Amount::from_human(asset.asset, human_amount))
+            .ok_or(Error::AssetNotFound(name.to_owned()))??)
     }
 
-    pub fn amount(&self, id: AssetId, cents: AmountCents) -> Result<Amount, Error> {
+    pub fn amount_by_and_cents(&self, id: AssetId, cents: AmountCents) -> Result<Amount, Error> {
         self.asset(id).map(|asset| Amount::new(asset, cents))
     }
 }

+ 19 - 2
utxo/src/error.rs

@@ -1,4 +1,5 @@
-use crate::{asset::AssetId, storage, transaction, AccountId};
+use crate::{amount, asset::AssetId, storage, transaction, AccountId};
+use serde::{Serialize, Serializer};
 
 #[derive(thiserror::Error, Debug)]
 pub enum Error {
@@ -9,8 +10,24 @@ pub enum Error {
     Storage(#[from] storage::Error),
 
     #[error("Asset {0} is not defined")]
-    AssetNotFound(AssetId),
+    AssetIdNotFound(AssetId),
+
+    #[error("Asset {0} is not defined")]
+    AssetNotFound(String),
 
     #[error("Not enough funds (asset {1}) for account {0}")]
     InsufficientBalance(AccountId, AssetId),
+
+    #[error("Invalid amount: {0}")]
+    InvalidAmount(#[from] amount::Error),
+}
+
+impl Serialize for Error {
+    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
+    where
+        S: Serializer,
+    {
+        let serialized = self.to_string();
+        serializer.serialize_str(&serialized)
+    }
 }

+ 99 - 3
utxo/src/id.rs

@@ -1,16 +1,20 @@
-use serde::Serialize;
+use serde::{Deserialize, Deserializer, Serialize};
 use sha2::{Digest, Sha256};
 use std::fmt::Display;
+use std::str::FromStr;
 
 #[derive(thiserror::Error, Debug)]
 pub enum Error {
     #[error("Invalid length for {0}: {1} (expected: {2})")]
     InvalidLength(String, usize, usize),
+
+    #[error("Unknown prefix {0}")]
+    UnknownPrefix(String),
 }
 
 macro_rules! Id {
     ($id:ident, $suffix:expr) => {
-        #[derive(Clone, Copy, Debug, Eq, Hash, PartialEq, Serialize)]
+        #[derive(Clone, Debug, Eq, PartialOrd, Ord, Hash, PartialEq)]
         pub struct $id {
             bytes: [u8; 32],
         }
@@ -21,7 +25,7 @@ macro_rules! Id {
             }
         }
 
-        impl std::str::FromStr for $id {
+        impl FromStr for $id {
             type Err = Error;
             fn from_str(value: &str) -> Result<Self, Self::Err> {
                 Ok(Self::try_from(value).unwrap_or_else(|_| {
@@ -34,6 +38,26 @@ macro_rules! Id {
             }
         }
 
+        impl Serialize for $id {
+            fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
+            where
+                S: serde::Serializer,
+            {
+                serializer.collect_str(&self)
+            }
+        }
+
+        impl<'de> Deserialize<'de> for $id {
+            fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
+            where
+                D: Deserializer<'de>,
+            {
+                let s = String::deserialize(deserializer)?;
+                // Use FromStr to parse the string and construct the struct
+                $id::from_str(&s).map_err(serde::de::Error::custom)
+            }
+        }
+
         impl TryFrom<&str> for $id {
             type Error = Error;
             fn try_from(value: &str) -> Result<Self, Self::Error> {
@@ -60,6 +84,23 @@ macro_rules! Id {
             }
         }
 
+        impl TryFrom<&[u8]> for $id {
+            type Error = Error;
+
+            fn try_from(value: &[u8]) -> Result<Self, Self::Error> {
+                if value.len() != 32 {
+                    return Err(Error::InvalidLength(
+                        stringify!($id).to_owned(),
+                        value.len(),
+                        32,
+                    ));
+                }
+                let mut bytes = [0u8; 32];
+                bytes.copy_from_slice(&value);
+                Ok(Self { bytes })
+            }
+        }
+
         impl TryFrom<Vec<u8>> for $id {
             type Error = Error;
 
@@ -99,3 +140,58 @@ macro_rules! Id {
 
 Id!(AccountId, "account");
 Id!(TransactionId, "tx");
+
+#[derive(Debug)]
+pub enum AnyId {
+    Account(AccountId),
+    Transaction(TransactionId),
+}
+
+impl FromStr for AnyId {
+    type Err = Error;
+
+    fn from_str(value: &str) -> Result<Self, Self::Err> {
+        if value.starts_with("account") {
+            Ok(Self::Account(value.parse()?))
+        } else if value.starts_with("tx") {
+            Ok(Self::Transaction(value.parse()?))
+        } else {
+            Err(Error::UnknownPrefix(value.to_owned()))
+        }
+    }
+}
+
+impl<'de> Deserialize<'de> for AnyId {
+    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
+    where
+        D: Deserializer<'de>,
+    {
+        let s = String::deserialize(deserializer)?;
+        AnyId::from_str(&s).map_err(serde::de::Error::custom)
+    }
+}
+
+#[cfg(test)]
+mod test {
+    use super::*;
+
+    #[test]
+    fn from_random_data() {
+        let id = "something".parse::<AccountId>().expect("hashed value");
+        let id_str = id.to_string();
+        let id_bin: &[u8] = id.as_ref();
+        assert_eq!(71, id_str.len());
+        assert_eq!(
+            <&str as TryInto<AccountId>>::try_into(id_str.as_str()).expect("valid"),
+            id
+        );
+        assert_eq!(
+            <Vec<u8> as TryInto<AccountId>>::try_into(id_bin.to_owned()).expect("valid"),
+            id
+        );
+        assert_eq!(
+            <&[u8] as TryInto<AccountId>>::try_into(id_bin).expect("valid"),
+            id
+        );
+    }
+}

+ 82 - 15
utxo/src/ledger.rs

@@ -1,5 +1,7 @@
 use crate::{
-    AccountId, Amount, Batch, Error, Payment, Status, Storage, Transaction, TransactionId,
+    storage::{Batch, Storage},
+    transaction::Type,
+    AccountId, Amount, AssetManager, Error, Payment, Status, Transaction, TransactionId,
 };
 use std::{cmp::Ordering, collections::HashMap};
 
@@ -10,6 +12,7 @@ where
     S: Storage<'a, B> + Sync + Send,
 {
     storage: S,
+    asset_manager: AssetManager,
     _phantom: std::marker::PhantomData<&'a B>,
 }
 
@@ -18,13 +21,22 @@ where
     B: Batch<'a>,
     S: Storage<'a, B> + Sync + Send,
 {
-    pub fn new(storage: S) -> Self {
+    pub fn new(storage: S, asset_manager: AssetManager) -> Self {
         Self {
             storage,
+            asset_manager,
             _phantom: std::marker::PhantomData,
         }
     }
 
+    pub fn parse_amount(&self, asset: &str, amount: &str) -> Result<Amount, Error> {
+        Ok(self.asset_manager.human_amount_by_name(asset, amount)?)
+    }
+
+    pub fn asset_manager(&self) -> &AssetManager {
+        &self.asset_manager
+    }
+
     /// Selects the unspent payments to be used as inputs of the new transaction.
     ///
     /// This function also returns a list of transactions that will be used as
@@ -48,7 +60,7 @@ where
             }
         }
 
-        let mut change = vec![];
+        let mut change_transactions = vec![];
         let mut payments: Vec<Payment> = vec![];
 
         for ((account, asset), mut to_spend_cents) in to_spend.into_iter() {
@@ -77,9 +89,9 @@ where
                         let to_spend_cents = to_spend_cents.abs();
                         let input = payments
                             .pop()
-                            .ok_or(Error::InsufficientBalance(account, asset.id))?;
+                            .ok_or(Error::InsufficientBalance(account.clone(), asset.id))?;
                         let split_input = Transaction::new(
-                            "Exchange".to_owned(),
+                            "Exchange transaction".to_owned(),
                             // Set the change transaction as settled. This is an
                             // internal transaction to split an existing payment
                             // into two. Since this is an internal transaction it
@@ -90,18 +102,19 @@ where
                             // otherwise it would be locked until the main
                             // transaction settles.
                             Status::Settled,
+                            Type::Internal,
                             vec![input],
                             vec![
-                                (account, asset.new_amount(cents - to_spend_cents)),
-                                (account, asset.new_amount(to_spend_cents)),
+                                (account.clone(), asset.new_amount(cents - to_spend_cents)),
+                                (account.clone(), asset.new_amount(to_spend_cents)),
                             ],
                         )
                         .await?;
                         // Spend the new payment
-                        payments.push(split_input.created()[0].clone());
+                        payments.push(split_input.creates()[0].clone());
                         // Return the split payment transaction to be executed
                         // later as a pre-requisite for the new transaction
-                        change.push(split_input);
+                        change_transactions.push(split_input);
 
                         // Go to the next payment input or exit the loop
                         break;
@@ -120,7 +133,7 @@ where
             }
         }
 
-        Ok((change, payments))
+        Ok((change_transactions, payments))
     }
 
     /// Creates a new transaction and returns it.
@@ -156,10 +169,9 @@ where
             change_tx.persist(&self.storage).await?;
         }
 
-        let mut transaction = Transaction::new(reference, status, payments, to).await?;
-
+        let mut transaction =
+            Transaction::new(reference, status, Type::Transaction, payments, to).await?;
         transaction.persist(&self.storage).await?;
-
         Ok(transaction)
     }
 
@@ -182,17 +194,72 @@ where
         reference: String,
     ) -> Result<Transaction, Error> {
         let mut transaction =
-            Transaction::new_external_deposit(reference, status, vec![(*account, amount)])?;
+            Transaction::new_external_deposit(reference, status, vec![(account.clone(), amount)])?;
         transaction.persist(&self.storage).await?;
         Ok(transaction)
     }
 
+    pub async fn withdrawal(
+        &'a self,
+        account: &AccountId,
+        amount: Amount,
+        status: Status,
+        reference: String,
+    ) -> Result<Transaction, Error> {
+        let (change_transactions, payments) = self
+            .create_inputs_to_pay_from_accounts(vec![(account.clone(), amount)])
+            .await?;
+        for mut change_tx in change_transactions.into_iter() {
+            change_tx.persist(&self.storage).await?;
+        }
+        let mut transaction = Transaction::new_external_withdrawal(reference, status, payments)?;
+        transaction.persist(&self.storage).await?;
+        Ok(transaction)
+    }
+
+    /// Returns the transaction object by a given transaction id
+    pub async fn get_transaction(
+        &'a self,
+        transaction_id: &TransactionId,
+    ) -> Result<Transaction, Error> {
+        Ok(self
+            .storage
+            .get_transaction(transaction_id)
+            .await?
+            .try_into()?)
+    }
+
+    /// Returns all transactions from a given account. It can be optionally be
+    /// sorted by transaction type. The transactions are sorted from newest to
+    /// oldest.
+    pub async fn get_transactions(
+        &'a self,
+        account_id: &AccountId,
+        typ: Option<Type>,
+    ) -> Result<Vec<Transaction>, Error> {
+        let r = self
+            .storage
+            .get_transactions(account_id, typ)
+            .await?
+            .into_iter()
+            .map(|x| x.try_into())
+            .collect::<Result<Vec<Transaction>, _>>()?;
+
+        Ok(r)
+    }
+
+    /// Attemps to change the status of a given transaction id. On success the
+    /// new transaction object is returned, otherwise an error is returned.
     pub async fn change_status(
         &'a self,
         transaction_id: &TransactionId,
         new_status: Status,
     ) -> Result<Transaction, Error> {
-        let mut tx = self.storage.get_transaction(transaction_id).await?;
+        let mut tx: Transaction = self
+            .storage
+            .get_transaction(transaction_id)
+            .await?
+            .try_into()?;
         tx.change_status(new_status)?;
         tx.persist(&self.storage).await?;
         Ok(tx)

+ 6 - 4
utxo/src/lib.rs

@@ -6,9 +6,9 @@ mod id;
 mod ledger;
 mod payment;
 #[cfg(any(feature = "sqlite", test))]
-mod sqlite;
+pub mod sqlite;
 mod status;
-mod storage;
+pub mod storage;
 #[cfg(test)]
 mod tests;
 mod transaction;
@@ -16,12 +16,14 @@ mod transaction;
 pub use self::{
     amount::Amount,
     asset::Asset,
-    asset_manager::AssetManager,
+    asset_manager::{AssetDefinition, AssetManager},
     error::Error,
     id::*,
     ledger::Ledger,
     payment::{Payment, PaymentId},
     status::Status,
-    storage::{Batch, Storage},
     transaction::Transaction,
 };
+
+#[cfg(any(feature = "sqlite", test))]
+pub use self::sqlite::{Batch, Sqlite};

+ 18 - 2
utxo/src/payment.rs

@@ -1,12 +1,22 @@
 use crate::{AccountId, Amount, Status, TransactionId};
-use serde::Serialize;
+use serde::{Serialize, Serializer};
 
-#[derive(Copy, Clone, Debug, Eq, PartialEq, Serialize)]
+#[derive(Clone, Debug, Eq, Ord, PartialOrd, PartialEq)]
 pub struct PaymentId {
     pub transaction: TransactionId,
     pub position: usize,
 }
 
+impl Serialize for PaymentId {
+    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
+    where
+        S: Serializer,
+    {
+        let serialized = self.to_string();
+        serializer.serialize_str(&serialized)
+    }
+}
+
 impl ToString for PaymentId {
     fn to_string(&self) -> String {
         format!("{}:{}", self.transaction, self.position)
@@ -23,3 +33,9 @@ pub struct Payment {
     #[serde(skip)]
     pub spent_by: Option<TransactionId>,
 }
+
+impl Payment {
+    pub fn is_spendable(&self) -> bool {
+        self.spent_by.is_none() && self.status == Status::Settled
+    }
+}

+ 55 - 30
utxo/src/sqlite/batch.rs

@@ -1,6 +1,6 @@
 use crate::{
     storage::{self, Error},
-    Payment, PaymentId, Status, Transaction, TransactionId,
+    AccountId, Payment, PaymentId, Status, Transaction, TransactionId,
 };
 use sqlx::{Row, Sqlite, Transaction as SqlxTransaction};
 use std::marker::PhantomData;
@@ -21,11 +21,25 @@ impl<'a> Batch<'a> {
 
 #[async_trait::async_trait]
 impl<'a> storage::Batch<'a> for Batch<'a> {
+    async fn rollback(self) -> Result<(), Error> {
+        self.inner
+            .rollback()
+            .await
+            .map_err(|e| Error::Storage(e.to_string()))
+    }
+
+    async fn commit(self) -> Result<(), Error> {
+        self.inner
+            .commit()
+            .await
+            .map_err(|e| Error::Storage(e.to_string()))
+    }
+
     async fn spend_payment(
         &mut self,
-        payment_id: PaymentId,
+        payment_id: &PaymentId,
         status: Status,
-        transaction_id: TransactionId,
+        transaction_id: &TransactionId,
     ) -> Result<(), Error> {
         let result = sqlx::query(
             r#"
@@ -85,23 +99,8 @@ impl<'a> storage::Batch<'a> for Batch<'a> {
         }
     }
 
-    async fn rollback(self) -> Result<(), Error> {
-        self.inner
-            .rollback()
-            .await
-            .map_err(|e| Error::Storage(e.to_string()))
-    }
-
-    async fn commit(self) -> Result<(), Error> {
-        self.inner
-            .commit()
-            .await
-            .map_err(|e| Error::Storage(e.to_string()))
-    }
-
-    async fn store_new_payments(&mut self, payments: &[Payment]) -> Result<(), Error> {
-        for payment in payments.iter() {
-            sqlx::query(
+    async fn store_new_payment(&mut self, payment: &Payment) -> Result<(), Error> {
+        sqlx::query(
                 r#"
                 INSERT INTO payments("transaction_id", "position_id", "to", "cents", "asset_id", "status")
                 VALUES (?, ?, ?, ?, ?, ?)
@@ -118,36 +117,38 @@ impl<'a> storage::Batch<'a> for Batch<'a> {
             .execute(&mut *self.inner)
             .await
             .map_err(|e| Error::Storage(e.to_string()))?;
-        }
         Ok(())
     }
 
     async fn store_transaction(&mut self, transaction: &Transaction) -> Result<(), Error> {
         sqlx::query(
             r#"
-                INSERT INTO "transactions"("transaction_id", "status", "reference")
-                VALUES(?, ?, ?)
+                INSERT INTO "transactions"("transaction_id", "status", "type", "reference", "created_at", "updated_at")
+                VALUES(?, ?, ?, ?, ?, ?)
                 ON CONFLICT("transaction_id")
-                    DO UPDATE SET "status" = excluded."status", "reference" = excluded."reference"
+                    DO UPDATE SET "status" = excluded."status", "updated_at" = excluded."updated_at"
             "#,
         )
-        .bind(transaction.id.to_string())
-        .bind::<u32>((&transaction.status).into())
-        .bind(transaction.reference.to_string())
+        .bind(transaction.id().to_string())
+        .bind::<u32>(transaction.status().into())
+        .bind::<u32>(transaction.typ().into())
+        .bind(transaction.reference())
+        .bind(transaction.created_at())
+        .bind(transaction.updated_at())
         .execute(&mut *self.inner)
         .await
         .map_err(|e| Error::Storage(e.to_string()))?;
 
-        for payment in transaction.spend.iter() {
+        for payment in transaction.spends().iter() {
             sqlx::query(
                 r#"
-            INSERT INTO "transaction_payments"("transaction_id", "payment_transaction_id", "payment_position_id")
+            INSERT INTO "transaction_input_payments"("transaction_id", "payment_transaction_id", "payment_position_id")
             VALUES(?, ?, ?)
             ON CONFLICT("transaction_id", "payment_transaction_id", "payment_position_id")
                 DO NOTHING
             "#,
             )
-            .bind(transaction.id.to_string())
+            .bind(transaction.id().to_string())
             .bind(payment.id.transaction.to_string())
             .bind(payment.id.position.to_string())
             .execute(&mut *self.inner)
@@ -157,4 +158,28 @@ impl<'a> storage::Batch<'a> for Batch<'a> {
 
         Ok(())
     }
+
+    async fn relate_account_to_transaction(
+        &mut self,
+        transaction: &Transaction,
+        account: &AccountId,
+    ) -> Result<(), Error> {
+        sqlx::query(
+            r#"
+            INSERT INTO "transaction_accounts"("transaction_id", "account_id", "type", "created_at", "updated_at")
+            VALUES(?, ?, ?, ?, ?)
+            ON CONFLICT("transaction_id", "account_id")
+                DO NOTHING
+            "#,
+        )
+        .bind(transaction.id().to_string())
+        .bind(account.to_string())
+        .bind::<u32>(transaction.typ().into())
+        .bind(transaction.created_at())
+        .bind(transaction.updated_at())
+        .execute(&mut *self.inner)
+        .await
+        .map_err(|e| Error::Storage(e.to_string()))?;
+        Ok(())
+    }
 }

+ 99 - 13
utxo/src/sqlite/mod.rs

@@ -1,7 +1,11 @@
 use crate::{
-    amount::AmountCents, asset::AssetId, storage::Error, AccountId, Amount, Asset, AssetManager,
-    Payment, PaymentId, Status, Storage, Transaction, TransactionId,
+    amount::AmountCents,
+    asset::AssetId,
+    storage::{Error, Storage},
+    transaction::{from_db, Type},
+    AccountId, Amount, Asset, AssetManager, Payment, PaymentId, Status, TransactionId,
 };
+use chrono::NaiveDateTime;
 use futures::TryStreamExt;
 use sqlx::{sqlite::SqliteRow, Executor, Row};
 use std::{collections::HashMap, marker::PhantomData};
@@ -9,6 +13,7 @@ use std::{collections::HashMap, marker::PhantomData};
 mod batch;
 
 pub use batch::Batch;
+pub use sqlx::sqlite::SqlitePoolOptions;
 
 pub struct Sqlite<'a> {
     db: sqlx::SqlitePool,
@@ -40,16 +45,26 @@ impl<'a> Sqlite<'a> {
                 "updated_at" DATETIME DEFAULT CURRENT_TIMESTAMP,
                 PRIMARY KEY ("transaction_id", "position_id")
             );
-            CREATE INDEX IF NOT EXISTS payments_to ON payments ("to", "asset_id", "status", "spent_by");
+            CREATE INDEX IF NOT EXISTS payments_to ON payments ("to", "asset_id", "spent_by", "status");
             CREATE TABLE IF NOT EXISTS "transactions" (
                 "transaction_id" VARCHAR(66) NOT NULL,
                 "status" INTEGER NOT NULL,
+                "type" INTEGER NOT NULL,
                 "reference" TEXT NOT NULL,
                 "created_at" DATETIME DEFAULT CURRENT_TIMESTAMP,
                 "updated_at" DATETIME DEFAULT CURRENT_TIMESTAMP,
                 PRIMARY KEY ("transaction_id")
             );
-            CREATE TABLE IF NOT EXISTS "transaction_payments" (
+            CREATE TABLE IF NOT EXISTS "transaction_accounts" (
+                "transaction_id" VARCHAR(66) NOT NULL,
+                "account_id" VARCHAR(71) NOT NULL,
+                "type" INTEGER NOT NULL,
+                "created_at" DATETIME DEFAULT CURRENT_TIMESTAMP,
+                "updated_at" DATETIME DEFAULT CURRENT_TIMESTAMP,
+                PRIMARY KEY("account_id", "transaction_id")
+            );
+            CREATE INDEX IF NOT EXISTS "type" ON "transaction_accounts" ("account_id", "type", "created_at");
+            CREATE TABLE IF NOT EXISTS "transaction_input_payments" (
                 "transaction_id" VARCHAR(66) NOT NULL,
                 "payment_transaction_id" VARCHAR(66) NOT NULL,
                 "payment_position_id"  INTEGER NOT NULL,
@@ -180,7 +195,7 @@ impl<'a> Storage<'a, Batch<'a>> for Sqlite<'a> {
             FROM
                 "payments"
             WHERE
-                "to" = ? AND status = ? AND "spent_by" IS NULL
+                "to" = ? AND "spent_by" IS NULL AND status = ? 
             "#,
         )
         .bind(account.to_string())
@@ -278,7 +293,10 @@ impl<'a> Storage<'a, Batch<'a>> for Sqlite<'a> {
         }
     }
 
-    async fn get_transaction(&self, transaction_id: &TransactionId) -> Result<Transaction, Error> {
+    async fn get_transaction(
+        &self,
+        transaction_id: &TransactionId,
+    ) -> Result<from_db::Transaction, Error> {
         let mut conn = self
             .db
             .acquire()
@@ -289,7 +307,10 @@ impl<'a> Storage<'a, Batch<'a>> for Sqlite<'a> {
             r#"
             SELECT
                 "t"."status",
-                "t"."reference"
+                "t"."type",
+                "t"."reference",
+                "t"."created_at",
+                "t"."updated_at"
             FROM
                 "transactions" "t"
             WHERE
@@ -313,11 +334,11 @@ impl<'a> Storage<'a, Batch<'a>> for Sqlite<'a> {
                 "p"."status",
                 "p"."spent_by"
             FROM
+                "transaction_input_payments" "tp"
+            INNER JOIN
                 "payments" "p"
-            INNER JOIN 
-                "transaction_payments" "tp" 
                 ON (
-                    "tp"."payment_transaction_id" = "p"."transaction_id" 
+                    "tp"."payment_transaction_id" = "p"."transaction_id"
                     AND "tp"."payment_position_id" = "p"."position_id"
                 )
             WHERE
@@ -373,17 +394,82 @@ impl<'a> Storage<'a, Batch<'a>> for Sqlite<'a> {
             .map_err(|_| Error::Storage("Invalid status".to_string()))?
             .try_into()
             .map_err(|_| Error::Storage("Invalid status".to_string()))?;
+        let typ = transaction_row
+            .try_get::<u32, usize>(1)
+            .map_err(|_| Error::Storage("Invalid type".to_string()))?
+            .try_into()
+            .map_err(|_| Error::Storage("Invalid type".to_string()))?;
+
         let reference = transaction_row
-            .try_get::<String, usize>(1)
+            .try_get::<String, usize>(2)
             .map_err(|_| Error::Storage("Invalid reference".to_string()))?;
 
-        Ok(Transaction {
+        let created_at = transaction_row
+            .try_get::<NaiveDateTime, usize>(3)
+            .map_err(|e| Error::InvalidDate(e.to_string()))?
+            .and_utc();
+
+        let updated_at = transaction_row
+            .try_get::<NaiveDateTime, usize>(4)
+            .map_err(|e| Error::InvalidDate(e.to_string()))?
+            .and_utc();
+
+        Ok(from_db::Transaction {
             id: transaction_id.clone(),
-            is_external_deposit: spend.is_empty(),
             spend,
             create,
+            typ,
             status,
             reference,
+            created_at,
+            updated_at,
         })
     }
+
+    async fn get_transactions(
+        &self,
+        account: &AccountId,
+        typ: Option<Type>,
+    ) -> Result<Vec<from_db::Transaction>, Error> {
+        let mut conn = self
+            .db
+            .acquire()
+            .await
+            .map_err(|e| Error::Storage(e.to_string()))?;
+
+        let sql = sqlx::query(if typ.is_some() {
+            r#"SELECT "transaction_id" FROM "transaction_accounts" WHERE "account_id" = ? AND "type" = ?" ORDER BY "created_at" DESC"#
+        } else {
+            r#"SELECT "transaction_id" FROM "transaction_accounts" WHERE "account_id" = ? ORDER BY "created_at" DESC"#
+        }).bind(account.to_string());
+
+        let ids = if let Some(typ) = typ {
+            sql.bind::<u32>(typ.into())
+        } else {
+            sql
+        }
+        .fetch_all(&mut *conn)
+        .await
+        .map_err(|e| Error::Storage(e.to_string()))?
+        .iter()
+        .map(|row| {
+            let id: Result<TransactionId, _> = row
+                .try_get::<String, usize>(0)
+                .map_err(|_| Error::Storage("Invalid transaction_id".to_string()))?
+                .as_str()
+                .try_into();
+
+            id.map_err(|_| Error::Storage("Invalid transaction_id length".to_string()))
+        })
+        .collect::<Result<Vec<_>, Error>>()?;
+
+        drop(conn);
+
+        let mut transactions = vec![];
+        for id in ids.into_iter() {
+            transactions.push(self.get_transaction(&id).await?);
+        }
+
+        Ok(transactions)
+    }
 }

+ 4 - 8
utxo/src/status.rs

@@ -1,7 +1,9 @@
+use serde::{Deserialize, Serialize};
 use strum_macros::Display;
 
 /// Transaction status
-#[derive(Clone, Eq, PartialEq, Debug, Display)]
+#[derive(Clone, Eq, PartialEq, Debug, Display, Serialize, Deserialize)]
+#[serde(rename_all = "snake_case")]
 pub enum Status {
     /// Pending status
     ///
@@ -59,13 +61,7 @@ impl TryFrom<u32> for Status {
 
 impl From<Status> for u32 {
     fn from(value: Status) -> Self {
-        match value {
-            Status::Pending => 0,
-            Status::Processing => 10,
-            Status::Cancelled => 20,
-            Status::Settled => 30,
-            Status::Failed => 40,
-        }
+        (&value).into()
     }
 }
 

+ 56 - 13
utxo/src/storage.rs

@@ -1,6 +1,8 @@
 use crate::{
-    amount::AmountCents, asset::AssetId, AccountId, Amount, Payment, PaymentId, Status,
-    Transaction, TransactionId,
+    amount::AmountCents,
+    asset::AssetId,
+    transaction::{from_db, Type},
+    AccountId, Amount, Payment, PaymentId, Status, Transaction, TransactionId,
 };
 
 #[derive(thiserror::Error, Debug)]
@@ -8,6 +10,9 @@ pub enum Error {
     #[error("Storage error: {0}")]
     Storage(String),
 
+    #[error("Invalid date format: {0}")]
+    InvalidDate(String),
+
     #[error("Spend payment: {0}")]
     SpendPayment(String),
 
@@ -24,25 +29,31 @@ pub enum Error {
 
 #[async_trait::async_trait]
 pub trait Batch<'a> {
+    async fn rollback(self) -> Result<(), Error>;
+
+    async fn commit(self) -> Result<(), Error>;
+
     async fn spend_payment(
         &mut self,
-        payment_id: PaymentId,
+        payment_id: &PaymentId,
         status: Status,
-        transaction_id: TransactionId,
+        transaction_id: &TransactionId,
     ) -> Result<(), Error>;
 
-    async fn rollback(self) -> Result<(), Error>;
-
     async fn get_payment_status(
         &mut self,
         transaction_id: &TransactionId,
     ) -> Result<Option<Status>, Error>;
 
-    async fn commit(self) -> Result<(), Error>;
-
-    async fn store_new_payments(&mut self, outputs: &[Payment]) -> Result<(), Error>;
+    async fn store_new_payment(&mut self, payment: &Payment) -> Result<(), Error>;
 
     async fn store_transaction(&mut self, transaction: &Transaction) -> Result<(), Error>;
+
+    async fn relate_account_to_transaction(
+        &mut self,
+        transaction: &Transaction,
+        account: &AccountId,
+    ) -> Result<(), Error>;
 }
 
 #[async_trait::async_trait]
@@ -50,21 +61,40 @@ pub trait Storage<'a, B>
 where
     B: Batch<'a>,
 {
+    /// Begins a transaction
+    ///
+    /// A transaction prepares the storage for a batch of operations, where all
+    /// the operations are persisted or none.
+    ///
+    /// The transaction is returned as a Batch object, which is used to perform
+    /// the changes in the transactions and payments.
+    ///
+    /// The batch has also a rollback
     async fn begin(&'a self) -> Result<B, Error>;
 
+    /// Returns a payment object by ID.
     async fn get_payment(&self, id: PaymentId) -> Result<Payment, Error>;
 
+    /// Similar to get_payment() but errors if the requested payment is not spendable.
     async fn get_unspent_payment(&self, id: PaymentId) -> Result<Payment, Error> {
         let payment = self.get_payment(id).await?;
-        if payment.spent_by.is_some() {
-            Err(Error::NotFound)
-        } else {
+        if payment.is_spendable() {
             Ok(payment)
+        } else {
+            Err(Error::NotFound)
         }
     }
 
+    /// Returns the balances for a given account
     async fn get_balance(&self, account: &AccountId) -> Result<Vec<Amount>, Error>;
 
+    /// Returns a list of unspent payments for a given account and asset.
+    ///
+    /// The payments should be returned sorted by ascending amount, this bit is
+    /// important to make sure that all negative payments are always taken into
+    /// account. It will also improve the database by using many small payments
+    /// instead of a few large ones, which will make the database faster leaving
+    /// fewer unspent payments when checking for balances.
     async fn get_unspent_payments(
         &self,
         account: &AccountId,
@@ -72,5 +102,18 @@ where
         target_amount: AmountCents,
     ) -> Result<Vec<Payment>, Error>;
 
-    async fn get_transaction(&self, transaction_id: &TransactionId) -> Result<Transaction, Error>;
+    /// Returns a transaction object by id
+    async fn get_transaction(
+        &self,
+        transaction_id: &TransactionId,
+    ) -> Result<from_db::Transaction, Error>;
+
+    /// Returns a list of a transactions for a given account (and optionally
+    /// filter by types). The list of transactions is expected to be sorted by
+    /// date desc (newest transactions first)
+    async fn get_transactions(
+        &self,
+        account: &AccountId,
+        typ: Option<Type>,
+    ) -> Result<Vec<from_db::Transaction>, Error>;
 }

+ 156 - 58
utxo/src/tests/deposit.rs

@@ -1,19 +1,38 @@
-use crate::{
-    sqlite::{Batch, Sqlite},
-    tests::get_instance,
-    AccountId, Amount, Ledger, Status, TransactionId,
-};
-
-pub async fn deposit(
-    ledger: &Ledger<'static, Batch<'static>, Sqlite<'static>>,
-    account_id: &AccountId,
-    amount: Amount,
-) -> TransactionId {
-    ledger
-        .deposit(account_id, amount, Status::Settled, "Test".to_owned())
+use super::{deposit, get_instance};
+use crate::{AccountId, Status};
+
+#[tokio::test]
+async fn pending_deposit_and_failure() {
+    let source = "account1".parse::<AccountId>().expect("account");
+    let (assets, ledger) = get_instance().await;
+    let id = ledger
+        .deposit(
+            &source,
+            assets.amount_by_and_cents(2, 3000).expect("amount"),
+            Status::Processing,
+            "Test".to_owned(),
+        )
         .await
         .expect("valid tx")
-        .id
+        .id()
+        .clone();
+
+    assert!(ledger
+        .get_balance(&source)
+        .await
+        .expect("balance")
+        .is_empty());
+
+    ledger
+        .change_status(&id, Status::Failed)
+        .await
+        .expect("valid tx");
+
+    assert!(ledger
+        .get_balance(&source)
+        .await
+        .expect("balance")
+        .is_empty());
 }
 
 #[tokio::test]
@@ -23,11 +42,21 @@ async fn deposit_and_transfer() {
     let fee = "fee".parse::<AccountId>().expect("account");
     let (assets, ledger) = get_instance().await;
 
-    deposit(&ledger, &source, assets.amount(2, 1000).expect("amount")).await;
-    deposit(&ledger, &source, assets.amount(2, 2000).expect("amount")).await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 1000).expect("amount"),
+    )
+    .await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 2000).expect("amount"),
+    )
+    .await;
 
     assert_eq!(
-        vec![assets.amount(2, 3000).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 3000).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
 
@@ -35,25 +64,34 @@ async fn deposit_and_transfer() {
         .new_transaction(
             "Exchange one".to_owned(),
             Status::Settled,
-            vec![(source.clone(), assets.amount(2, 1300).expect("amount"))],
+            vec![(
+                source.clone(),
+                assets.amount_by_and_cents(2, 1300).expect("amount"),
+            )],
             vec![
-                (dest.clone(), assets.amount(2, 1250).expect("amount")),
-                (fee.clone(), assets.amount(2, 50).expect("amount")),
+                (
+                    dest.clone(),
+                    assets.amount_by_and_cents(2, 1250).expect("amount"),
+                ),
+                (
+                    fee.clone(),
+                    assets.amount_by_and_cents(2, 50).expect("amount"),
+                ),
             ],
         )
         .await
         .expect("valid tx");
 
     assert_eq!(
-        vec![assets.amount(2, 1700).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
     assert_eq!(
-        vec![assets.amount(2, 1250).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 1250).expect("amount")],
         ledger.get_balance(&dest).await.expect("balance")
     );
     assert_eq!(
-        vec![assets.amount(2, 50).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
         ledger.get_balance(&fee).await.expect("balance")
     );
 }
@@ -65,11 +103,21 @@ async fn balance_decreases_while_pending_spending_and_confirm() {
     let fee = "fee".parse::<AccountId>().expect("account");
     let (assets, ledger) = get_instance().await;
 
-    deposit(&ledger, &source, assets.amount(2, 1000).expect("amount")).await;
-    deposit(&ledger, &source, assets.amount(2, 2000).expect("amount")).await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 1000).expect("amount"),
+    )
+    .await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 2000).expect("amount"),
+    )
+    .await;
 
     assert_eq!(
-        vec![assets.amount(2, 3000).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 3000).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
 
@@ -77,18 +125,28 @@ async fn balance_decreases_while_pending_spending_and_confirm() {
         .new_transaction(
             "Exchange one".to_owned(),
             Status::Pending,
-            vec![(source.clone(), assets.amount(2, 1300).expect("amount"))],
+            vec![(
+                source.clone(),
+                assets.amount_by_and_cents(2, 1300).expect("amount"),
+            )],
             vec![
-                (dest.clone(), assets.amount(2, 1250).expect("amount")),
-                (fee.clone(), assets.amount(2, 50).expect("amount")),
+                (
+                    dest.clone(),
+                    assets.amount_by_and_cents(2, 1250).expect("amount"),
+                ),
+                (
+                    fee.clone(),
+                    assets.amount_by_and_cents(2, 50).expect("amount"),
+                ),
             ],
         )
         .await
         .expect("valid tx")
-        .id;
+        .id()
+        .clone();
 
     assert_eq!(
-        vec![assets.amount(2, 1700).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
     assert!(ledger.get_balance(&dest).await.expect("balance").is_empty());
@@ -100,15 +158,15 @@ async fn balance_decreases_while_pending_spending_and_confirm() {
         .expect("valid tx");
 
     assert_eq!(
-        vec![assets.amount(2, 1700).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
     assert_eq!(
-        vec![assets.amount(2, 1250).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 1250).expect("amount")],
         ledger.get_balance(&dest).await.expect("balance")
     );
     assert_eq!(
-        vec![assets.amount(2, 50).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
         ledger.get_balance(&fee).await.expect("balance")
     );
 }
@@ -120,11 +178,21 @@ async fn balance_decreases_while_pending_spending_and_cancel() {
     let fee = "fee".parse::<AccountId>().expect("account");
     let (assets, ledger) = get_instance().await;
 
-    deposit(&ledger, &source, assets.amount(2, 1000).expect("amount")).await;
-    deposit(&ledger, &source, assets.amount(2, 2000).expect("amount")).await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 1000).expect("amount"),
+    )
+    .await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 2000).expect("amount"),
+    )
+    .await;
 
     assert_eq!(
-        vec![assets.amount(2, 3000).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 3000).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
 
@@ -132,18 +200,28 @@ async fn balance_decreases_while_pending_spending_and_cancel() {
         .new_transaction(
             "Exchange one".to_owned(),
             Status::Pending,
-            vec![(source.clone(), assets.amount(2, 1300).expect("amount"))],
+            vec![(
+                source.clone(),
+                assets.amount_by_and_cents(2, 1300).expect("amount"),
+            )],
             vec![
-                (dest.clone(), assets.amount(2, 1250).expect("amount")),
-                (fee.clone(), assets.amount(2, 50).expect("amount")),
+                (
+                    dest.clone(),
+                    assets.amount_by_and_cents(2, 1250).expect("amount"),
+                ),
+                (
+                    fee.clone(),
+                    assets.amount_by_and_cents(2, 50).expect("amount"),
+                ),
             ],
         )
         .await
         .expect("valid tx")
-        .id;
+        .id()
+        .clone();
 
     assert_eq!(
-        vec![assets.amount(2, 1700).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
     assert!(ledger.get_balance(&dest).await.expect("balance").is_empty());
@@ -155,7 +233,7 @@ async fn balance_decreases_while_pending_spending_and_cancel() {
         .expect("valid tx");
 
     assert_eq!(
-        vec![assets.amount(2, 3000).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 3000).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
     assert!(ledger.get_balance(&dest).await.expect("balance").is_empty());
@@ -169,42 +247,62 @@ async fn balance_decreases_while_pending_spending_and_failed() {
     let fee = "fee".parse::<AccountId>().expect("account");
     let (assets, ledger) = get_instance().await;
 
-    deposit(&ledger, &source, assets.amount(2, 1000).expect("amount")).await;
-    deposit(&ledger, &source, assets.amount(2, 2000).expect("amount")).await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 1000).expect("amount"),
+    )
+    .await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 2000).expect("amount"),
+    )
+    .await;
 
     assert_eq!(
-        vec![assets.amount(2, 3000).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 3000).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
 
-    let id = ledger
+    let tx = ledger
         .new_transaction(
             "Exchange one".to_owned(),
             Status::Pending,
-            vec![(source.clone(), assets.amount(2, 1300).expect("amount"))],
+            vec![(
+                source.clone(),
+                assets.amount_by_and_cents(2, 1300).expect("amount"),
+            )],
             vec![
-                (dest.clone(), assets.amount(2, 1250).expect("amount")),
-                (fee.clone(), assets.amount(2, 50).expect("amount")),
+                (
+                    dest.clone(),
+                    assets.amount_by_and_cents(2, 1250).expect("amount"),
+                ),
+                (
+                    fee.clone(),
+                    assets.amount_by_and_cents(2, 50).expect("amount"),
+                ),
             ],
         )
         .await
         .expect("valid tx")
-        .id;
+        .id()
+        .clone();
 
     assert_eq!(
-        vec![assets.amount(2, 1700).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
     assert!(ledger.get_balance(&dest).await.expect("balance").is_empty());
     assert!(ledger.get_balance(&fee).await.expect("balance").is_empty());
 
     ledger
-        .change_status(&id, Status::Processing)
+        .change_status(&tx, Status::Processing)
         .await
         .expect("valid tx");
 
     assert_eq!(
-        vec![assets.amount(2, 1700).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
     assert!(ledger.get_balance(&dest).await.expect("balance").is_empty());
@@ -213,26 +311,26 @@ async fn balance_decreases_while_pending_spending_and_failed() {
     assert_eq!(
         "Transaction: Status transition from Processing to Cancelled is not allowed".to_owned(),
         ledger
-            .change_status(&id, Status::Cancelled)
+            .change_status(&tx, Status::Cancelled)
             .await
             .unwrap_err()
             .to_string()
     );
 
     assert_eq!(
-        vec![assets.amount(2, 1700).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
     assert!(ledger.get_balance(&dest).await.expect("balance").is_empty());
     assert!(ledger.get_balance(&fee).await.expect("balance").is_empty());
 
     ledger
-        .change_status(&id, Status::Failed)
+        .change_status(&tx, Status::Failed)
         .await
         .expect("valid");
 
     assert_eq!(
-        vec![assets.amount(2, 3000).expect("amount")],
+        vec![assets.amount_by_and_cents(2, 3000).expect("amount")],
         ledger.get_balance(&source).await.expect("balance")
     );
     assert!(ledger.get_balance(&dest).await.expect("balance").is_empty());

+ 30 - 2
utxo/src/tests/mod.rs

@@ -1,7 +1,7 @@
 use crate::{
     asset_manager::AssetDefinition,
     sqlite::{Batch, Sqlite},
-    AssetManager, Ledger,
+    AccountId, Amount, AssetManager, Error, Ledger, Status, TransactionId,
 };
 use sqlx::sqlite::SqlitePoolOptions;
 
@@ -25,7 +25,35 @@ pub async fn get_instance() -> (
 
     let db = Sqlite::new(pool, assets.clone());
     db.setup().await.expect("setup");
-    (assets, Ledger::new(db))
+    (assets.clone(), Ledger::new(db, assets))
+}
+
+pub async fn withdrawal(
+    ledger: &Ledger<'static, Batch<'static>, Sqlite<'static>>,
+    account_id: &AccountId,
+    status: Status,
+    amount: Amount,
+) -> Result<TransactionId, Error> {
+    Ok(ledger
+        .withdrawal(account_id, amount, status, "Test".to_owned())
+        .await?
+        .id()
+        .clone())
+}
+
+pub async fn deposit(
+    ledger: &Ledger<'static, Batch<'static>, Sqlite<'static>>,
+    account_id: &AccountId,
+    amount: Amount,
+) -> TransactionId {
+    ledger
+        .deposit(account_id, amount, Status::Settled, "Test".to_owned())
+        .await
+        .expect("valid tx")
+        .id()
+        .clone()
 }
 
 mod deposit;
+mod negative_deposit;
+mod withdrawal;

+ 155 - 0
utxo/src/tests/negative_deposit.rs

@@ -0,0 +1,155 @@
+use super::{deposit, get_instance};
+use crate::{AccountId, Status};
+
+#[tokio::test]
+async fn negative_deposit_prevent_spending() {
+    let source = "account1".parse::<AccountId>().expect("account");
+    let dest = "account2".parse::<AccountId>().expect("account");
+    let fee = "fee".parse::<AccountId>().expect("account");
+    let (assets, ledger) = get_instance().await;
+
+    // Deposit some money
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 5000).expect("amount"),
+    )
+    .await;
+    // Take money of source's account
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, -10000).expect("amount"),
+    )
+    .await;
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, -5000).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+
+    // Try to spend money
+    assert_eq!(
+        "Storage: Not enough unspent payments (missing 6000 cents)".to_owned(),
+        ledger
+            .new_transaction(
+                "Exchange one".to_owned(),
+                Status::Settled,
+                vec![(
+                    source.clone(),
+                    assets.amount_by_and_cents(2, 1000).expect("amount")
+                )],
+                vec![
+                    (
+                        dest.clone(),
+                        assets.amount_by_and_cents(2, 950).expect("amount")
+                    ),
+                    (
+                        fee.clone(),
+                        assets.amount_by_and_cents(2, 50).expect("amount")
+                    ),
+                ],
+            )
+            .await
+            .unwrap_err()
+            .to_string()
+    );
+}
+
+#[tokio::test]
+async fn negative_deposit_prevent_spending_payback() {
+    let source = "account1".parse::<AccountId>().expect("account");
+    let dest = "account2".parse::<AccountId>().expect("account");
+    let fee = "fee".parse::<AccountId>().expect("account");
+    let (assets, ledger) = get_instance().await;
+
+    // Deposit some money
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 5000).expect("amount"),
+    )
+    .await;
+    // Take money of source's account
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, -10000).expect("amount"),
+    )
+    .await;
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, -5000).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+
+    // Try to spend money
+    assert_eq!(
+        "Storage: Not enough unspent payments (missing 6000 cents)".to_owned(),
+        ledger
+            .new_transaction(
+                "Exchange one".to_owned(),
+                Status::Settled,
+                vec![(
+                    source.clone(),
+                    assets.amount_by_and_cents(2, 1000).expect("amount")
+                )],
+                vec![
+                    (
+                        dest.clone(),
+                        assets.amount_by_and_cents(2, 950).expect("amount")
+                    ),
+                    (
+                        fee.clone(),
+                        assets.amount_by_and_cents(2, 50).expect("amount")
+                    ),
+                ],
+            )
+            .await
+            .unwrap_err()
+            .to_string()
+    );
+
+    // Payback the debt
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 15000).expect("amount"),
+    )
+    .await;
+
+    ledger
+        .new_transaction(
+            "Exchange one".to_owned(),
+            Status::Settled,
+            vec![(
+                source.clone(),
+                assets.amount_by_and_cents(2, 1000).expect("amount"),
+            )],
+            vec![
+                (
+                    dest.clone(),
+                    assets.amount_by_and_cents(2, 950).expect("amount"),
+                ),
+                (
+                    fee.clone(),
+                    assets.amount_by_and_cents(2, 50).expect("amount"),
+                ),
+            ],
+        )
+        .await
+        .expect("valid tx");
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 950).expect("amount")],
+        ledger.get_balance(&dest).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
+        ledger.get_balance(&fee).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 9000).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+}

+ 336 - 0
utxo/src/tests/withdrawal.rs

@@ -0,0 +1,336 @@
+use super::{deposit, get_instance, withdrawal};
+use crate::{AccountId, Status};
+
+#[tokio::test]
+async fn deposit_and_transfer_and_withdrawal() {
+    let source = "account1".parse::<AccountId>().expect("account");
+    let dest = "account2".parse::<AccountId>().expect("account");
+    let fee = "fee".parse::<AccountId>().expect("account");
+    let (assets, ledger) = get_instance().await;
+
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 1000).expect("amount"),
+    )
+    .await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 2000).expect("amount"),
+    )
+    .await;
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 3000).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+
+    ledger
+        .new_transaction(
+            "Exchange one".to_owned(),
+            Status::Settled,
+            vec![(
+                source.clone(),
+                assets.amount_by_and_cents(2, 1300).expect("amount"),
+            )],
+            vec![
+                (
+                    dest.clone(),
+                    assets.amount_by_and_cents(2, 1250).expect("amount"),
+                ),
+                (
+                    fee.clone(),
+                    assets.amount_by_and_cents(2, 50).expect("amount"),
+                ),
+            ],
+        )
+        .await
+        .expect("valid tx");
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1250).expect("amount")],
+        ledger.get_balance(&dest).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
+        ledger.get_balance(&fee).await.expect("balance")
+    );
+
+    assert!(withdrawal(
+        &ledger,
+        &source,
+        Status::Settled,
+        assets.amount_by_and_cents(2, 1700).expect("amount")
+    )
+    .await
+    .is_ok());
+
+    assert!(ledger
+        .get_balance(&source)
+        .await
+        .expect("balance")
+        .is_empty());
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1250).expect("amount")],
+        ledger.get_balance(&dest).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
+        ledger.get_balance(&fee).await.expect("balance")
+    );
+}
+
+#[tokio::test]
+async fn fail_to_overwithdrawal() {
+    let source = "account1".parse::<AccountId>().expect("account");
+    let dest = "account2".parse::<AccountId>().expect("account");
+    let fee = "fee".parse::<AccountId>().expect("account");
+    let (assets, ledger) = get_instance().await;
+
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 1000).expect("amount"),
+    )
+    .await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 2000).expect("amount"),
+    )
+    .await;
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 3000).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+
+    ledger
+        .new_transaction(
+            "Exchange one".to_owned(),
+            Status::Settled,
+            vec![(
+                source.clone(),
+                assets.amount_by_and_cents(2, 1300).expect("amount"),
+            )],
+            vec![
+                (
+                    dest.clone(),
+                    assets.amount_by_and_cents(2, 1250).expect("amount"),
+                ),
+                (
+                    fee.clone(),
+                    assets.amount_by_and_cents(2, 50).expect("amount"),
+                ),
+            ],
+        )
+        .await
+        .expect("valid tx");
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1250).expect("amount")],
+        ledger.get_balance(&dest).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
+        ledger.get_balance(&fee).await.expect("balance")
+    );
+
+    assert!(withdrawal(
+        &ledger,
+        &source,
+        Status::Settled,
+        assets.amount_by_and_cents(2, 170000).expect("amount")
+    )
+    .await
+    .is_err());
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1250).expect("amount")],
+        ledger.get_balance(&dest).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
+        ledger.get_balance(&fee).await.expect("balance")
+    );
+}
+
+#[tokio::test]
+async fn cancelled_withdrawal() {
+    let source = "account1".parse::<AccountId>().expect("account");
+    let dest = "account2".parse::<AccountId>().expect("account");
+    let fee = "fee".parse::<AccountId>().expect("account");
+    let (assets, ledger) = get_instance().await;
+
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 1000).expect("amount"),
+    )
+    .await;
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, 2000).expect("amount"),
+    )
+    .await;
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 3000).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+
+    ledger
+        .new_transaction(
+            "Exchange one".to_owned(),
+            Status::Settled,
+            vec![(
+                source.clone(),
+                assets.amount_by_and_cents(2, 1300).expect("amount"),
+            )],
+            vec![
+                (
+                    dest.clone(),
+                    assets.amount_by_and_cents(2, 1250).expect("amount"),
+                ),
+                (
+                    fee.clone(),
+                    assets.amount_by_and_cents(2, 50).expect("amount"),
+                ),
+            ],
+        )
+        .await
+        .expect("valid tx");
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1250).expect("amount")],
+        ledger.get_balance(&dest).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
+        ledger.get_balance(&fee).await.expect("balance")
+    );
+
+    let tx_id = withdrawal(
+        &ledger,
+        &source,
+        Status::Pending,
+        assets.amount_by_and_cents(2, 1700).expect("amount"),
+    )
+    .await
+    .expect("valid tx");
+
+    assert!(ledger
+        .get_balance(&source)
+        .await
+        .expect("balance")
+        .is_empty());
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1250).expect("amount")],
+        ledger.get_balance(&dest).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
+        ledger.get_balance(&fee).await.expect("balance")
+    );
+
+    ledger
+        .change_status(&tx_id, Status::Cancelled)
+        .await
+        .expect("valid tx");
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1700).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 1250).expect("amount")],
+        ledger.get_balance(&dest).await.expect("balance")
+    );
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, 50).expect("amount")],
+        ledger.get_balance(&fee).await.expect("balance")
+    );
+}
+
+#[tokio::test]
+async fn negative_withdrawal() {
+    let source = "account1".parse::<AccountId>().expect("account");
+    let (assets, ledger) = get_instance().await;
+
+    deposit(
+        &ledger,
+        &source,
+        assets.amount_by_and_cents(2, -1000).expect("amount"),
+    )
+    .await;
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, -1000).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, -1000).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+
+    assert!(withdrawal(
+        &ledger,
+        &source,
+        Status::Settled,
+        assets.amount_by_and_cents(2, -1000).expect("amount"),
+    )
+    .await
+    .is_err());
+
+    assert!(withdrawal(
+        &ledger,
+        &source,
+        Status::Settled,
+        assets.amount_by_and_cents(2, -1).expect("amount"),
+    )
+    .await
+    .is_err());
+
+    assert!(withdrawal(
+        &ledger,
+        &source,
+        Status::Settled,
+        assets.amount_by_and_cents(2, 0).expect("amount"),
+    )
+    .await
+    .is_err());
+
+    assert!(withdrawal(
+        &ledger,
+        &source,
+        Status::Settled,
+        assets.amount_by_and_cents(2, 10).expect("amount"),
+    )
+    .await
+    .is_err());
+
+    assert_eq!(
+        vec![assets.amount_by_and_cents(2, -1000).expect("amount")],
+        ledger.get_balance(&source).await.expect("balance")
+    );
+}

+ 6 - 5
utxo/src/transaction/error.rs

@@ -1,4 +1,4 @@
-use crate::{amount::AmountCents, storage, Asset, Status};
+use crate::{storage, Amount, Asset, Status, TransactionId};
 
 #[derive(thiserror::Error, Debug)]
 pub enum Error {
@@ -8,10 +8,8 @@ pub enum Error {
     #[error("Payment {0} is in the incorrect status {1}")]
     InvalidPaymentStatus(usize, Status),
 
-    #[error(
-        "Payment {0} is not a valid amount. Spending {1} and issuing {2}. Both amounts must match"
-    )]
-    InvalidAmount(Asset, AmountCents, AmountCents),
+    #[error("Spending {0:?} and issuing {1:?}. Both amounts must match")]
+    InvalidAmount(Amount, Amount),
 
     #[error("Missing input payment for asset {0}")]
     MissingSpendingAsset(Asset),
@@ -34,6 +32,9 @@ pub enum Error {
     #[error("Internal error at serializing: {0}")]
     Internal(#[from] Box<bincode::ErrorKind>),
 
+    #[error("Invalid calculated id {0} (expected {1})")]
+    InvalidTransactionId(TransactionId, TransactionId),
+
     #[error("Overflow")]
     Overflow,
 }

+ 14 - 0
utxo/src/transaction/from_db.rs

@@ -0,0 +1,14 @@
+use super::Type;
+use crate::{Payment, Status, TransactionId};
+use chrono::{DateTime, Utc};
+
+pub struct Transaction {
+    pub id: TransactionId,
+    pub spend: Vec<Payment>,
+    pub create: Vec<Payment>,
+    pub reference: String,
+    pub typ: Type,
+    pub status: Status,
+    pub created_at: DateTime<Utc>,
+    pub updated_at: DateTime<Utc>,
+}

+ 415 - 0
utxo/src/transaction/inner.rs

@@ -0,0 +1,415 @@
+use crate::{
+    amount::AmountCents,
+    storage::{Batch, Storage},
+    transaction::*,
+    AccountId, Amount, Asset, Payment, PaymentId, Status, TransactionId,
+};
+use chrono::{serde::ts_milliseconds, DateTime, Utc};
+use serde::Serialize;
+use sha2::{Digest, Sha256};
+use std::collections::HashMap;
+
+/// Transactions
+///
+/// Transactions are the core components of the ledger. The transactions are a
+/// list of unspent payments that are about to be spend, to create a new set of
+/// Payments, that can be spend in the future. This model is heavily inspired in
+/// Bitcoin's UTXO model. The terms in this context are payments, spend and
+/// create instead of unspent transactions, input and output.
+///
+/// This simple architecture allows to track accounts pretty efficiently,
+/// because all that matters are unspent payments owned by a given account.
+/// Every spent payment is stored for historical reasons but it is not relevant
+/// for any calculations regarding available funds.
+///
+/// The transaction has a few rules, for instance the sum of spend Payments
+/// should be the same as create Payments, for each easy. There is no 'fee'
+/// concept, so any mismatch in any direction will error the constructor.
+///
+/// Transactions are immutable after they are finalized, and the payments can
+/// only be re-usable if the transaction failed or was cancelled. Once the
+/// transaction settles the spent payments are forever spent. Any rollback
+/// should be a new transaction, initiated by a higher layer.
+///
+/// The spent payments are unavailable until the transaction is finalized,
+/// either as settled, cancelled or failed. A higher layer should split any
+/// available payment to be spend into a new transaction, and then finalize the
+/// transaction, and reserve only the exact amount to be spent, otherwise
+/// unrelated funds will be held unspentable until the transaction is finalized.
+#[derive(Debug, Clone, Serialize)]
+pub struct Transaction {
+    id: TransactionId,
+    #[serde(skip_serializing_if = "Vec::is_empty")]
+    spends: Vec<Payment>,
+    #[allow(dead_code)]
+    reference: String,
+    #[serde(rename = "type")]
+    typ: Type,
+    #[serde(skip_serializing_if = "Vec::is_empty")]
+    creates: Vec<Payment>,
+    status: Status,
+    #[serde(with = "ts_milliseconds")]
+    created_at: DateTime<Utc>,
+    #[serde(with = "ts_milliseconds")]
+    updated_at: DateTime<Utc>,
+}
+
+impl Transaction {
+    pub fn new_external_withdrawal(
+        reference: String,
+        status: Status,
+        spend: Vec<Payment>,
+    ) -> Result<Transaction, Error> {
+        let created_at = Utc::now();
+        let id = Self::calculate_hash(
+            &reference,
+            spend.iter().map(|t| &t.id).collect::<Vec<&PaymentId>>(),
+            vec![],
+            created_at,
+        )?;
+        let spend = spend
+            .into_iter()
+            .map(|mut input| {
+                input.spent_by = Some(id.clone());
+                input
+            })
+            .collect();
+        Ok(Self {
+            id,
+            spends: spend,
+            creates: vec![],
+            typ: Type::Withdrawal,
+            reference,
+            status,
+            created_at,
+            updated_at: Utc::now(),
+        })
+    }
+
+    pub fn new_external_deposit(
+        reference: String,
+        status: Status,
+        pay_to: Vec<(AccountId, Amount)>,
+    ) -> Result<Transaction, Error> {
+        let created_at = Utc::now();
+        let id = Self::calculate_hash(
+            &reference,
+            vec![],
+            pay_to
+                .iter()
+                .map(|t| (&t.0, &t.1))
+                .collect::<Vec<(&AccountId, &Amount)>>(),
+            created_at,
+        )?;
+        let create = pay_to
+            .into_iter()
+            .enumerate()
+            .map(|(position, (to, amount))| Payment {
+                id: crate::PaymentId {
+                    transaction: id.clone(),
+                    position,
+                },
+                to,
+                amount,
+                spent_by: None,
+                status: status.clone(),
+            })
+            .collect();
+
+        Ok(Self {
+            id,
+            spends: vec![],
+            creates: create,
+            reference,
+            typ: Type::Deposit,
+            status,
+            created_at,
+            updated_at: Utc::now(),
+        })
+    }
+
+    pub async fn new(
+        reference: String,
+        status: Status,
+        typ: Type,
+        spend: Vec<Payment>,
+        pay_to: Vec<(AccountId, Amount)>,
+    ) -> Result<Transaction, Error> {
+        let created_at = Utc::now();
+        let id = Self::calculate_hash(
+            &reference,
+            spend.iter().map(|t| &t.id).collect::<Vec<&PaymentId>>(),
+            pay_to
+                .iter()
+                .map(|t| (&t.0, &t.1))
+                .collect::<Vec<(&AccountId, &Amount)>>(),
+            created_at,
+        )?;
+
+        for (i, input) in spend.iter().enumerate() {
+            if input.spent_by.as_ref() != Some(&id) && !input.is_spendable() {
+                return Err(Error::InvalidPaymentStatus(i, input.status.clone()));
+            }
+        }
+        let spend = spend
+            .into_iter()
+            .map(|mut input| {
+                input.spent_by = Some(id.clone());
+                input
+            })
+            .collect();
+
+        let create = pay_to
+            .into_iter()
+            .enumerate()
+            .map(|(position, (to, amount))| Payment {
+                id: crate::PaymentId {
+                    transaction: id.clone(),
+                    position,
+                },
+                to,
+                amount,
+                spent_by: None,
+                status: status.clone(),
+            })
+            .collect();
+
+        Ok(Self {
+            id,
+            reference,
+            spends: spend,
+            typ,
+            creates: create,
+            status,
+            created_at,
+            updated_at: Utc::now(),
+        })
+    }
+
+    fn calculate_hash(
+        reference: &str,
+        spend: Vec<&PaymentId>,
+        create: Vec<(&AccountId, &Amount)>,
+        created_at: DateTime<Utc>,
+    ) -> Result<TransactionId, Error> {
+        let mut hasher = Sha256::new();
+        let mut spend = spend;
+
+        spend.sort();
+
+        for id in spend.into_iter() {
+            hasher.update(&bincode::serialize(id)?);
+        }
+        for (account, amount) in create.into_iter() {
+            hasher.update(&bincode::serialize(account)?);
+            hasher.update(&bincode::serialize(amount)?);
+        }
+        hasher.update(&created_at.timestamp_millis().to_le_bytes());
+        hasher.update(&reference);
+        Ok(TransactionId::new(hasher.finalize().into()))
+    }
+
+    pub async fn settle<'a, B, S>(&mut self, storage: &'a S) -> Result<(), Error>
+    where
+        B: Batch<'a>,
+        S: Storage<'a, B> + Sync + Send,
+    {
+        self.change_status(Status::Settled)?;
+        self.persist::<B, S>(storage).await
+    }
+
+    #[inline]
+    pub fn change_status(&mut self, new_status: Status) -> Result<(), Error> {
+        if self.status.can_transition_to(&new_status) {
+            self.spends.iter_mut().for_each(|payment| {
+                payment.status = new_status.clone();
+                if new_status.is_rollback() {
+                    payment.spent_by = None;
+                }
+            });
+            self.creates.iter_mut().for_each(|payment| {
+                payment.status = new_status.clone();
+            });
+            self.status = new_status;
+            Ok(())
+        } else {
+            Err(Error::StatusTransitionNotAllowed(
+                self.status.clone(),
+                new_status,
+            ))
+        }
+    }
+
+    #[inline]
+    fn check_no_negative_amounts_are_spent(
+        &self,
+        debit: &HashMap<Asset, i128>,
+    ) -> Result<(), Error> {
+        for (asset, amount) in debit.iter() {
+            if *amount <= 0 {
+                return Err(Error::InvalidAmount(
+                    asset.new_amount(*amount),
+                    asset.new_amount(*amount),
+                ));
+            }
+        }
+
+        Ok(())
+    }
+
+    pub(crate) fn validate(&self) -> Result<(), Error> {
+        let calculated_id = Self::calculate_hash(
+            &self.reference,
+            self.spends.iter().map(|p| &p.id).collect::<Vec<_>>(),
+            self.creates
+                .iter()
+                .map(|p| (&p.to, &p.amount))
+                .collect::<Vec<_>>(),
+            self.created_at,
+        )?;
+
+        if calculated_id != self.id {
+            return Err(Error::InvalidTransactionId(self.id.clone(), calculated_id));
+        }
+
+        let mut debit = HashMap::<Asset, AmountCents>::new();
+        let mut credit = HashMap::<Asset, AmountCents>::new();
+
+        for (i, input) in self.spends.iter().enumerate() {
+            if input.spent_by.is_some() && input.spent_by.as_ref() != Some(&self.id) {
+                return Err(Error::SpentPayment(i));
+            }
+            if let Some(value) = debit.get_mut(input.amount.asset()) {
+                *value = input
+                    .amount
+                    .cents()
+                    .checked_add(*value)
+                    .ok_or(Error::Overflow)?;
+            } else {
+                debit.insert(*input.amount.asset(), input.amount.cents());
+            }
+        }
+
+        self.check_no_negative_amounts_are_spent(&debit)?;
+
+        if !self.typ.is_transaction() {
+            // We don't care input/output balance in external operations
+            // (withdrawals/deposits), because these operations are inbalanced
+            return Ok(());
+        }
+
+        for output in self.creates.iter() {
+            if let Some(value) = credit.get_mut(output.amount.asset()) {
+                *value = output
+                    .amount
+                    .cents()
+                    .checked_add(*value)
+                    .ok_or(Error::Overflow)?;
+            } else {
+                credit.insert(*output.amount.asset(), output.amount.cents());
+            }
+        }
+
+        for (asset, credit_amount) in credit.into_iter() {
+            if let Some(debit_amount) = debit.remove(&asset) {
+                if debit_amount != credit_amount {
+                    return Err(Error::InvalidAmount(
+                        asset.new_amount(debit_amount),
+                        asset.new_amount(credit_amount),
+                    ));
+                }
+            } else {
+                return Err(Error::MissingSpendingAsset(asset));
+            }
+        }
+
+        if let Some((asset, _)) = debit.into_iter().next() {
+            return Err(Error::MissingPaymentAsset(asset));
+        }
+
+        Ok(())
+    }
+
+    pub fn spends(&self) -> &[Payment] {
+        &self.spends
+    }
+
+    pub fn creates(&self) -> &[Payment] {
+        &self.creates
+    }
+
+    pub fn id(&self) -> &TransactionId {
+        &self.id
+    }
+
+    pub fn status(&self) -> &Status {
+        &self.status
+    }
+
+    pub fn typ(&self) -> &Type {
+        &self.typ
+    }
+
+    pub fn reference(&self) -> &str {
+        &self.reference
+    }
+
+    pub fn created_at(&self) -> DateTime<Utc> {
+        self.created_at
+    }
+
+    pub fn updated_at(&self) -> DateTime<Utc> {
+        self.updated_at
+    }
+
+    pub async fn persist<'a, B, S>(&mut self, storage: &'a S) -> Result<(), Error>
+    where
+        B: Batch<'a>,
+        S: Storage<'a, B> + Sync + Send,
+    {
+        let mut batch = storage.begin().await?;
+        if let Some(status) = batch.get_payment_status(&self.id).await? {
+            if status.is_finalized() {
+                return Err(Error::TransactionUpdatesNotAllowed);
+            }
+        }
+        self.validate()?;
+        self.updated_at = Utc::now();
+        batch.store_transaction(self).await?;
+        for payment in self.creates.iter() {
+            batch.store_new_payment(payment).await?;
+            batch
+                .relate_account_to_transaction(&self, &payment.to)
+                .await?;
+        }
+        for input in self.spends.iter() {
+            batch
+                .spend_payment(&input.id, self.status.clone(), &self.id)
+                .await?;
+            batch
+                .relate_account_to_transaction(&self, &input.to)
+                .await?;
+        }
+        batch.commit().await?;
+        Ok(())
+    }
+}
+
+impl TryFrom<from_db::Transaction> for Transaction {
+    type Error = Error;
+
+    fn try_from(value: from_db::Transaction) -> Result<Self, Self::Error> {
+        let tx = Transaction {
+            id: value.id,
+            typ: value.typ,
+            spends: value.spend,
+            creates: value.create,
+            reference: value.reference,
+            status: value.status,
+            created_at: value.created_at,
+            updated_at: value.updated_at,
+        };
+        tx.validate()?;
+        Ok(tx)
+    }
+}

+ 4 - 254
utxo/src/transaction/mod.rs

@@ -1,256 +1,6 @@
-use crate::{
-    amount::AmountCents, AccountId, Amount, Asset, Batch, Payment, Status, Storage, TransactionId,
-};
-use sha2::{Digest, Sha256};
-use std::collections::HashMap;
-
 mod error;
+pub mod from_db;
+mod inner;
+mod typ;
 
-pub use error::Error;
-
-/// Transactions
-///
-/// Transactions are the core components of the ledger. The transactions are a
-/// list of unspent payments that are about to be spend, to create a new set of
-/// Payments, that can be spend in the future. This model is heavily inspired in
-/// Bitcoin's UTXO model. The terms in this context are payments, spend and
-/// create instead of unspent transactions, input and output.
-///
-/// This simple architecture allows to track accounts pretty efficiently,
-/// because all that matters are unspent payments owned by a given account.
-/// Every spent payment is stored for historical reasons but it is not relevant
-/// for any calculations regarding available funds.
-///
-/// The transaction has a few rules, for instance the sum of spend Payments
-/// should be the same as create Payments, for each easy. There is no 'fee'
-/// concept, so any mismatch in any direction will error the constructor.
-///
-/// Transactions are immutable after they are finalized, and the payments can
-/// only be re-usable if the transaction failed or was cancelled. Once the
-/// transaction settles the spent payments are forever spent. Any rollback
-/// should be a new transaction, initiated by a higher layer.
-///
-/// The spent payments are unavailable until the transaction is finalized,
-/// either as settled, cancelled or failed. A higher layer should split any
-/// available payment to be spend into a new transaction, and then finalize the
-/// transaction, and reserve only the exact amount to be spent, otherwise
-/// unrelated funds will be held unspentable until the transaction is finalized.
-#[derive(Debug, Clone)]
-pub struct Transaction {
-    pub(crate) id: TransactionId,
-    pub(crate) spend: Vec<Payment>,
-    #[allow(dead_code)]
-    pub(crate) reference: String,
-    pub(crate) create: Vec<Payment>,
-    pub(crate) status: Status,
-    pub(crate) is_external_deposit: bool,
-}
-
-impl Transaction {
-    pub fn new_external_deposit(
-        reference: String,
-        status: Status,
-        pay_to: Vec<(AccountId, Amount)>,
-    ) -> Result<Transaction, Error> {
-        let mut hasher = Sha256::new();
-        for (account, amount) in pay_to.iter() {
-            hasher.update(&bincode::serialize(&(account, amount))?);
-        }
-
-        let id = TransactionId::new(hasher.finalize().into());
-
-        Ok(Self {
-            id,
-            spend: vec![],
-            reference,
-            create: pay_to
-                .into_iter()
-                .enumerate()
-                .map(|(position, (to, amount))| Payment {
-                    id: crate::PaymentId {
-                        transaction: id,
-                        position,
-                    },
-                    to,
-                    amount,
-                    spent_by: None,
-                    status: status.clone(),
-                })
-                .collect(),
-            is_external_deposit: true,
-            status,
-        })
-    }
-
-    pub async fn new(
-        reference: String,
-        status: Status,
-        spend: Vec<Payment>,
-        pay_to: Vec<(AccountId, Amount)>,
-    ) -> Result<Transaction, Error> {
-        let mut hasher = Sha256::new();
-        for input in spend.iter() {
-            hasher.update(&bincode::serialize(&input.id)?);
-        }
-
-        for (account, amount) in pay_to.iter() {
-            hasher.update(&bincode::serialize(&(account, amount))?);
-        }
-
-        let id = TransactionId::new(hasher.finalize().into());
-
-        for (i, input) in spend.iter().enumerate() {
-            if input.spent_by.is_some() && input.spent_by != Some(id) {
-                return Err(Error::SpentPayment(i));
-            }
-            if input.spent_by.is_none() && input.status != Status::Settled {
-                return Err(Error::InvalidPaymentStatus(i, input.status.clone()));
-            }
-        }
-
-        Ok(Self {
-            id,
-            reference,
-            spend: spend
-                .into_iter()
-                .map(|mut input| {
-                    input.spent_by = Some(id);
-                    input
-                })
-                .collect(),
-            create: pay_to
-                .into_iter()
-                .enumerate()
-                .map(|(position, (to, amount))| Payment {
-                    id: crate::PaymentId {
-                        transaction: id,
-                        position,
-                    },
-                    to,
-                    amount,
-                    spent_by: None,
-                    status: status.clone(),
-                })
-                .collect(),
-            is_external_deposit: false,
-            status,
-        })
-    }
-
-    pub async fn settle<'a, B, S>(&mut self, storage: &'a S) -> Result<(), Error>
-    where
-        B: Batch<'a>,
-        S: Storage<'a, B> + Sync + Send,
-    {
-        self.change_status(Status::Settled)?;
-        self.persist::<B, S>(storage).await
-    }
-
-    #[inline]
-    pub fn change_status(&mut self, new_status: Status) -> Result<(), Error> {
-        if self.status.can_transition_to(&new_status) {
-            self.spend.iter_mut().for_each(|payment| {
-                payment.status = new_status.clone();
-                if new_status.is_rollback() {
-                    payment.spent_by = None;
-                }
-            });
-            self.create.iter_mut().for_each(|payment| {
-                payment.status = new_status.clone();
-            });
-            self.status = new_status;
-            Ok(())
-        } else {
-            Err(Error::StatusTransitionNotAllowed(
-                self.status.clone(),
-                new_status,
-            ))
-        }
-    }
-
-    fn validate(&mut self) -> Result<(), Error> {
-        if self.is_external_deposit {
-            return Ok(());
-        }
-
-        let mut debit = HashMap::<Asset, AmountCents>::new();
-        let mut credit = HashMap::<Asset, AmountCents>::new();
-
-        for (i, input) in self.spend.iter().enumerate() {
-            if input.spent_by.is_some() && input.spent_by != Some(self.id) {
-                return Err(Error::SpentPayment(i));
-            }
-            if let Some(value) = debit.get_mut(input.amount.asset()) {
-                *value = input
-                    .amount
-                    .cents()
-                    .checked_add(*value)
-                    .ok_or(Error::Overflow)?;
-            } else {
-                debit.insert(*input.amount.asset(), input.amount.cents());
-            }
-        }
-
-        for (i, output) in self.create.iter().enumerate() {
-            if output.spent_by.is_some() {
-                return Err(Error::SpentPayment(i));
-            }
-            if let Some(value) = credit.get_mut(output.amount.asset()) {
-                *value = output
-                    .amount
-                    .cents()
-                    .checked_add(*value)
-                    .ok_or(Error::Overflow)?;
-            } else {
-                credit.insert(*output.amount.asset(), output.amount.cents());
-            }
-        }
-
-        for (asset, credit_amount) in credit.into_iter() {
-            if let Some(debit_amount) = debit.remove(&asset) {
-                if debit_amount != credit_amount {
-                    return Err(Error::InvalidAmount(asset, debit_amount, credit_amount));
-                }
-            } else {
-                return Err(Error::MissingSpendingAsset(asset));
-            }
-        }
-
-        if let Some((asset, _)) = debit.into_iter().next() {
-            return Err(Error::MissingPaymentAsset(asset));
-        }
-
-        Ok(())
-    }
-
-    pub fn spent(&self) -> &[Payment] {
-        &self.spend
-    }
-
-    pub fn created(&self) -> &[Payment] {
-        &self.create
-    }
-
-    pub async fn persist<'a, B, S>(&mut self, storage: &'a S) -> Result<(), Error>
-    where
-        B: Batch<'a>,
-        S: Storage<'a, B> + Sync + Send,
-    {
-        let mut batch = storage.begin().await?;
-        if let Some(status) = batch.get_payment_status(&self.id).await? {
-            if status.is_finalized() {
-                return Err(Error::TransactionUpdatesNotAllowed);
-            }
-        }
-        self.validate()?;
-        batch.store_transaction(self).await?;
-        batch.store_new_payments(&self.create).await?;
-        for input in self.spend.iter_mut() {
-            batch
-                .spend_payment(input.id, self.status.clone(), self.id)
-                .await?;
-        }
-        batch.commit().await?;
-        Ok(())
-    }
-}
+pub use self::{error::Error, inner::Transaction, typ::Type};

+ 53 - 0
utxo/src/transaction/typ.rs

@@ -0,0 +1,53 @@
+use serde::{Deserialize, Serialize};
+
+#[derive(thiserror::Error, Debug)]
+pub enum Error {
+    #[error("Invalid status: {0}")]
+    InvalidStatus(u32),
+}
+
+#[derive(Debug, Clone, Copy, PartialEq, Serialize, Deserialize)]
+#[serde(rename_all = "snake_case")]
+pub enum Type {
+    Deposit,
+    Withdrawal,
+    Transaction,
+    Internal,
+}
+
+impl Type {
+    #[inline]
+    pub fn is_transaction(&self) -> bool {
+        matches!(self, Self::Transaction | Self::Internal)
+    }
+}
+
+impl TryFrom<u32> for Type {
+    type Error = Error;
+    fn try_from(value: u32) -> Result<Self, Self::Error> {
+        match value {
+            0 => Ok(Self::Transaction),
+            1 => Ok(Self::Deposit),
+            2 => Ok(Self::Withdrawal),
+            1000 => Ok(Self::Internal),
+            _ => Err(Error::InvalidStatus(value)),
+        }
+    }
+}
+
+impl From<Type> for u32 {
+    fn from(value: Type) -> Self {
+        (&value).into()
+    }
+}
+
+impl From<&Type> for u32 {
+    fn from(value: &Type) -> Self {
+        match value {
+            Type::Transaction => 0,
+            Type::Deposit => 1,
+            Type::Withdrawal => 2,
+            Type::Internal => 1000,
+        }
+    }
+}