v0.2.9
This commit is contained in:
parent
6a8ec2354f
commit
7cbdbde955
|
@ -224,7 +224,7 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "breeze"
|
name = "breeze"
|
||||||
version = "0.2.8"
|
version = "0.2.9"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"argh",
|
"argh",
|
||||||
|
@ -237,7 +237,6 @@ dependencies = [
|
||||||
"http",
|
"http",
|
||||||
"img-parts",
|
"img-parts",
|
||||||
"rand",
|
"rand",
|
||||||
"rayon",
|
|
||||||
"serde",
|
"serde",
|
||||||
"serde_with",
|
"serde_with",
|
||||||
"tokio",
|
"tokio",
|
||||||
|
@ -320,25 +319,6 @@ dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "crossbeam-deque"
|
|
||||||
version = "0.8.6"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "9dd111b7b7f7d55b72c0a6ae361660ee5853c9af73f70c3c2ef6858b950e2e51"
|
|
||||||
dependencies = [
|
|
||||||
"crossbeam-epoch",
|
|
||||||
"crossbeam-utils",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "crossbeam-epoch"
|
|
||||||
version = "0.9.18"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e"
|
|
||||||
dependencies = [
|
|
||||||
"crossbeam-utils",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "crossbeam-utils"
|
name = "crossbeam-utils"
|
||||||
version = "0.8.21"
|
version = "0.8.21"
|
||||||
|
@ -402,7 +382,6 @@ dependencies = [
|
||||||
"lock_api",
|
"lock_api",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"parking_lot_core",
|
"parking_lot_core",
|
||||||
"rayon",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -425,12 +404,6 @@ dependencies = [
|
||||||
"crypto-common",
|
"crypto-common",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "either"
|
|
||||||
version = "1.13.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "60b1af1c220855b6ceac025d3f6ecdd2b7c4894bfe9cd9bda4fbb4bc7c0d4cf0"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "equivalent"
|
name = "equivalent"
|
||||||
version = "1.0.1"
|
version = "1.0.1"
|
||||||
|
@ -467,23 +440,6 @@ version = "0.3.31"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "05f29059c0c2090612e8d742178b0580d2dc940c837851ad723096f87af6663e"
|
checksum = "05f29059c0c2090612e8d742178b0580d2dc940c837851ad723096f87af6663e"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "futures-io"
|
|
||||||
version = "0.3.31"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "9e5c1b78ca4aae1ac06c48a526a655760685149f0d465d21f37abfe57ce075c6"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "futures-macro"
|
|
||||||
version = "0.3.31"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "futures-sink"
|
name = "futures-sink"
|
||||||
version = "0.3.31"
|
version = "0.3.31"
|
||||||
|
@ -503,11 +459,9 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "9fa08315bb612088cc391249efdc3bc77536f16c91f6cf495e6fbe85b20a4a81"
|
checksum = "9fa08315bb612088cc391249efdc3bc77536f16c91f6cf495e6fbe85b20a4a81"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"futures-core",
|
"futures-core",
|
||||||
"futures-macro",
|
|
||||||
"futures-task",
|
"futures-task",
|
||||||
"pin-project-lite",
|
"pin-project-lite",
|
||||||
"pin-utils",
|
"pin-utils",
|
||||||
"slab",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -522,13 +476,14 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "getrandom"
|
name = "getrandom"
|
||||||
version = "0.2.15"
|
version = "0.3.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c4567c8db10ae91089c99af84c68c38da3ec2f087c3f82960bcdbf3656b6f4d7"
|
checksum = "43a49c392881ce6d5c3b8cb70f98717b7c07aabbdff06687b9030dbfbe2725f8"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
"libc",
|
"libc",
|
||||||
"wasi",
|
"wasi 0.13.3+wasi-0.2.2",
|
||||||
|
"windows-targets",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -826,7 +781,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "2886843bf800fba2e3377cff24abf6379b4c4d5c6681eaf9ea5b0d15090450bd"
|
checksum = "2886843bf800fba2e3377cff24abf6379b4c4d5c6681eaf9ea5b0d15090450bd"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"libc",
|
"libc",
|
||||||
"wasi",
|
"wasi 0.11.0+wasi-snapshot-preview1",
|
||||||
"windows-sys 0.52.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -876,16 +831,6 @@ version = "0.1.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "b15813163c1d831bf4a13c3610c05c0d03b39feb07f7e09fa234dac9b15aaf39"
|
checksum = "b15813163c1d831bf4a13c3610c05c0d03b39feb07f7e09fa234dac9b15aaf39"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "parking_lot"
|
|
||||||
version = "0.12.3"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "f1bf18183cf54e8d6059647fc3063646a1801cf30896933ec2311622cc4b9a27"
|
|
||||||
dependencies = [
|
|
||||||
"lock_api",
|
|
||||||
"parking_lot_core",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "parking_lot_core"
|
name = "parking_lot_core"
|
||||||
version = "0.9.10"
|
version = "0.9.10"
|
||||||
|
@ -935,7 +880,7 @@ version = "0.2.20"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04"
|
checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"zerocopy",
|
"zerocopy 0.7.35",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -958,20 +903,20 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rand"
|
name = "rand"
|
||||||
version = "0.8.5"
|
version = "0.9.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
|
checksum = "3779b94aeb87e8bd4e834cee3650289ee9e0d5677f976ecdb6d219e5f4f6cd94"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"libc",
|
|
||||||
"rand_chacha",
|
"rand_chacha",
|
||||||
"rand_core",
|
"rand_core",
|
||||||
|
"zerocopy 0.8.17",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rand_chacha"
|
name = "rand_chacha"
|
||||||
version = "0.3.1"
|
version = "0.9.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
|
checksum = "d3022b5f1df60f26e1ffddd6c66e8aa15de382ae63b3a0c1bfc0e4d3e3f325cb"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"ppv-lite86",
|
"ppv-lite86",
|
||||||
"rand_core",
|
"rand_core",
|
||||||
|
@ -979,31 +924,12 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rand_core"
|
name = "rand_core"
|
||||||
version = "0.6.4"
|
version = "0.9.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c"
|
checksum = "b08f3c9802962f7e1b25113931d94f43ed9725bebc59db9d0c3e9a23b67e15ff"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"getrandom",
|
"getrandom",
|
||||||
]
|
"zerocopy 0.8.17",
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rayon"
|
|
||||||
version = "1.10.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "b418a60154510ca1a002a752ca9714984e21e4241e804d32555251faf8b78ffa"
|
|
||||||
dependencies = [
|
|
||||||
"either",
|
|
||||||
"rayon-core",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rayon-core"
|
|
||||||
version = "1.12.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "1465873a3dfdaa8ae7cb14b4383657caab0b3e8a0aa9ae8e04b044854c8dfce2"
|
|
||||||
dependencies = [
|
|
||||||
"crossbeam-deque",
|
|
||||||
"crossbeam-utils",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -1281,7 +1207,6 @@ dependencies = [
|
||||||
"bytes",
|
"bytes",
|
||||||
"libc",
|
"libc",
|
||||||
"mio",
|
"mio",
|
||||||
"parking_lot",
|
|
||||||
"pin-project-lite",
|
"pin-project-lite",
|
||||||
"signal-hook-registry",
|
"signal-hook-registry",
|
||||||
"socket2",
|
"socket2",
|
||||||
|
@ -1319,12 +1244,8 @@ checksum = "d7fcaa8d55a2bdd6b83ace262b016eca0d79ee02818c5c1bcdf0305114081078"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bytes",
|
"bytes",
|
||||||
"futures-core",
|
"futures-core",
|
||||||
"futures-io",
|
|
||||||
"futures-sink",
|
"futures-sink",
|
||||||
"futures-util",
|
|
||||||
"hashbrown 0.14.5",
|
|
||||||
"pin-project-lite",
|
"pin-project-lite",
|
||||||
"slab",
|
|
||||||
"tokio",
|
"tokio",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -1488,6 +1409,15 @@ version = "0.11.0+wasi-snapshot-preview1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423"
|
checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "wasi"
|
||||||
|
version = "0.13.3+wasi-0.2.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "26816d2e1a4a36a2940b96c5296ce403917633dff8f3440e9b236ed6f6bacad2"
|
||||||
|
dependencies = [
|
||||||
|
"wit-bindgen-rt",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "wasm-bindgen"
|
name = "wasm-bindgen"
|
||||||
version = "0.2.99"
|
version = "0.2.99"
|
||||||
|
@ -1673,6 +1603,15 @@ dependencies = [
|
||||||
"memchr",
|
"memchr",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "wit-bindgen-rt"
|
||||||
|
version = "0.33.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3268f3d866458b787f390cf61f4bbb563b922d091359f9608842999eaee3943c"
|
||||||
|
dependencies = [
|
||||||
|
"bitflags",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "zerocopy"
|
name = "zerocopy"
|
||||||
version = "0.7.35"
|
version = "0.7.35"
|
||||||
|
@ -1680,7 +1619,16 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0"
|
checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"byteorder",
|
"byteorder",
|
||||||
"zerocopy-derive",
|
"zerocopy-derive 0.7.35",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "zerocopy"
|
||||||
|
version = "0.8.17"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "aa91407dacce3a68c56de03abe2760159582b846c6a4acd2f456618087f12713"
|
||||||
|
dependencies = [
|
||||||
|
"zerocopy-derive 0.8.17",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -1693,3 +1641,14 @@ dependencies = [
|
||||||
"quote",
|
"quote",
|
||||||
"syn",
|
"syn",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "zerocopy-derive"
|
||||||
|
version = "0.8.17"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "06718a168365cad3d5ff0bb133aad346959a2074bd4a85c121255a11304a8626"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"syn",
|
||||||
|
]
|
||||||
|
|
17
Cargo.toml
17
Cargo.toml
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "breeze"
|
name = "breeze"
|
||||||
version = "0.2.8"
|
version = "0.2.9"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
@ -12,20 +12,25 @@ axum = { version = "0.8.1", features = ["macros", "http2"] }
|
||||||
tower = "0.5"
|
tower = "0.5"
|
||||||
http = "1.2"
|
http = "1.2"
|
||||||
headers = "0.4"
|
headers = "0.4"
|
||||||
tokio = { version = "1", features = ["full"] }
|
tokio = { version = "1", features = [
|
||||||
tokio-util = { version = "0.7", features = ["full"] }
|
"rt-multi-thread",
|
||||||
|
"macros",
|
||||||
|
"net",
|
||||||
|
"fs",
|
||||||
|
"signal",
|
||||||
|
] }
|
||||||
|
tokio-util = { version = "0.7", features = ["io"] }
|
||||||
tokio-stream = "0.1"
|
tokio-stream = "0.1"
|
||||||
tracing = "0.1"
|
tracing = "0.1"
|
||||||
tracing-subscriber = "0.3"
|
tracing-subscriber = "0.3"
|
||||||
bytes = "1"
|
bytes = "1"
|
||||||
rand = "0.8.5"
|
rand = "0.9"
|
||||||
walkdir = "2"
|
walkdir = "2"
|
||||||
anyhow = "1.0"
|
anyhow = "1.0"
|
||||||
serde = { version = "1.0", features = ["derive"] }
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
serde_with = "3.12"
|
serde_with = "3.12"
|
||||||
toml = "0.8.2"
|
toml = "0.8.2"
|
||||||
argh = "0.1.12"
|
argh = "0.1.12"
|
||||||
dashmap = { version = "6.1.0", features = ["rayon", "inline"] }
|
dashmap = { version = "6.1.0", features = ["inline"] }
|
||||||
rayon = "1.8"
|
|
||||||
atomic-time = "0.1.4"
|
atomic-time = "0.1.4"
|
||||||
img-parts = "0.3"
|
img-parts = "0.3"
|
||||||
|
|
17
src/cache.rs
17
src/cache.rs
|
@ -6,7 +6,6 @@ use std::{
|
||||||
use atomic_time::AtomicSystemTime;
|
use atomic_time::AtomicSystemTime;
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
use dashmap::{mapref::one::Ref, DashMap};
|
use dashmap::{mapref::one::Ref, DashMap};
|
||||||
use rayon::prelude::*;
|
|
||||||
use tokio::time;
|
use tokio::time;
|
||||||
|
|
||||||
use crate::config;
|
use crate::config;
|
||||||
|
@ -81,7 +80,7 @@ impl Cache {
|
||||||
let mut sorted: Vec<_> = self.map.iter().collect();
|
let mut sorted: Vec<_> = self.map.iter().collect();
|
||||||
|
|
||||||
// Sort by least recently used
|
// Sort by least recently used
|
||||||
sorted.par_sort_unstable_by(|e1, e2| e1.last_used().cmp(&e2.last_used()));
|
sorted.sort_unstable_by_key(|e| e.last_used());
|
||||||
|
|
||||||
// Total bytes we would be removing
|
// Total bytes we would be removing
|
||||||
let mut total = 0;
|
let mut total = 0;
|
||||||
|
@ -142,12 +141,12 @@ impl Cache {
|
||||||
// How far we went above the limit
|
// How far we went above the limit
|
||||||
let needed = new_total - self.cfg.mem_capacity;
|
let needed = new_total - self.cfg.mem_capacity;
|
||||||
|
|
||||||
self.next_out(needed).par_iter().for_each(|k| {
|
self.next_out(needed).iter().for_each(|k| {
|
||||||
// Remove the element, and ignore the result
|
// Remove the element, and ignore the result
|
||||||
// The only reason it should be failing is if it couldn't find it,
|
// The only reason it should be failing is if it couldn't find it,
|
||||||
// in which case it was already removed
|
// in which case it was already removed
|
||||||
self.remove(k);
|
self.remove(k);
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Atomically add to total cached data length
|
// Atomically add to total cached data length
|
||||||
|
@ -170,7 +169,7 @@ impl Cache {
|
||||||
///
|
///
|
||||||
/// It exists so we can run the expiry check before
|
/// It exists so we can run the expiry check before
|
||||||
/// actually working with any entries, so no weird bugs happen
|
/// actually working with any entries, so no weird bugs happen
|
||||||
fn _get(&self, key: &str) -> Option<Ref<String, Entry>> {
|
fn get_(&self, key: &str) -> Option<Ref<String, Entry>> {
|
||||||
let e = self.map.get(key)?;
|
let e = self.map.get(key)?;
|
||||||
|
|
||||||
// if the entry is expired get rid of it now
|
// if the entry is expired get rid of it now
|
||||||
|
@ -190,7 +189,7 @@ impl Cache {
|
||||||
|
|
||||||
/// Get an item from the cache, if it exists.
|
/// Get an item from the cache, if it exists.
|
||||||
pub fn get(&self, key: &str) -> Option<Bytes> {
|
pub fn get(&self, key: &str) -> Option<Bytes> {
|
||||||
let e = self._get(key)?;
|
let e = self.get_(key)?;
|
||||||
|
|
||||||
if e.update_used {
|
if e.update_used {
|
||||||
e.last_used.store(SystemTime::now(), Ordering::Relaxed);
|
e.last_used.store(SystemTime::now(), Ordering::Relaxed);
|
||||||
|
@ -206,7 +205,7 @@ impl Cache {
|
||||||
/// We don't use [`DashMap::contains_key`] here because it would just do
|
/// We don't use [`DashMap::contains_key`] here because it would just do
|
||||||
/// the exact same thing I do here, but without running the expiry check logic
|
/// the exact same thing I do here, but without running the expiry check logic
|
||||||
pub fn has(&self, key: &str) -> bool {
|
pub fn has(&self, key: &str) -> bool {
|
||||||
self._get(key).is_some()
|
self.get_(key).is_some()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns if an upload is able to be cached
|
/// Returns if an upload is able to be cached
|
||||||
|
@ -235,7 +234,7 @@ impl Cache {
|
||||||
// If we fail to compare the times, it gets added to the list anyways
|
// If we fail to compare the times, it gets added to the list anyways
|
||||||
let expired: Vec<_> = self
|
let expired: Vec<_> = self
|
||||||
.map
|
.map
|
||||||
.par_iter()
|
.iter()
|
||||||
.filter_map(|e| {
|
.filter_map(|e| {
|
||||||
let elapsed = now.duration_since(e.last_used()).unwrap_or(Duration::MAX);
|
let elapsed = now.duration_since(e.last_used()).unwrap_or(Duration::MAX);
|
||||||
let is_expired = elapsed >= e.lifetime;
|
let is_expired = elapsed >= e.lifetime;
|
||||||
|
@ -252,7 +251,7 @@ impl Cache {
|
||||||
if !expired.is_empty() {
|
if !expired.is_empty() {
|
||||||
// Use a retain call, should be less locks that way
|
// Use a retain call, should be less locks that way
|
||||||
// (instead of many remove calls)
|
// (instead of many remove calls)
|
||||||
self.map.retain(|k, _| !expired.contains(k))
|
self.map.retain(|k, _| !expired.contains(k));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
11
src/disk.rs
11
src/disk.rs
|
@ -1,4 +1,4 @@
|
||||||
use std::path::PathBuf;
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
use tokio::{
|
use tokio::{
|
||||||
|
@ -32,8 +32,11 @@ impl Disk {
|
||||||
|
|
||||||
/// Formats the path on disk for a `saved_name`.
|
/// Formats the path on disk for a `saved_name`.
|
||||||
fn path_for(&self, saved_name: &str) -> PathBuf {
|
fn path_for(&self, saved_name: &str) -> PathBuf {
|
||||||
let mut p = self.cfg.save_path.clone();
|
// try to prevent path traversal by ignoring everything except the file name
|
||||||
p.push(saved_name);
|
let name = Path::new(saved_name).file_name().unwrap_or_default();
|
||||||
|
|
||||||
|
let mut p: PathBuf = self.cfg.save_path.clone();
|
||||||
|
p.push(name);
|
||||||
|
|
||||||
p
|
p
|
||||||
}
|
}
|
||||||
|
@ -65,7 +68,7 @@ impl Disk {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create a background I/O task
|
/// Create a background I/O task
|
||||||
pub async fn start_save(&self, saved_name: &str) -> mpsc::UnboundedSender<Bytes> {
|
pub fn start_save(&self, saved_name: &str) -> mpsc::UnboundedSender<Bytes> {
|
||||||
// start a task that handles saving files to disk (we can save to cache/disk in parallel that way)
|
// start a task that handles saving files to disk (we can save to cache/disk in parallel that way)
|
||||||
let (tx, mut rx): (mpsc::UnboundedSender<Bytes>, mpsc::UnboundedReceiver<Bytes>) =
|
let (tx, mut rx): (mpsc::UnboundedSender<Bytes>, mpsc::UnboundedReceiver<Bytes>) =
|
||||||
mpsc::unbounded_channel();
|
mpsc::unbounded_channel();
|
||||||
|
|
|
@ -7,10 +7,11 @@ use std::{
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use anyhow::Context as _;
|
||||||
use axum::body::BodyDataStream;
|
use axum::body::BodyDataStream;
|
||||||
use bytes::{BufMut, Bytes, BytesMut};
|
use bytes::{BufMut, Bytes, BytesMut};
|
||||||
use img_parts::{DynImage, ImageEXIF};
|
use img_parts::{DynImage, ImageEXIF};
|
||||||
use rand::distributions::{Alphanumeric, DistString};
|
use rand::distr::{Alphanumeric, SampleString};
|
||||||
use tokio::{
|
use tokio::{
|
||||||
fs::File,
|
fs::File,
|
||||||
io::{AsyncReadExt, AsyncSeekExt},
|
io::{AsyncReadExt, AsyncSeekExt},
|
||||||
|
@ -146,9 +147,7 @@ impl Engine {
|
||||||
u
|
u
|
||||||
} else {
|
} else {
|
||||||
// now, check if we have it on disk
|
// now, check if we have it on disk
|
||||||
let mut f = if let Some(f) = self.disk.open(saved_name).await? {
|
let Some(mut f) = self.disk.open(saved_name).await? else {
|
||||||
f
|
|
||||||
} else {
|
|
||||||
// file didn't exist
|
// file didn't exist
|
||||||
return Ok(GetOutcome::NotFound);
|
return Ok(GetOutcome::NotFound);
|
||||||
};
|
};
|
||||||
|
@ -180,9 +179,7 @@ impl Engine {
|
||||||
|
|
||||||
data
|
data
|
||||||
} else {
|
} else {
|
||||||
let (start, end) = if let Some(range) = resolve_range(range, full_len) {
|
let Some((start, end)) = resolve_range(range, full_len) else {
|
||||||
range
|
|
||||||
} else {
|
|
||||||
return Ok(GetOutcome::RangeNotSatisfiable);
|
return Ok(GetOutcome::RangeNotSatisfiable);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -201,9 +198,7 @@ impl Engine {
|
||||||
};
|
};
|
||||||
|
|
||||||
let full_len = data.len() as u64;
|
let full_len = data.len() as u64;
|
||||||
let (start, end) = if let Some(range) = resolve_range(range, full_len) {
|
let Some((start, end)) = resolve_range(range, full_len) else {
|
||||||
range
|
|
||||||
} else {
|
|
||||||
return Ok(GetOutcome::RangeNotSatisfiable);
|
return Ok(GetOutcome::RangeNotSatisfiable);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -243,7 +238,7 @@ impl Engine {
|
||||||
pub async fn gen_saved_name(&self, ext: Option<String>) -> String {
|
pub async fn gen_saved_name(&self, ext: Option<String>) -> String {
|
||||||
loop {
|
loop {
|
||||||
// generate a 6-character alphanumeric string
|
// generate a 6-character alphanumeric string
|
||||||
let mut saved_name: String = Alphanumeric.sample_string(&mut rand::thread_rng(), 6);
|
let mut saved_name: String = Alphanumeric.sample_string(&mut rand::rng(), 6);
|
||||||
|
|
||||||
// if we have an extension, add it now
|
// if we have an extension, add it now
|
||||||
if let Some(ref ext) = ext {
|
if let Some(ref ext) = ext {
|
||||||
|
@ -267,7 +262,10 @@ impl Engine {
|
||||||
info!("!! removing upload: {saved_name}");
|
info!("!! removing upload: {saved_name}");
|
||||||
|
|
||||||
self.cache.remove(saved_name);
|
self.cache.remove(saved_name);
|
||||||
self.disk.remove(saved_name).await?;
|
self.disk
|
||||||
|
.remove(saved_name)
|
||||||
|
.await
|
||||||
|
.context("failed to remove file from disk")?;
|
||||||
|
|
||||||
info!("!! successfully removed upload");
|
info!("!! successfully removed upload");
|
||||||
|
|
||||||
|
@ -295,12 +293,12 @@ impl Engine {
|
||||||
|
|
||||||
// don't begin a disk save if we're using temporary lifetimes
|
// don't begin a disk save if we're using temporary lifetimes
|
||||||
let tx = if lifetime.is_none() {
|
let tx = if lifetime.is_none() {
|
||||||
Some(self.disk.start_save(saved_name).await)
|
Some(self.disk.start_save(saved_name))
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
};
|
};
|
||||||
|
|
||||||
// whether or not we're gonna coalesce the data
|
// whether or not we are going to coalesce the data
|
||||||
// in order to strip the exif data at the end,
|
// in order to strip the exif data at the end,
|
||||||
// instead of just sending it off to the i/o task
|
// instead of just sending it off to the i/o task
|
||||||
let coalesce_and_strip = use_cache
|
let coalesce_and_strip = use_cache
|
||||||
|
@ -323,7 +321,8 @@ impl Engine {
|
||||||
if !coalesce_and_strip {
|
if !coalesce_and_strip {
|
||||||
if let Some(ref tx) = tx {
|
if let Some(ref tx) = tx {
|
||||||
debug!("sending chunk to i/o task");
|
debug!("sending chunk to i/o task");
|
||||||
tx.send(chunk.clone())?;
|
tx.send(chunk.clone())
|
||||||
|
.context("failed to send chunk to i/o task!")?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -365,7 +364,8 @@ impl Engine {
|
||||||
// send what we did over to the i/o task, all in one chunk
|
// send what we did over to the i/o task, all in one chunk
|
||||||
if let Some(ref tx) = tx {
|
if let Some(ref tx) = tx {
|
||||||
debug!("sending filled buffer to i/o task");
|
debug!("sending filled buffer to i/o task");
|
||||||
tx.send(data.clone())?;
|
tx.send(data.clone())
|
||||||
|
.context("failed to send coalesced buffer to i/o task!")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
data
|
data
|
||||||
|
|
|
@ -99,8 +99,8 @@ async fn shutdown_signal() {
|
||||||
let terminate = std::future::pending::<()>();
|
let terminate = std::future::pending::<()>();
|
||||||
|
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
_ = ctrl_c => {},
|
() = ctrl_c => {},
|
||||||
_ = terminate => {},
|
() = terminate => {},
|
||||||
}
|
}
|
||||||
|
|
||||||
info!("shutting down!");
|
info!("shutting down!");
|
||||||
|
|
|
@ -14,6 +14,7 @@ use headers::ContentLength;
|
||||||
use http::StatusCode;
|
use http::StatusCode;
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use serde_with::{serde_as, DurationSeconds};
|
use serde_with::{serde_as, DurationSeconds};
|
||||||
|
use tracing::error;
|
||||||
|
|
||||||
use crate::engine::ProcessOutcome;
|
use crate::engine::ProcessOutcome;
|
||||||
|
|
||||||
|
@ -90,7 +91,7 @@ pub async fn new(
|
||||||
|
|
||||||
// pass it off to the engine to be processed
|
// pass it off to the engine to be processed
|
||||||
// --
|
// --
|
||||||
// also, error responses here don't get represented properly in ShareX most of the time
|
// also, error responses here don't get presented properly in ShareX most of the time
|
||||||
// they don't expect the connection to close before they're done uploading, i think
|
// they don't expect the connection to close before they're done uploading, i think
|
||||||
// so it will just present the user with a "connection closed" error
|
// so it will just present the user with a "connection closed" error
|
||||||
match engine
|
match engine
|
||||||
|
@ -111,6 +112,9 @@ pub async fn new(
|
||||||
},
|
},
|
||||||
|
|
||||||
// 500 Internal Server Error
|
// 500 Internal Server Error
|
||||||
Err(_) => Err(StatusCode::INTERNAL_SERVER_ERROR),
|
Err(err) => {
|
||||||
|
error!("failed to process upload!! {err:#}");
|
||||||
|
Err(StatusCode::INTERNAL_SERVER_ERROR)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
17
src/view.rs
17
src/view.rs
|
@ -10,6 +10,7 @@ use axum_extra::TypedHeader;
|
||||||
use headers::Range;
|
use headers::Range;
|
||||||
use http::{HeaderValue, StatusCode};
|
use http::{HeaderValue, StatusCode};
|
||||||
use tokio_util::io::ReaderStream;
|
use tokio_util::io::ReaderStream;
|
||||||
|
use tracing::error;
|
||||||
|
|
||||||
use crate::engine::{GetOutcome, UploadData, UploadResponse};
|
use crate::engine::{GetOutcome, UploadData, UploadResponse};
|
||||||
|
|
||||||
|
@ -91,19 +92,23 @@ pub async fn view(
|
||||||
Path(original_path): Path<PathBuf>,
|
Path(original_path): Path<PathBuf>,
|
||||||
range: Option<TypedHeader<Range>>,
|
range: Option<TypedHeader<Range>>,
|
||||||
) -> Result<UploadResponse, ViewError> {
|
) -> Result<UploadResponse, ViewError> {
|
||||||
let saved_name = if let Some(Some(n)) = original_path.file_name().map(OsStr::to_str) {
|
// try to extract the file name (if it's the only component)
|
||||||
n
|
// this makes paths like `asdf%2fabcdef.png` invalid
|
||||||
} else {
|
let saved_name = match original_path.file_name().map(OsStr::to_str) {
|
||||||
return Err(ViewError::NotFound);
|
Some(Some(n)) if original_path.components().count() == 1 => n,
|
||||||
|
_ => return Err(ViewError::NotFound),
|
||||||
};
|
};
|
||||||
|
|
||||||
let range = range.map(|th| th.0);
|
let range = range.map(|TypedHeader(range)| range);
|
||||||
|
|
||||||
// get result from the engine
|
// get result from the engine
|
||||||
match engine.get(saved_name, range).await {
|
match engine.get(saved_name, range).await {
|
||||||
Ok(GetOutcome::Success(res)) => Ok(res),
|
Ok(GetOutcome::Success(res)) => Ok(res),
|
||||||
Ok(GetOutcome::NotFound) => Err(ViewError::NotFound),
|
Ok(GetOutcome::NotFound) => Err(ViewError::NotFound),
|
||||||
Ok(GetOutcome::RangeNotSatisfiable) => Err(ViewError::RangeNotSatisfiable),
|
Ok(GetOutcome::RangeNotSatisfiable) => Err(ViewError::RangeNotSatisfiable),
|
||||||
Err(_) => Err(ViewError::InternalServerError),
|
Err(err) => {
|
||||||
|
error!("failed to get upload!! {err:#}");
|
||||||
|
Err(ViewError::InternalServerError)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue