From 2de9baa38c8bc5c309206164eea2c4525f68c0ea Mon Sep 17 00:00:00 2001 From: Takumasa Sakao Date: Fri, 16 Aug 2024 15:54:05 +0900 Subject: [PATCH 1/4] Rename to MemoryStoreData --- src/app.rs | 2 +- src/runner.rs | 6 +++--- src/store.rs | 10 +++++----- 3 files changed, 9 insertions(+), 9 deletions(-) diff --git a/src/app.rs b/src/app.rs index b0f9803..5d17a76 100644 --- a/src/app.rs +++ b/src/app.rs @@ -138,7 +138,7 @@ impl App { pub async fn run(&mut self) -> Result<()> { let (action_tx, mut action_rx) = mpsc::unbounded_channel(); - let store = crate::store::Store::new(); + let store = crate::store::MemoryStore::new(); let executor_handle = if self.is_precise { tokio::spawn(run_executor_precise( action_tx.clone(), diff --git a/src/runner.rs b/src/runner.rs index bbc1d41..72e6606 100644 --- a/src/runner.rs +++ b/src/runner.rs @@ -10,13 +10,13 @@ use crate::{ action::Action, config::{Config, RuntimeConfig}, exec::exec, - store::{Record, Store}, + store::{Record, MemoryStore}, types::ExecutionId, }; pub async fn run_executor( actions: mpsc::UnboundedSender, - mut store: Store, + mut store: MemoryStore, runtime_config: RuntimeConfig, shell: Option<(String, Vec)>, is_suspend: Arc>, @@ -90,7 +90,7 @@ pub async fn run_executor( pub async fn run_executor_precise( actions: mpsc::UnboundedSender, - mut store: Store, + mut store: MemoryStore, runtime_config: RuntimeConfig, shell: Option<(String, Vec)>, is_suspend: Arc>, diff --git a/src/store.rs b/src/store.rs index f79c11d..094672f 100644 --- a/src/store.rs +++ b/src/store.rs @@ -20,20 +20,20 @@ pub struct Record { } #[derive(Debug)] -struct StoreData { +struct MemoryStoreData { records: HashMap, latest_id: Option, } #[derive(Clone, Debug)] -pub struct Store { - data: Arc>, +pub struct MemoryStore { + data: Arc>, } -impl Store { +impl MemoryStore { pub fn new() -> Self { Self { - data: Arc::new(RwLock::new(StoreData { + data: Arc::new(RwLock::new(MemoryStoreData { records: HashMap::new(), latest_id: None, })), From b0eabd4ec6bf53cd7381da3c52e43a1b0dae5290 Mon Sep 17 00:00:00 2001 From: Takumasa Sakao Date: Fri, 16 Aug 2024 16:08:02 +0900 Subject: [PATCH 2/4] Add Store trait --- src/app.rs | 31 +++++++++++-------------------- src/main.rs | 3 ++- src/runner.rs | 10 +++++----- src/store.rs | 14 +++++++++++--- 4 files changed, 29 insertions(+), 29 deletions(-) diff --git a/src/app.rs b/src/app.rs index 5d17a76..5b1e672 100644 --- a/src/app.rs +++ b/src/app.rs @@ -13,20 +13,10 @@ use tokio::{ use tracing_subscriber::field::debug; use crate::{ - action::{self, Action, DiffMode}, - cli::Cli, - components::{fps::FpsCounter, home::Home, Component}, - config::{Config, RuntimeConfig}, - diff::{diff_and_mark, diff_and_mark_delete}, - mode::Mode, - old_config::OldConfig, - runner::{run_executor, run_executor_precise}, - search::search_and_mark, - termtext, tui, - types::ExecutionId, + action::{self, Action, DiffMode}, cli::Cli, components::{fps::FpsCounter, home::Home, Component}, config::{Config, RuntimeConfig}, diff::{diff_and_mark, diff_and_mark_delete}, mode::Mode, old_config::OldConfig, runner::{run_executor, run_executor_precise}, search::search_and_mark, store::Store, termtext, tui, types::ExecutionId }; -pub struct App { +pub struct App { pub config: Config, pub runtime_config: RuntimeConfig, pub tick_rate: f64, @@ -38,6 +28,7 @@ pub struct App { pub last_tick_key_events: Vec, pub timemachine_mode: bool, pub search_query: Option, + store: S, is_precise: bool, diff_mode: Option, is_suspend: Arc>, @@ -49,8 +40,8 @@ pub struct App { shell: Option<(String, Vec)>, } -impl App { - pub fn new(cli: Cli) -> Result { +impl App { + pub fn new(cli: Cli, store: S) -> Result { let runtime_config = RuntimeConfig { interval: cli.interval, command: cli.command, @@ -108,6 +99,7 @@ impl App { let is_skip_empty_diffs = cli.is_skip_empty_diffs || default_skip_empty_diffs; Ok(Self { + store, tick_rate: 1.0, frame_rate: 20.0, components, @@ -138,11 +130,10 @@ impl App { pub async fn run(&mut self) -> Result<()> { let (action_tx, mut action_rx) = mpsc::unbounded_channel(); - let store = crate::store::MemoryStore::new(); let executor_handle = if self.is_precise { tokio::spawn(run_executor_precise( action_tx.clone(), - store.clone(), + self.store.clone(), self.runtime_config.clone(), self.shell.clone(), self.is_suspend.clone(), @@ -150,7 +141,7 @@ impl App { } else { tokio::spawn(run_executor( action_tx.clone(), - store.clone(), + self.store.clone(), self.runtime_config.clone(), self.shell.clone(), self.is_suspend.clone(), @@ -283,7 +274,7 @@ impl App { Action::ShowExecution(id, end_id) => { let style = termtext::convert_to_anstyle(self.config.get_style("background")); - let record = store.get_record(id); + let record = self.store.get_record(id); let mut string = "".to_string(); if let Some(record) = record { action_tx.send(Action::SetClock(record.start_time))?; @@ -301,7 +292,7 @@ impl App { string = result.plain_text(); if let Some(diff_mode) = self.diff_mode { if let Some(previous_id) = record.previous_id { - let previous_record = store.get_record(previous_id); + let previous_record = self.store.get_record(previous_id); if let Some(previous_record) = previous_record { let previous_result = termtext::Converter::new(style) .convert(&previous_record.stdout); @@ -346,7 +337,7 @@ impl App { Action::SetTimemachineMode(timemachine_mode) => { self.timemachine_mode = timemachine_mode; if !timemachine_mode { - if let Some(latest_id) = store.get_latest_id() { + if let Some(latest_id) = self.store.get_latest_id() { action_tx.send(Action::ShowExecution(latest_id, latest_id))?; } } diff --git a/src/main.rs b/src/main.rs index 2ff5cb9..9695786 100644 --- a/src/main.rs +++ b/src/main.rs @@ -38,7 +38,8 @@ async fn tokio_main() -> Result<()> { let args = Cli::parse(); let interval = Duration::from(args.interval); - let mut app = App::new(args)?; + let store = store::MemoryStore::new(); + let mut app = App::new(args, store)?; app.run().await?; Ok(()) diff --git a/src/runner.rs b/src/runner.rs index 72e6606..301852a 100644 --- a/src/runner.rs +++ b/src/runner.rs @@ -10,13 +10,13 @@ use crate::{ action::Action, config::{Config, RuntimeConfig}, exec::exec, - store::{Record, MemoryStore}, + store::{MemoryStore, Record, Store}, types::ExecutionId, }; -pub async fn run_executor( +pub async fn run_executor( actions: mpsc::UnboundedSender, - mut store: MemoryStore, + mut store: S, runtime_config: RuntimeConfig, shell: Option<(String, Vec)>, is_suspend: Arc>, @@ -88,9 +88,9 @@ pub async fn run_executor( } } -pub async fn run_executor_precise( +pub async fn run_executor_precise( actions: mpsc::UnboundedSender, - mut store: MemoryStore, + mut store: S, runtime_config: RuntimeConfig, shell: Option<(String, Vec)>, is_suspend: Arc>, diff --git a/src/store.rs b/src/store.rs index 094672f..cd40794 100644 --- a/src/store.rs +++ b/src/store.rs @@ -7,6 +7,12 @@ use chrono::{DateTime, Local}; use crate::types::ExecutionId; +pub trait Store { + fn add_record(&mut self, record: Record); + fn get_record(&self, id: ExecutionId) -> Option; + fn get_latest_id(&self) -> Option; +} + #[derive(Debug, Clone)] pub struct Record { pub id: ExecutionId, @@ -39,15 +45,17 @@ impl MemoryStore { })), } } +} - pub fn add_record(&mut self, record: Record) { +impl Store for MemoryStore { + fn add_record(&mut self, record: Record) { if let Ok(mut data) = self.data.write() { data.latest_id = Some(record.id); data.records.insert(record.id, record); } } - pub fn get_record(&self, id: ExecutionId) -> Option { + fn get_record(&self, id: ExecutionId) -> Option { if let Ok(data) = self.data.read() { data.records.get(&id).cloned() } else { @@ -55,7 +63,7 @@ impl MemoryStore { } } - pub fn get_latest_id(&self) -> Option { + fn get_latest_id(&self) -> Option { if let Ok(data) = self.data.read() { data.latest_id } else { From 309931ea8915c1da4d22f9d6960f045de55d4c27 Mon Sep 17 00:00:00 2001 From: Takumasa Sakao Date: Fri, 16 Aug 2024 16:28:12 +0900 Subject: [PATCH 3/4] Move MemoryStore to memory.rs --- Cargo.lock | 682 ++++++++++++++++++++++++-------------------- Cargo.toml | 1 + src/main.rs | 2 +- src/runner.rs | 2 +- src/store.rs | 49 +--- src/store/memory.rs | 52 ++++ 6 files changed, 431 insertions(+), 357 deletions(-) create mode 100644 src/store/memory.rs diff --git a/Cargo.lock b/Cargo.lock index 8dca99b..81a73eb 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -71,9 +71,9 @@ dependencies = [ [[package]] name = "ansi-to-tui" -version = "4.0.1" +version = "4.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8438af3d7e7dccdb98eff55e5351587d9bec2294daff505fc9a061bd14d22db0" +checksum = "ba3bf628a79452df9614d933012dc500f8cb6ddad8c897ff8122ea1c0b187ff7" dependencies = [ "nom", "ratatui", @@ -84,9 +84,9 @@ dependencies = [ [[package]] name = "anstream" -version = "0.6.14" +version = "0.6.15" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "418c75fa768af9c03be99d17643f93f79bbba589895012a80e3452a19ddda15b" +checksum = "64e15c1ab1f89faffbf04a634d5e1962e9074f2741eef6d97f3c4e322426d526" dependencies = [ "anstyle", "anstyle-parse", @@ -99,33 +99,33 @@ dependencies = [ [[package]] name = "anstyle" -version = "1.0.7" +version = "1.0.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "038dfcf04a5feb68e9c60b21c9625a54c2c0616e79b72b0fd87075a056ae1d1b" +checksum = "1bec1de6f59aedf83baf9ff929c98f2ad654b97c9510f4e70cf6f661d49fd5b1" [[package]] name = "anstyle-parse" -version = "0.2.4" +version = "0.2.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c03a11a9034d92058ceb6ee011ce58af4a9bf61491aa7e1e59ecd24bd40d22d4" +checksum = "eb47de1e80c2b463c735db5b217a0ddc39d612e7ac9e2e96a5aed1f57616c1cb" dependencies = [ "utf8parse", ] [[package]] name = "anstyle-query" -version = "1.1.0" +version = "1.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ad186efb764318d35165f1758e7dcef3b10628e26d41a44bc5550652e6804391" +checksum = "6d36fc52c7f6c869915e99412912f22093507da8d9e942ceaf66fe4b7c14422a" dependencies = [ "windows-sys 0.52.0", ] [[package]] name = "anstyle-wincon" -version = "3.0.3" +version = "3.0.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "61a38449feb7068f52bb06c12759005cf459ee52bb4adc1d5a7c4322d716fb19" +checksum = "5bf74e1b6e971609db8ca7a9ce79fd5768ab6ae46441c572e46cf596f59e57f8" dependencies = [ "anstyle", "windows-sys 0.52.0", @@ -145,13 +145,13 @@ checksum = "69f7f8c3906b62b754cd5326047894316021dcfe5a194c8ea52bdd94934a3457" [[package]] name = "async-trait" -version = "0.1.80" +version = "0.1.81" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c6fa2087f2753a7da8cc1c0dbfcf89579dd57458e36769de5ac750b4671737ca" +checksum = "6e0c28dcc82d7c8ead5cb13beb15405b57b8546e93215673ff8ca0349a028107" dependencies = [ "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] @@ -199,15 +199,9 @@ dependencies = [ [[package]] name = "bitflags" -version = "1.3.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" - -[[package]] -name = "bitflags" -version = "2.5.0" +version = "2.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cf4b9d6a944f767f8e5e0db018570623c85f3d925ac718db4e06d0187adb21c1" +checksum = "b048fb63fd8b5923fc5aa7b340d8e156aec7ec02f0c78fa8a6ddc2613f6f71de" dependencies = [ "serde", ] @@ -223,24 +217,15 @@ dependencies = [ [[package]] name = "bstr" -version = "1.9.1" +version = "1.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "05efc5cfd9110c8416e471df0e96702d58690178e206e61b7173706673c93706" +checksum = "40723b8fb387abc38f4f4a37c09073622e41dd12327033091ef8950659e6dc0c" dependencies = [ "memchr", "regex-automata 0.4.7", "serde", ] -[[package]] -name = "btoi" -version = "0.4.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9dd6407f73a9b8b6162d8a2ef999fe6afd7cc15902ebf42c5cd296addf17e0ad" -dependencies = [ - "num-traits", -] - [[package]] name = "bumpalo" version = "3.16.0" @@ -255,15 +240,15 @@ checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" [[package]] name = "bytes" -version = "1.6.0" +version = "1.7.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "514de17de45fdb8dc022b1a7975556c53c86f9f0aa5f534b98977b171857c2c9" +checksum = "8318a53db07bb3f8dca91a600466bdb3f2eaadeedfdbcf02e1accbad9271ba50" [[package]] name = "camino" -version = "1.1.7" +version = "1.1.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e0ec6b951b160caa93cc0c7b209e5a3bff7aae9062213451ac99493cd844c239" +checksum = "3054fea8a20d8ff3968d5b22cc27501d2b08dc4decdb31b184323f00c5ef23bb" dependencies = [ "serde", ] @@ -299,18 +284,21 @@ checksum = "df8670b8c7b9dae1793364eafadf7239c40d669904660c5960d74cfd80b46a53" [[package]] name = "castaway" -version = "0.2.2" +version = "0.2.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8a17ed5635fc8536268e5d4de1e22e81ac34419e5f052d4d51f4e01dcc263fcc" +checksum = "0abae9be0aaf9ea96a3b1b8b1b55c602ca751eba1b1500220cea4ecbafe7c0d5" dependencies = [ "rustversion", ] [[package]] name = "cc" -version = "1.0.99" +version = "1.1.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "96c51067fd44124faa7f870b4b1c969379ad32b2ba805aa959430ceaa384f695" +checksum = "68064e60dbf1f17005c2fde4d07c16d8baa506fd7ffed8ccab702d93617975c7" +dependencies = [ + "shlex", +] [[package]] name = "cfg-if" @@ -330,14 +318,14 @@ dependencies = [ "num-traits", "serde", "wasm-bindgen", - "windows-targets 0.52.5", + "windows-targets 0.52.6", ] [[package]] name = "clap" -version = "4.5.7" +version = "4.5.16" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5db83dced34638ad474f39f250d7fea9598bdd239eaced1bdf45d597da0f433f" +checksum = "ed6719fffa43d0d87e5fd8caeab59be1554fb028cd30edc88fc4369b17971019" dependencies = [ "clap_builder", "clap_derive", @@ -345,9 +333,9 @@ dependencies = [ [[package]] name = "clap_builder" -version = "4.5.7" +version = "4.5.15" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f7e204572485eb3fbf28f871612191521df159bc3e15a9f5064c66dba3a8c05f" +checksum = "216aec2b177652e3846684cbfe25c9964d18ec45234f0f5da5157b207ed1aab6" dependencies = [ "anstream", "anstyle", @@ -360,21 +348,21 @@ dependencies = [ [[package]] name = "clap_derive" -version = "4.5.5" +version = "4.5.13" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c780290ccf4fb26629baa7a1081e68ced113f1d3ec302fa5948f1c381ebf06c6" +checksum = "501d359d5f3dcaf6ecdeee48833ae73ec6e42723a1e52419c79abf9507eec0a0" dependencies = [ "heck", "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] name = "clap_lex" -version = "0.7.1" +version = "0.7.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4b82cf0babdbd58558212896d1a4272303a57bdb245c2bf1147185fb45640e70" +checksum = "1462739cb27611015575c0c11df5df7601141071f07518d56fcc1be504cbec97" [[package]] name = "clru" @@ -411,9 +399,9 @@ dependencies = [ [[package]] name = "colorchoice" -version = "1.0.1" +version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0b6a852b24ab71dffc585bcb46eaf7959d175cb865a7152e35b348d1b2960422" +checksum = "d3fd119d74b830634cea2a0f58bbd0d54540518a14397557951e79340abc28c0" [[package]] name = "compact_str" @@ -492,15 +480,15 @@ dependencies = [ [[package]] name = "core-foundation-sys" -version = "0.8.6" +version = "0.8.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "06ea2b9bc92be3c2baa9334a323ebca2d6f074ff852cd1d7b11064035cd3868f" +checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b" [[package]] name = "cpufeatures" -version = "0.2.12" +version = "0.2.13" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "53fe5e26ff1b7aef8bca9c6080520cfb8d9333c7568e1829cef191a9723e5504" +checksum = "51e852e6dc9a5bed1fae92dd2375037bf2b768725bf3be87811edee3249d09ad" dependencies = [ "libc", ] @@ -520,11 +508,11 @@ version = "0.27.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f476fe445d41c9e991fd07515a6f463074b782242ccf4a5b7b1d1012e70824df" dependencies = [ - "bitflags 2.5.0", + "bitflags", "crossterm_winapi", "futures-core", "libc", - "mio", + "mio 0.8.11", "parking_lot", "serde", "signal-hook", @@ -559,9 +547,9 @@ dependencies = [ [[package]] name = "darling" -version = "0.20.9" +version = "0.20.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "83b2eb4d90d12bdda5ed17de686c2acb4c57914f8f921b8da7e112b5a36f3fe1" +checksum = "6f63b86c8a8826a49b8c21f08a2d07338eec8d900540f8630dc76284be802989" dependencies = [ "darling_core", "darling_macro", @@ -569,27 +557,27 @@ dependencies = [ [[package]] name = "darling_core" -version = "0.20.9" +version = "0.20.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "622687fe0bac72a04e5599029151f5796111b90f1baaa9b544d807a5e31cd120" +checksum = "95133861a8032aaea082871032f5815eb9e98cef03fa916ab4500513994df9e5" dependencies = [ "fnv", "ident_case", "proc-macro2", "quote", "strsim", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] name = "darling_macro" -version = "0.20.9" +version = "0.20.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "733cabb43482b1a1b53eee8583c2b9e8684d592215ea83efd305dd31bc2f0178" +checksum = "d336a2a514f6ccccaa3e09b02d41d35330c07ddf03a62165fcec10bb561c7806" dependencies = [ "darling_core", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] @@ -667,15 +655,15 @@ dependencies = [ [[package]] name = "dunce" -version = "1.0.4" +version = "1.0.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "56ce8c6da7551ec6c462cbaf3bfbc75131ebbfa1c944aeaa9dab51ca1c5f0c3b" +checksum = "92773504d58c093f6de2459af4af33faa518c13451eb8f2b5698ed3d36e7c813" [[package]] name = "either" -version = "1.12.0" +version = "1.13.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3dca9240753cf90908d7e4aac30f630662b02aebaa1b58a3cadabdb23385b58b" +checksum = "60b1af1c220855b6ceac025d3f6ecdd2b7c4894bfe9cd9bda4fbb4bc7c0d4cf0" [[package]] name = "encode_unicode" @@ -709,6 +697,18 @@ dependencies = [ "once_cell", ] +[[package]] +name = "fallible-iterator" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2acce4a10f12dc2fb14a218589d4f1f62ef011b2d0cc4b3cb1bba8e94da14649" + +[[package]] +name = "fallible-streaming-iterator" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7360491ce676a36bf9bb3c56c1aa791658183a54d2744120f27285738d90465a" + [[package]] name = "faster-hex" version = "0.9.0" @@ -723,21 +723,21 @@ checksum = "9fc0510504f03c51ada170672ac806f1f105a88aa97a5281117e1ddc3368e51a" [[package]] name = "filetime" -version = "0.2.23" +version = "0.2.24" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1ee447700ac8aa0b2f2bd7bc4462ad686ba06baa6727ac149a2d6277f0d240fd" +checksum = "bf401df4a4e3872c4fe8151134cf483738e74b67fc934d6532c882b3d24a4550" dependencies = [ "cfg-if", "libc", - "redox_syscall 0.4.1", - "windows-sys 0.52.0", + "libredox", + "windows-sys 0.59.0", ] [[package]] name = "flate2" -version = "1.0.30" +version = "1.0.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f54427cfd1c7829e2a139fcefea601bf088ebca651d2bf53ebc600eac295dae" +checksum = "7f211bbe8e69bbd0cfdea405084f128ae8b4aaa6b0b522fc8f2b009084797920" dependencies = [ "crc32fast", "miniz_oxide", @@ -814,7 +814,7 @@ checksum = "87750cf4b7a4c0625b1529e4c543c2182106e4dedc60a2a6455e00d212c489ac" dependencies = [ "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] @@ -876,9 +876,9 @@ checksum = "4271d37baee1b8c7e4b708028c57d816cf9d2434acb33a549475f78c181f6253" [[package]] name = "gix" -version = "0.57.1" +version = "0.63.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6dd025382892c7b500a9ce1582cd803f9c2ebfe44aff52e9c7f86feee7ced75e" +checksum = "984c5018adfa7a4536ade67990b3ebc6e11ab57b3d6cd9968de0947ca99b4b06" dependencies = [ "gix-actor", "gix-commitgraph", @@ -914,21 +914,20 @@ dependencies = [ "signal-hook", "smallvec", "thiserror", - "unicode-normalization", ] [[package]] name = "gix-actor" -version = "0.29.1" +version = "0.31.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "da27b5ab4ab5c75ff891dccd48409f8cc53c28a79480f1efdd33184b2dc1d958" +checksum = "a0e454357e34b833cc3a00b6efbbd3dd4d18b24b9fb0c023876ec2645e8aa3f2" dependencies = [ "bstr", - "btoi", "gix-date", + "gix-utils", "itoa", "thiserror", - "winnow 0.5.40", + "winnow", ] [[package]] @@ -951,9 +950,9 @@ dependencies = [ [[package]] name = "gix-commitgraph" -version = "0.23.2" +version = "0.24.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7e8dcbf434951fa477063e05fea59722615af70dc2567377e58c2f7853b010fc" +checksum = "133b06f67f565836ec0c473e2116a60fb74f80b6435e21d88013ac0e3c60fc78" dependencies = [ "bstr", "gix-chunk", @@ -965,9 +964,9 @@ dependencies = [ [[package]] name = "gix-config" -version = "0.33.1" +version = "0.37.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "367304855b369cadcac4ee5fb5a3a20da9378dd7905106141070b79f85241079" +checksum = "53fafe42957e11d98e354a66b6bd70aeea00faf2f62dd11164188224a507c840" dependencies = [ "bstr", "gix-config-value", @@ -981,16 +980,16 @@ dependencies = [ "smallvec", "thiserror", "unicode-bom", - "winnow 0.5.40", + "winnow", ] [[package]] name = "gix-config-value" -version = "0.14.6" +version = "0.14.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fbd06203b1a9b33a78c88252a625031b094d9e1b647260070c25b09910c0a804" +checksum = "b328997d74dd15dc71b2773b162cb4af9a25c424105e4876e6d0686ab41c383e" dependencies = [ - "bitflags 2.5.0", + "bitflags", "bstr", "gix-path", "libc", @@ -999,9 +998,9 @@ dependencies = [ [[package]] name = "gix-date" -version = "0.8.6" +version = "0.8.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "367ee9093b0c2b04fd04c5c7c8b6a1082713534eab537597ae343663a518fa99" +checksum = "9eed6931f21491ee0aeb922751bd7ec97b4b2fe8fbfedcb678e2a2dce5f3b8c0" dependencies = [ "bstr", "itoa", @@ -1011,9 +1010,9 @@ dependencies = [ [[package]] name = "gix-diff" -version = "0.39.1" +version = "0.44.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fd6a0454f8c42d686f17e7f084057c717c082b7dbb8209729e4e8f26749eb93a" +checksum = "1996d5c8a305b59709467d80617c9fde48d9d75fd1f4179ea970912630886c9d" dependencies = [ "bstr", "gix-hash", @@ -1023,12 +1022,13 @@ dependencies = [ [[package]] name = "gix-discover" -version = "0.28.1" +version = "0.32.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b8d7b2896edc3d899d28a646ccc6df729827a6600e546570b2783466404a42d6" +checksum = "fc27c699b63da66b50d50c00668bc0b7e90c3a382ef302865e891559935f3dbf" dependencies = [ "bstr", "dunce", + "gix-fs", "gix-hash", "gix-path", "gix-ref", @@ -1038,14 +1038,15 @@ dependencies = [ [[package]] name = "gix-features" -version = "0.37.2" +version = "0.38.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d50270e8dcc665f30ba0735b17984b9535bdf1e646c76e638e007846164d57af" +checksum = "ac7045ac9fe5f9c727f38799d002a7ed3583cd777e3322a7c4b43e3cf437dc69" dependencies = [ "crc32fast", "flate2", "gix-hash", "gix-trace", + "gix-utils", "libc", "once_cell", "prodash", @@ -1056,20 +1057,22 @@ dependencies = [ [[package]] name = "gix-fs" -version = "0.9.1" +version = "0.11.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7555c23a005537434bbfcb8939694e18cad42602961d0de617f8477cc2adecdd" +checksum = "6adf99c27cdf17b1c4d77680c917e0d94d8783d4e1c73d3be0d1d63107163d7a" dependencies = [ + "fastrand", "gix-features", + "gix-utils", ] [[package]] name = "gix-glob" -version = "0.15.1" +version = "0.16.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ae6232f18b262770e343dcdd461c0011c9b9ae27f0c805e115012aa2b902c1b8" +checksum = "fa7df15afa265cc8abe92813cd354d522f1ac06b29ec6dfa163ad320575cb447" dependencies = [ - "bitflags 2.5.0", + "bitflags", "bstr", "gix-features", "gix-path", @@ -1098,14 +1101,14 @@ dependencies = [ [[package]] name = "gix-index" -version = "0.28.2" +version = "0.33.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9e50e63df6c8d4137f7fb882f27643b3a9756c468a1a2cdbe1ce443010ca8778" +checksum = "9a9a44eb55bd84bb48f8a44980e951968ced21e171b22d115d1cdcef82a7d73f" dependencies = [ - "bitflags 2.5.0", + "bitflags", "bstr", - "btoi", "filetime", + "fnv", "gix-bitmap", "gix-features", "gix-fs", @@ -1113,6 +1116,9 @@ dependencies = [ "gix-lock", "gix-object", "gix-traverse", + "gix-utils", + "gix-validate", + "hashbrown 0.14.5", "itoa", "libc", "memmap2", @@ -1123,9 +1129,9 @@ dependencies = [ [[package]] name = "gix-lock" -version = "12.0.1" +version = "14.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f40a439397f1e230b54cf85d52af87e5ea44cc1e7748379785d3f6d03d802b00" +checksum = "e3bc7fe297f1f4614774989c00ec8b1add59571dc9b024b4c00acb7dedd4e19d" dependencies = [ "gix-tempfile", "gix-utils", @@ -1140,37 +1146,38 @@ checksum = "999ce923619f88194171a67fb3e6d613653b8d4d6078b529b15a765da0edcc17" dependencies = [ "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] name = "gix-object" -version = "0.40.1" +version = "0.42.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0c89402e8faa41b49fde348665a8f38589e461036475af43b6b70615a6a313a2" +checksum = "25da2f46b4e7c2fa7b413ce4dffb87f69eaf89c2057e386491f4c55cadbfe386" dependencies = [ "bstr", - "btoi", "gix-actor", "gix-date", "gix-features", "gix-hash", + "gix-utils", "gix-validate", "itoa", "smallvec", "thiserror", - "winnow 0.5.40", + "winnow", ] [[package]] name = "gix-odb" -version = "0.56.1" +version = "0.61.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "46ae6da873de41c6c2b73570e82c571b69df5154dcd8f46dfafc6687767c33b1" +checksum = "20d384fe541d93d8a3bb7d5d5ef210780d6df4f50c4e684ccba32665a5e3bc9b" dependencies = [ "arc-swap", "gix-date", "gix-features", + "gix-fs", "gix-hash", "gix-object", "gix-pack", @@ -1183,9 +1190,9 @@ dependencies = [ [[package]] name = "gix-pack" -version = "0.46.1" +version = "0.51.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "782b4d42790a14072d5c400deda9851f5765f50fe72bca6dece0da1cd6f05a9a" +checksum = "3e0594491fffe55df94ba1c111a6566b7f56b3f8d2e1efc750e77d572f5f5229" dependencies = [ "clru", "gix-chunk", @@ -1194,18 +1201,16 @@ dependencies = [ "gix-hashtable", "gix-object", "gix-path", - "gix-tempfile", "memmap2", - "parking_lot", "smallvec", "thiserror", ] [[package]] name = "gix-path" -version = "0.10.7" +version = "0.10.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "23623cf0f475691a6d943f898c4d0b89f5c1a2a64d0f92bce0e0322ee6528783" +checksum = "8d23d5bbda31344d8abc8de7c075b3cf26e5873feba7c4a15d916bce67382bd9" dependencies = [ "bstr", "gix-trace", @@ -1227,9 +1232,9 @@ dependencies = [ [[package]] name = "gix-ref" -version = "0.40.1" +version = "0.44.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "64d9bd1984638d8f3511a2fcbe84fcedb8a5b5d64df677353620572383f42649" +checksum = "3394a2997e5bc6b22ebc1e1a87b41eeefbcfcff3dbfa7c4bd73cb0ac8f1f3e2e" dependencies = [ "gix-actor", "gix-date", @@ -1240,17 +1245,18 @@ dependencies = [ "gix-object", "gix-path", "gix-tempfile", + "gix-utils", "gix-validate", "memmap2", "thiserror", - "winnow 0.5.40", + "winnow", ] [[package]] name = "gix-refspec" -version = "0.21.1" +version = "0.23.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "be219df5092c1735abb2a53eccdf775e945eea6986ee1b6e7a5896dccc0be704" +checksum = "6868f8cd2e62555d1f7c78b784bece43ace40dd2a462daf3b588d5416e603f37" dependencies = [ "bstr", "gix-hash", @@ -1262,9 +1268,9 @@ dependencies = [ [[package]] name = "gix-revision" -version = "0.25.1" +version = "0.27.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "aa78e1df3633bc937d4db15f8dca2abdb1300ca971c0fabcf9fa97e38cf4cd9f" +checksum = "01b13e43c2118c4b0537ddac7d0821ae0dfa90b7b8dbf20c711e153fb749adce" dependencies = [ "bstr", "gix-date", @@ -1278,9 +1284,9 @@ dependencies = [ [[package]] name = "gix-revwalk" -version = "0.11.1" +version = "0.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "702de5fe5c2bbdde80219f3a8b9723eb927466e7ecd187cfd1b45d986408e45f" +checksum = "1b030ccaab71af141f537e0225f19b9e74f25fefdba0372246b844491cab43e0" dependencies = [ "gix-commitgraph", "gix-date", @@ -1293,11 +1299,11 @@ dependencies = [ [[package]] name = "gix-sec" -version = "0.10.6" +version = "0.10.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fddc27984a643b20dd03e97790555804f98cf07404e0e552c0ad8133266a79a1" +checksum = "1547d26fa5693a7f34f05b4a3b59a90890972922172653bcb891ab3f09f436df" dependencies = [ - "bitflags 2.5.0", + "bitflags", "gix-path", "libc", "windows-sys 0.52.0", @@ -1305,9 +1311,9 @@ dependencies = [ [[package]] name = "gix-tempfile" -version = "12.0.1" +version = "14.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a8ef376d718b1f5f119b458e21b00fbf576bc9d4e26f8f383d29f5ffe3ba3eaa" +checksum = "006acf5a613e0b5cf095d8e4b3f48c12a60d9062aa2b2dd105afaf8344a5600c" dependencies = [ "gix-fs", "libc", @@ -1326,10 +1332,11 @@ checksum = "f924267408915fddcd558e3f37295cc7d6a3e50f8bd8b606cee0808c3915157e" [[package]] name = "gix-traverse" -version = "0.36.2" +version = "0.39.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "65109e445ba7a409b48f34f570a4d7db72eade1dc1bcff81990a490e86c07161" +checksum = "e499a18c511e71cf4a20413b743b9f5bcf64b3d9e81e9c3c6cd399eae55a8840" dependencies = [ + "bitflags", "gix-commitgraph", "gix-date", "gix-hash", @@ -1342,9 +1349,9 @@ dependencies = [ [[package]] name = "gix-url" -version = "0.26.1" +version = "0.27.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8f0f17cceb7552a231d1fec690bc2740c346554e3be6f5d2c41dfa809594dc44" +checksum = "e2eb9b35bba92ea8f0b5ab406fad3cf6b87f7929aa677ff10aa042c6da621156" dependencies = [ "bstr", "gix-features", @@ -1405,6 +1412,15 @@ dependencies = [ "allocator-api2", ] +[[package]] +name = "hashlink" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ba4ff7128dee98c7dc9794b6a411377e1404dba1c97deb8d1a55297bd25d8af" +dependencies = [ + "hashbrown 0.14.5", +] + [[package]] name = "heapless" version = "0.8.0" @@ -1522,9 +1538,9 @@ dependencies = [ [[package]] name = "indexmap" -version = "2.2.6" +version = "2.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "168fb715dda47215e360912c096649d23d58bf392ac62f73919e831745e40f26" +checksum = "93ead53efc7ea8ed3cfb0c79fc8023fbb782a5432b52830b6518941cebe6505c" dependencies = [ "equivalent", "hashbrown 0.14.5", @@ -1533,9 +1549,9 @@ dependencies = [ [[package]] name = "is_terminal_polyfill" -version = "1.70.0" +version = "1.70.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f8478577c03552c21db0e2724ffb8986a5ce7af88107e6be5d2ee6e158c12800" +checksum = "7943c866cc5cd64cbc25b2e01621d07fa8eb2a1a23160ee81ce38704e97b8ecf" [[package]] name = "itertools" @@ -1546,6 +1562,15 @@ dependencies = [ "either", ] +[[package]] +name = "itertools" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "413ee7dfc52ee1a4949ceeb7dbc8a33f2d6c088194d9f922fb8318faf1f01186" +dependencies = [ + "either", +] + [[package]] name = "itoa" version = "1.0.11" @@ -1554,9 +1579,9 @@ checksum = "49f1f14873335454500d59611f1cf4a4b0f786f9ac11f4312a78e4cf2566695b" [[package]] name = "js-sys" -version = "0.3.69" +version = "0.3.70" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "29c15563dc2726973df627357ce0c9ddddbea194836909d655df6a75d2cf296d" +checksum = "1868808506b929d7b0cfa8f75951347aa71bb21144b7791bae35d9bccfcfe37a" dependencies = [ "wasm-bindgen", ] @@ -1580,9 +1605,9 @@ checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe" [[package]] name = "libc" -version = "0.2.155" +version = "0.2.156" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "97b3888a4aecf77e811145cadf6eef5901f4782c53886191b2f693f24761847c" +checksum = "a5f43f184355eefb8d17fc948dbecf6c13be3c141f20d834ae842193a448c72a" [[package]] name = "libredox" @@ -1590,8 +1615,20 @@ version = "0.1.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c0ff37bd590ca25063e35af745c343cb7a0271906fb7b37e4813e8f79f00268d" dependencies = [ - "bitflags 2.5.0", + "bitflags", "libc", + "redox_syscall", +] + +[[package]] +name = "libsqlite3-sys" +version = "0.30.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2e99fb7a497b1e3339bc746195567ed8d3e24945ecd636e3619d20b9de9e9149" +dependencies = [ + "cc", + "pkg-config", + "vcpkg", ] [[package]] @@ -1618,15 +1655,15 @@ dependencies = [ [[package]] name = "log" -version = "0.4.21" +version = "0.4.22" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "90ed8c1e510134f979dbc4f070f87d4313098b704861a105fe34231c70a3901c" +checksum = "a7a70ba024b9dc04c27ea2f0c0548feb474ec5c54bba33a7f72f873a39d07b24" [[package]] name = "lru" -version = "0.12.3" +version = "0.12.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d3262e75e648fce39813cb56ac41f3c3e3f65217ebf3844d818d1f9398cfb0dc" +checksum = "37ee39891760e7d94734f6f63fedc29a2e4a152f836120753a72503f09fcf904" dependencies = [ "hashbrown 0.14.5", ] @@ -1682,6 +1719,18 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "mio" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "80e04d1dcff3aae0704555fe5fee3bcfaf3d1fdf8a7e521d5b9d2b42acb52cec" +dependencies = [ + "hermit-abi", + "libc", + "wasi", + "windows-sys 0.52.0", +] + [[package]] name = "nom" version = "7.1.3" @@ -1717,16 +1766,6 @@ dependencies = [ "autocfg", ] -[[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 = "num_threads" version = "0.1.7" @@ -1808,9 +1847,9 @@ checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8" dependencies = [ "cfg-if", "libc", - "redox_syscall 0.5.2", + "redox_syscall", "smallvec", - "windows-targets 0.52.5", + "windows-targets 0.52.6", ] [[package]] @@ -1833,9 +1872,9 @@ checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e" [[package]] name = "pest" -version = "2.7.10" +version = "2.7.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "560131c633294438da9f7c4b08189194b20946c8274c6b9e38881a7874dc8ee8" +checksum = "cd53dff83f26735fdc1ca837098ccf133605d794cdae66acfc2bfac3ec809d95" dependencies = [ "memchr", "thiserror", @@ -1844,9 +1883,9 @@ dependencies = [ [[package]] name = "pest_derive" -version = "2.7.10" +version = "2.7.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "26293c9193fbca7b1a3bf9b79dc1e388e927e6cacaa78b4a3ab705a1d3d41459" +checksum = "2a548d2beca6773b1c244554d36fcf8548a8a58e74156968211567250e48e49a" dependencies = [ "pest", "pest_generator", @@ -1854,22 +1893,22 @@ dependencies = [ [[package]] name = "pest_generator" -version = "2.7.10" +version = "2.7.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3ec22af7d3fb470a85dd2ca96b7c577a1eb4ef6f1683a9fe9a8c16e136c04687" +checksum = "3c93a82e8d145725dcbaf44e5ea887c8a869efdcc28706df2d08c69e17077183" dependencies = [ "pest", "pest_meta", "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] name = "pest_meta" -version = "2.7.10" +version = "2.7.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d7a240022f37c361ec1878d646fc5b7d7c4d28d5946e1a80ad5a7a4f4ca0bdcd" +checksum = "a941429fea7e08bedec25e4f6785b6ffaacc6b755da98df5ef3e7dcf4a124c4f" dependencies = [ "once_cell", "pest", @@ -1888,6 +1927,12 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" +[[package]] +name = "pkg-config" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d231b230927b5e4ad203db57bbcbee2802f6bce620b1e4a9024a07d94e2907ec" + [[package]] name = "powerfmt" version = "0.2.0" @@ -1934,11 +1979,11 @@ version = "0.26.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f44c9e68fd46eda15c646fbb85e1040b657a58cdc8c98db1d97a55930d991eef" dependencies = [ - "bitflags 2.5.0", + "bitflags", "cassowary", "compact_str", "crossterm", - "itertools", + "itertools 0.12.1", "lru", "paste", "serde", @@ -1951,20 +1996,11 @@ dependencies = [ [[package]] name = "redox_syscall" -version = "0.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4722d768eff46b75989dd134e5c353f0d6296e5aaa3132e776cbdb56be7731aa" -dependencies = [ - "bitflags 1.3.2", -] - -[[package]] -name = "redox_syscall" -version = "0.5.2" +version = "0.5.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c82cf8cff14456045f55ec4241383baeff27af886adb72ffb2162f99911de0fd" +checksum = "2a908a6e00f1fdd0dfd9c0eb08ce85126f6d8bbda50017e74bc4a4b7d4a926a4" dependencies = [ - "bitflags 2.5.0", + "bitflags", ] [[package]] @@ -1980,9 +2016,9 @@ dependencies = [ [[package]] name = "regex" -version = "1.10.5" +version = "1.10.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b91213439dad192326a0d7c6ee3955910425f441d7038e0d6933b0aec5c4517f" +checksum = "4219d74c6b67a3654a9fbebc4b419e22126d13d2f3c4a07ee0cb61ff79a79619" dependencies = [ "aho-corasick", "memchr", @@ -2029,11 +2065,25 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b91f7eff05f748767f183df4320a63d6936e9c6107d97c9e6bdd9784f4289c94" dependencies = [ "base64 0.21.7", - "bitflags 2.5.0", + "bitflags", "serde", "serde_derive", ] +[[package]] +name = "rusqlite" +version = "0.32.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7753b721174eb8ff87a9a0e799e2d7bc3749323e773db92e0984debb00019d6e" +dependencies = [ + "bitflags", + "fallible-iterator", + "fallible-streaming-iterator", + "hashlink", + "libsqlite3-sys", + "smallvec", +] + [[package]] name = "rust-ini" version = "0.19.0" @@ -2056,7 +2106,7 @@ version = "0.38.34" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "70dc5ec042f7a43c4a73241207cecc9873a06d45debb38b329f8541d85c2730f" dependencies = [ - "bitflags 2.5.0", + "bitflags", "errno", "libc", "linux-raw-sys", @@ -2101,31 +2151,32 @@ dependencies = [ [[package]] name = "serde" -version = "1.0.203" +version = "1.0.208" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7253ab4de971e72fb7be983802300c30b5a7f0c2e56fab8abfc6a214307c0094" +checksum = "cff085d2cb684faa248efb494c39b68e522822ac0de72ccf08109abde717cfb2" dependencies = [ "serde_derive", ] [[package]] name = "serde_derive" -version = "1.0.203" +version = "1.0.208" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "500cbc0ebeb6f46627f50f3f5811ccf6bf00643be300b4c3eabc0ef55dc5b5ba" +checksum = "24008e81ff7613ed8e5ba0cfaf24e2c2f1e5b8a0495711e44fcd4882fca62bcf" dependencies = [ "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] name = "serde_json" -version = "1.0.117" +version = "1.0.125" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "455182ea6142b14f93f4bc5320a2b31c1f266b66a4a5c858b013302a5d8cbfc3" +checksum = "83c8e735a073ccf5be70aa8066aa984eaf2fa000db6c8d0100ae605b366d31ed" dependencies = [ "itoa", + "memchr", "ryu", "serde", ] @@ -2141,15 +2192,15 @@ dependencies = [ [[package]] name = "serde_with" -version = "3.8.1" +version = "3.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0ad483d2ab0149d5a5ebcd9972a3852711e0153d863bf5a5d0391d28883c4a20" +checksum = "69cecfa94848272156ea67b2b1a53f20fc7bc638c4a46d2f8abde08f05f4b857" dependencies = [ "base64 0.22.1", "chrono", "hex", "indexmap 1.9.3", - "indexmap 2.2.6", + "indexmap 2.4.0", "serde", "serde_derive", "serde_json", @@ -2159,21 +2210,21 @@ dependencies = [ [[package]] name = "serde_with_macros" -version = "3.8.1" +version = "3.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "65569b702f41443e8bc8bbb1c5779bd0450bbe723b56198980e80ec45780bce2" +checksum = "a8fee4991ef4f274617a51ad4af30519438dacb2f56ac773b08a1922ff743350" dependencies = [ "darling", "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] name = "sha1_smol" -version = "1.0.0" +version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ae1a47186c03a32177042e55dbc5fd5aee900b8e0069a8d70fba96a9375cd012" +checksum = "bbfa15b3dddfee50a0fff136974b3e1bde555604ba463834a7eb7deb6417705d" [[package]] name = "sha2" @@ -2195,6 +2246,12 @@ dependencies = [ "lazy_static", ] +[[package]] +name = "shlex" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" + [[package]] name = "signal-hook" version = "0.3.17" @@ -2207,12 +2264,12 @@ dependencies = [ [[package]] name = "signal-hook-mio" -version = "0.2.3" +version = "0.2.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "29ad2e15f37ec9a6cc544097b78a1ec90001e9f71b81338ca39f430adaca99af" +checksum = "34db1a06d485c9142248b7a054f034b349b212551f3dfd19c94d45a754a217cd" dependencies = [ "libc", - "mio", + "mio 0.8.11", "signal-hook", ] @@ -2233,9 +2290,9 @@ checksum = "f27f6278552951f1f2b8cf9da965d10969b2efdea95a6ec47987ab46edfe263a" [[package]] name = "similar" -version = "2.5.0" +version = "2.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fa42c91313f1d05da9b26f267f931cf178d4aba455b4c4622dd7355eb80c6640" +checksum = "1de1d4f81173b03af4c0cbed3c898f6bff5b870e4a7f5d6f4057d62a7a4b686e" [[package]] name = "slab" @@ -2264,12 +2321,12 @@ dependencies = [ [[package]] name = "stability" -version = "0.2.0" +version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2ff9eaf853dec4c8802325d8b6d3dffa86cc707fd7a1a4cdbf416e13b061787a" +checksum = "d904e7009df136af5297832a3ace3370cd14ff1546a232f4f185036c2736fcac" dependencies = [ "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] @@ -2301,9 +2358,9 @@ checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f" [[package]] name = "strum" -version = "0.26.2" +version = "0.26.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5d8cec3501a5194c432b2b7976db6b7d10ec95c253208b45f83f7136aa985e29" +checksum = "8fec0f0aef304996cf250b31b5a10dee7980c85da9d759361292b8bca5a18f06" dependencies = [ "strum_macros", ] @@ -2318,7 +2375,7 @@ dependencies = [ "proc-macro2", "quote", "rustversion", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] @@ -2334,9 +2391,9 @@ dependencies = [ [[package]] name = "syn" -version = "2.0.67" +version = "2.0.74" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ff8655ed1d86f3af4ee3fd3263786bc14245ad17c4c7e85ba7187fb3ae028c90" +checksum = "1fceb41e3d546d0bd83421d3409b1460cc7444cd389341a4c880fe7a042cb3d7" dependencies = [ "proc-macro2", "quote", @@ -2345,14 +2402,15 @@ dependencies = [ [[package]] name = "tempfile" -version = "3.10.1" +version = "3.12.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "85b77fafb263dd9d05cbeac119526425676db3784113aa9295c88498cbf8bff1" +checksum = "04cbcdd0c794ebb0d4cf35e88edd2f7d2c4c3e9a5a6dab322839b321c6a87a64" dependencies = [ "cfg-if", "fastrand", + "once_cell", "rustix", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -2367,22 +2425,22 @@ dependencies = [ [[package]] name = "thiserror" -version = "1.0.61" +version = "1.0.63" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c546c80d6be4bc6a00c0f01730c08df82eaa7a7a61f11d656526506112cc1709" +checksum = "c0342370b38b6a11b6cc11d6a805569958d54cfa061a29969c3b5ce2ea405724" dependencies = [ "thiserror-impl", ] [[package]] name = "thiserror-impl" -version = "1.0.61" +version = "1.0.63" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "46c3384250002a6d5af4d114f2845d37b57521033f30d5c3f46c4d70e1197533" +checksum = "a4558b58466b9ad7ca0f102865eccc95938dca1a74a856f2b57b6629050da261" dependencies = [ "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] @@ -2439,9 +2497,9 @@ dependencies = [ [[package]] name = "tinyvec" -version = "1.6.0" +version = "1.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "87cc5ceb3875bb20c2890005a4e226a4651264a5c75edb2421b52861a0a0cb50" +checksum = "445e881f4f6d382d5f27c034e25eb92edd7c784ceab92a0937db7f2e9471b938" dependencies = [ "tinyvec_macros", ] @@ -2454,32 +2512,31 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" [[package]] name = "tokio" -version = "1.38.0" +version = "1.39.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ba4f4a02a7a80d6f274636f0aa95c7e383b912d41fe721a31f29e29698585a4a" +checksum = "daa4fb1bc778bd6f04cbfc4bb2d06a7396a8f299dc33ea1900cedaa316f467b1" dependencies = [ "backtrace", "bytes", "libc", - "mio", - "num_cpus", + "mio 1.0.2", "parking_lot", "pin-project-lite", "signal-hook-registry", "socket2", "tokio-macros", - "windows-sys 0.48.0", + "windows-sys 0.52.0", ] [[package]] name = "tokio-macros" -version = "2.3.0" +version = "2.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f5ae998a069d4b5aba8ee9dad856af7d520c3699e6159b185c2acd48155d39a" +checksum = "693d596312e88961bc67d7f1f97af8a70227d9f90c31bba5806eec004978d752" dependencies = [ "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] @@ -2522,11 +2579,11 @@ version = "0.22.20" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "583c44c02ad26b0c3f3066fe629275e50627026c51ac2e595cca4c230ce1ce1d" dependencies = [ - "indexmap 2.2.6", + "indexmap 2.4.0", "serde", "serde_spanned", "toml_datetime", - "winnow 0.6.18", + "winnow", ] [[package]] @@ -2548,7 +2605,7 @@ checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" dependencies = [ "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] [[package]] @@ -2676,11 +2733,12 @@ checksum = "d4c87d22b6e3f4a18d4d40ef354e97c90fcb14dd91d7dc0aa9d8a1172ebf7202" [[package]] name = "unicode-truncate" -version = "1.0.0" +version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5a5fbabedabe362c618c714dbefda9927b5afc8e2a8102f47f081089a9019226" +checksum = "b3644627a5af5fa321c95b9b235a72fd24cd29c648c2c379431e6628655627bf" dependencies = [ - "itertools", + "itertools 0.13.0", + "unicode-segmentation", "unicode-width", ] @@ -2722,11 +2780,17 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "830b7e5d4d90034032940e4ace0d9a9a057e7a45cd94e6c007832e39edb82f6d" +[[package]] +name = "vcpkg" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426" + [[package]] name = "vergen" -version = "8.3.1" +version = "8.3.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e27d6bdd219887a9eadd19e1c34f32e47fa332301184935c6d9bca26f3cca525" +checksum = "2990d9ea5967266ea0ccf413a4aa5c42a93dbcfda9cb49a97de6931726b12566" dependencies = [ "anyhow", "cargo_metadata", @@ -2739,9 +2803,9 @@ dependencies = [ [[package]] name = "version_check" -version = "0.9.4" +version = "0.9.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f" +checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a" [[package]] name = "viddy" @@ -2769,6 +2833,7 @@ dependencies = [ "log", "pretty_assertions", "ratatui", + "rusqlite", "serde", "serde_json", "serde_with", @@ -2826,34 +2891,35 @@ checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" [[package]] name = "wasm-bindgen" -version = "0.2.92" +version = "0.2.93" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4be2531df63900aeb2bca0daaaddec08491ee64ceecbee5076636a3b026795a8" +checksum = "a82edfc16a6c469f5f44dc7b571814045d60404b55a0ee849f9bcfa2e63dd9b5" dependencies = [ "cfg-if", + "once_cell", "wasm-bindgen-macro", ] [[package]] name = "wasm-bindgen-backend" -version = "0.2.92" +version = "0.2.93" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "614d787b966d3989fa7bb98a654e369c762374fd3213d212cfc0251257e747da" +checksum = "9de396da306523044d3302746f1208fa71d7532227f15e347e2d93e4145dd77b" dependencies = [ "bumpalo", "log", "once_cell", "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", "wasm-bindgen-shared", ] [[package]] name = "wasm-bindgen-macro" -version = "0.2.92" +version = "0.2.93" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a1f8823de937b71b9460c0c34e25f3da88250760bec0ebac694b49997550d726" +checksum = "585c4c91a46b072c92e908d99cb1dcdf95c5218eeb6f3bf1efa991ee7a68cccf" dependencies = [ "quote", "wasm-bindgen-macro-support", @@ -2861,22 +2927,22 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro-support" -version = "0.2.92" +version = "0.2.93" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e94f17b526d0a461a191c78ea52bbce64071ed5c04c9ffe424dcb38f74171bb7" +checksum = "afc340c74d9005395cf9dd098506f7f44e38f2b4a21c6aaacf9a105ea5e1e836" dependencies = [ "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", "wasm-bindgen-backend", "wasm-bindgen-shared", ] [[package]] name = "wasm-bindgen-shared" -version = "0.2.92" +version = "0.2.93" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "af190c94f2773fdb3729c55b007a722abb5384da03bc0986df4c289bf5567e96" +checksum = "c62a0a307cb4a311d3a07867860911ca130c3494e8c2719593806c08bc5d0484" [[package]] name = "winapi" @@ -2896,11 +2962,11 @@ checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" [[package]] name = "winapi-util" -version = "0.1.8" +version = "0.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4d4cc384e1e73b93bafa6fb4f1df8c41695c8a91cf9c4c64358067d15a7b6c6b" +checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb" dependencies = [ - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -2915,7 +2981,7 @@ version = "0.52.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "33ab640c8d7e35bf8ba19b884ba838ceb4fba93a4e8c65a9059d08afcfc683d9" dependencies = [ - "windows-targets 0.52.5", + "windows-targets 0.52.6", ] [[package]] @@ -2933,7 +2999,16 @@ version = "0.52.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "282be5f36a8ce781fad8c8ae18fa3f9beff57ec1b52cb3de0789201425d9a33d" dependencies = [ - "windows-targets 0.52.5", + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-sys" +version = "0.59.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e38bc4d79ed67fd075bcc251a1c39b32a1776bbe92e5bef1f0bf1f8c531853b" +dependencies = [ + "windows-targets 0.52.6", ] [[package]] @@ -2953,18 +3028,18 @@ dependencies = [ [[package]] name = "windows-targets" -version = "0.52.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6f0713a46559409d202e70e28227288446bf7841d3211583a4b53e3f6d96e7eb" +checksum = "9b724f72796e036ab90c1021d4780d4d3d648aca59e491e6b98e725b84e99973" dependencies = [ - "windows_aarch64_gnullvm 0.52.5", - "windows_aarch64_msvc 0.52.5", - "windows_i686_gnu 0.52.5", + "windows_aarch64_gnullvm 0.52.6", + "windows_aarch64_msvc 0.52.6", + "windows_i686_gnu 0.52.6", "windows_i686_gnullvm", - "windows_i686_msvc 0.52.5", - "windows_x86_64_gnu 0.52.5", - "windows_x86_64_gnullvm 0.52.5", - "windows_x86_64_msvc 0.52.5", + "windows_i686_msvc 0.52.6", + "windows_x86_64_gnu 0.52.6", + "windows_x86_64_gnullvm 0.52.6", + "windows_x86_64_msvc 0.52.6", ] [[package]] @@ -2975,9 +3050,9 @@ checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8" [[package]] name = "windows_aarch64_gnullvm" -version = "0.52.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7088eed71e8b8dda258ecc8bac5fb1153c5cffaf2578fc8ff5d61e23578d3263" +checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3" [[package]] name = "windows_aarch64_msvc" @@ -2987,9 +3062,9 @@ checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc" [[package]] name = "windows_aarch64_msvc" -version = "0.52.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9985fd1504e250c615ca5f281c3f7a6da76213ebd5ccc9561496568a2752afb6" +checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469" [[package]] name = "windows_i686_gnu" @@ -2999,15 +3074,15 @@ checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e" [[package]] name = "windows_i686_gnu" -version = "0.52.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "88ba073cf16d5372720ec942a8ccbf61626074c6d4dd2e745299726ce8b89670" +checksum = "8e9b5ad5ab802e97eb8e295ac6720e509ee4c243f69d781394014ebfe8bbfa0b" [[package]] name = "windows_i686_gnullvm" -version = "0.52.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "87f4261229030a858f36b459e748ae97545d6f1ec60e5e0d6a3d32e0dc232ee9" +checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66" [[package]] name = "windows_i686_msvc" @@ -3017,9 +3092,9 @@ checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406" [[package]] name = "windows_i686_msvc" -version = "0.52.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "db3c2bf3d13d5b658be73463284eaf12830ac9a26a90c717b7f771dfe97487bf" +checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66" [[package]] name = "windows_x86_64_gnu" @@ -3029,9 +3104,9 @@ checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e" [[package]] name = "windows_x86_64_gnu" -version = "0.52.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4e4246f76bdeff09eb48875a0fd3e2af6aada79d409d33011886d3e1581517d9" +checksum = "147a5c80aabfbf0c7d901cb5895d1de30ef2907eb21fbbab29ca94c5b08b1a78" [[package]] name = "windows_x86_64_gnullvm" @@ -3041,9 +3116,9 @@ checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc" [[package]] name = "windows_x86_64_gnullvm" -version = "0.52.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "852298e482cd67c356ddd9570386e2862b5673c85bd5f88df9ab6802b334c596" +checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d" [[package]] name = "windows_x86_64_msvc" @@ -3053,18 +3128,9 @@ checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538" [[package]] name = "windows_x86_64_msvc" -version = "0.52.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bec47e5bfd1bff0eeaf6d8b485cc1074891a197ab4225d504cb7a1ab88b02bf0" - -[[package]] -name = "winnow" -version = "0.5.40" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f593a95398737aeed53e489c785df13f3618e41dbcd6718c6addbf1395aa6876" -dependencies = [ - "memchr", -] +checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" [[package]] name = "winnow" @@ -3092,20 +3158,20 @@ checksum = "09041cd90cf85f7f8b2df60c646f853b7f535ce68f85244eb6731cf89fa498ec" [[package]] name = "zerocopy" -version = "0.7.34" +version = "0.7.35" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ae87e3fcd617500e5d106f0380cf7b77f3c6092aae37191433159dda23cfb087" +checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0" dependencies = [ "zerocopy-derive", ] [[package]] name = "zerocopy-derive" -version = "0.7.34" +version = "0.7.35" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "15e934569e47891f7d9411f1a451d947a60e000ab3bd24fbb970f000387d1b3b" +checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e" dependencies = [ "proc-macro2", "quote", - "syn 2.0.67", + "syn 2.0.74", ] diff --git a/Cargo.toml b/Cargo.toml index 906db10..939ed30 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -32,6 +32,7 @@ libc = "0.2.155" log = "0.4.21" pretty_assertions = "1.4.0" ratatui = { version = "0.26.3", features = ["serde", "macros"] } +rusqlite = { version = "0.32.1", features = ["bundled"] } serde = { version = "1.0.203", features = ["derive"] } serde_json = "1.0.117" serde_with = { version = "3.8.1", features = ["chrono_0_4"] } diff --git a/src/main.rs b/src/main.rs index 9695786..1c49eff 100644 --- a/src/main.rs +++ b/src/main.rs @@ -38,7 +38,7 @@ async fn tokio_main() -> Result<()> { let args = Cli::parse(); let interval = Duration::from(args.interval); - let store = store::MemoryStore::new(); + let store = store::memory::MemoryStore::new(); let mut app = App::new(args, store)?; app.run().await?; diff --git a/src/runner.rs b/src/runner.rs index 301852a..f076930 100644 --- a/src/runner.rs +++ b/src/runner.rs @@ -10,7 +10,7 @@ use crate::{ action::Action, config::{Config, RuntimeConfig}, exec::exec, - store::{MemoryStore, Record, Store}, + store::{Record, Store}, types::ExecutionId, }; diff --git a/src/store.rs b/src/store.rs index cd40794..8d49bcd 100644 --- a/src/store.rs +++ b/src/store.rs @@ -1,3 +1,5 @@ +pub mod memory; + use std::{ collections::HashMap, sync::{Arc, RwLock}, @@ -24,50 +26,3 @@ pub struct Record { pub diff: Option<(u32, u32)>, pub previous_id: Option, } - -#[derive(Debug)] -struct MemoryStoreData { - records: HashMap, - latest_id: Option, -} - -#[derive(Clone, Debug)] -pub struct MemoryStore { - data: Arc>, -} - -impl MemoryStore { - pub fn new() -> Self { - Self { - data: Arc::new(RwLock::new(MemoryStoreData { - records: HashMap::new(), - latest_id: None, - })), - } - } -} - -impl Store for MemoryStore { - fn add_record(&mut self, record: Record) { - if let Ok(mut data) = self.data.write() { - data.latest_id = Some(record.id); - data.records.insert(record.id, record); - } - } - - fn get_record(&self, id: ExecutionId) -> Option { - if let Ok(data) = self.data.read() { - data.records.get(&id).cloned() - } else { - None - } - } - - fn get_latest_id(&self) -> Option { - if let Ok(data) = self.data.read() { - data.latest_id - } else { - None - } - } -} diff --git a/src/store/memory.rs b/src/store/memory.rs new file mode 100644 index 0000000..70b7d5d --- /dev/null +++ b/src/store/memory.rs @@ -0,0 +1,52 @@ +use std::collections::HashMap; +use std::sync::{Arc, RwLock}; + +use crate::types::ExecutionId; +use crate::store::{Record, Store}; + +#[derive(Debug)] +struct MemoryStoreData { + records: HashMap, + latest_id: Option, +} + +#[derive(Clone, Debug)] +pub struct MemoryStore { + data: Arc>, +} + +impl MemoryStore { + pub fn new() -> Self { + Self { + data: Arc::new(RwLock::new(MemoryStoreData { + records: HashMap::new(), + latest_id: None, + })), + } + } +} + +impl Store for MemoryStore { + fn add_record(&mut self, record: Record) { + if let Ok(mut data) = self.data.write() { + data.latest_id = Some(record.id); + data.records.insert(record.id, record); + } + } + + fn get_record(&self, id: ExecutionId) -> Option { + if let Ok(data) = self.data.read() { + data.records.get(&id).cloned() + } else { + None + } + } + + fn get_latest_id(&self) -> Option { + if let Ok(data) = self.data.read() { + data.latest_id + } else { + None + } + } +} From 48c93654519c9dd029e262d9fb6bbd5ab3e78a6d Mon Sep 17 00:00:00 2001 From: Takumasa Sakao Date: Fri, 16 Aug 2024 23:30:55 +0900 Subject: [PATCH 4/4] Implement save & lookback feature --- .config/config.json5 | 3 +- Cargo.lock | 67 +++++++++- Cargo.toml | 6 +- src/app.rs | 110 +++++++++++++--- src/cli.rs | 26 +++- src/components/execution_result.rs | 12 ++ src/components/home.rs | 12 +- src/components/interval.rs | 1 - src/components/status.rs | 44 ++++--- src/config.rs | 12 +- src/main.rs | 44 ++++++- src/runner.rs | 33 ++--- src/store.rs | 19 ++- src/store/memory.rs | 46 +++++-- src/store/sqlite.rs | 194 +++++++++++++++++++++++++++++ src/types.rs | 18 +++ 16 files changed, 562 insertions(+), 85 deletions(-) create mode 100644 src/store/sqlite.rs diff --git a/.config/config.json5 b/.config/config.json5 index 6883563..02ef7a6 100644 --- a/.config/config.json5 +++ b/.config/config.json5 @@ -55,7 +55,8 @@ "border": "rgb111", "title": "rgb111", "scrollbar": "rgb111", - "search_highlight": "black on yellow" + "search_highlight": "black on yellow", + "readonly": "bold yellow" } } } diff --git a/Cargo.lock b/Cargo.lock index 81a73eb..5e3f900 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -758,6 +758,12 @@ dependencies = [ "percent-encoding", ] +[[package]] +name = "fuchsia-cprng" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a06f77d526c1a601b7c4cdd98f54b5eaabffc14d5f2f0296febdc7f357c6d3ba" + [[package]] name = "futures" version = "0.3.30" @@ -1973,6 +1979,34 @@ dependencies = [ "proc-macro2", ] +[[package]] +name = "rand" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "552840b97013b1a26992c11eac34bdd778e464601a4c2054b5f0bff7c6761293" +dependencies = [ + "fuchsia-cprng", + "libc", + "rand_core 0.3.1", + "rdrand", + "winapi", +] + +[[package]] +name = "rand_core" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a6fdeb83b075e8266dcc8762c22776f6877a63111121f5f8c7411e5be7eed4b" +dependencies = [ + "rand_core 0.4.2", +] + +[[package]] +name = "rand_core" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c33a3c44ca05fa6f1807d8e6743f3824e8509beca625669633be0acbdf509dc" + [[package]] name = "ratatui" version = "0.26.3" @@ -1994,6 +2028,15 @@ dependencies = [ "unicode-width", ] +[[package]] +name = "rdrand" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "678054eb77286b51581ba43620cc911abf02758c91f93f479767aed0f90458b2" +dependencies = [ + "rand_core 0.3.1", +] + [[package]] name = "redox_syscall" version = "0.5.3" @@ -2058,6 +2101,15 @@ version = "0.8.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7a66a03ae7c801facd77a29370b4faec201768915ac14a721ba36f20bc9c209b" +[[package]] +name = "remove_dir_all" +version = "0.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3acd125665422973a33ac9d3dd2df85edad0f4ae9b00dafb1a05e43a9f5ef8e7" +dependencies = [ + "winapi", +] + [[package]] name = "ron" version = "0.8.1" @@ -2077,6 +2129,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7753b721174eb8ff87a9a0e799e2d7bc3749323e773db92e0984debb00019d6e" dependencies = [ "bitflags", + "chrono", "fallible-iterator", "fallible-streaming-iterator", "hashlink", @@ -2400,6 +2453,16 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "tempdir" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15f2b5fb00ccdf689e0149d1b1b3c03fead81c2b37735d812fa8bddbbf41b6d8" +dependencies = [ + "rand", + "remove_dir_all", +] + [[package]] name = "tempfile" version = "3.12.0" @@ -2809,7 +2872,7 @@ checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a" [[package]] name = "viddy" -version = "1.0.0-rc.0" +version = "1.0.0-rc.2" dependencies = [ "ansi-parser", "ansi-to-tui", @@ -2841,6 +2904,8 @@ dependencies = [ "similar", "strip-ansi-escapes", "strum", + "tempdir", + "tempfile", "tokio", "tokio-util", "toml", diff --git a/Cargo.toml b/Cargo.toml index 939ed30..09f9a07 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "viddy" -version = "1.0.0-rc.0" +version = "1.0.0-rc.2" edition = "2021" description = "A modern watch command" @@ -32,7 +32,7 @@ libc = "0.2.155" log = "0.4.21" pretty_assertions = "1.4.0" ratatui = { version = "0.26.3", features = ["serde", "macros"] } -rusqlite = { version = "0.32.1", features = ["bundled"] } +rusqlite = { version = "0.32.1", features = ["bundled", "chrono"] } serde = { version = "1.0.203", features = ["derive"] } serde_json = "1.0.117" serde_with = { version = "3.8.1", features = ["chrono_0_4"] } @@ -40,6 +40,8 @@ signal-hook = "0.3.17" similar = "2.5.0" strip-ansi-escapes = "0.2.0" strum = { version = "0.26.2", features = ["derive"] } +tempdir = "0.3.7" +tempfile = "3.12.0" tokio = { version = "1.38.0", features = ["full"] } tokio-util = "0.7.11" toml = "0.8.19" diff --git a/src/app.rs b/src/app.rs index 5b1e672..e8157ed 100644 --- a/src/app.rs +++ b/src/app.rs @@ -1,3 +1,4 @@ +use core::time; use std::sync::Arc; use anstyle::{Color, RgbColor, Style}; @@ -13,10 +14,21 @@ use tokio::{ use tracing_subscriber::field::debug; use crate::{ - action::{self, Action, DiffMode}, cli::Cli, components::{fps::FpsCounter, home::Home, Component}, config::{Config, RuntimeConfig}, diff::{diff_and_mark, diff_and_mark_delete}, mode::Mode, old_config::OldConfig, runner::{run_executor, run_executor_precise}, search::search_and_mark, store::Store, termtext, tui, types::ExecutionId + action::{self, Action, DiffMode}, + cli::Cli, + components::{fps::FpsCounter, home::Home, Component}, + config::{Config, RuntimeConfig}, + diff::{diff_and_mark, diff_and_mark_delete}, + mode::Mode, + old_config::OldConfig, + runner::{run_executor, run_executor_precise}, + search::search_and_mark, + store::{self, RuntimeConfig as StoreRuntimeConfig, Store}, + termtext, tui, + types::ExecutionId, }; -pub struct App { +pub struct App { pub config: Config, pub runtime_config: RuntimeConfig, pub tick_rate: f64, @@ -28,7 +40,6 @@ pub struct App { pub last_tick_key_events: Vec, pub timemachine_mode: bool, pub search_query: Option, - store: S, is_precise: bool, diff_mode: Option, is_suspend: Arc>, @@ -38,14 +49,39 @@ pub struct App { is_skip_empty_diffs: bool, showing_execution_id: Option, shell: Option<(String, Vec)>, + store: S, + read_only: bool, } -impl App { - pub fn new(cli: Cli, store: S) -> Result { - let runtime_config = RuntimeConfig { - interval: cli.interval, - command: cli.command, +impl App { + pub fn new(cli: Cli, mut store: S, read_only: bool) -> Result { + let runtime_config = if read_only { + let store_runtime_config = store.get_runtime_config()?.unwrap_or_default(); + + RuntimeConfig { + interval: Duration::from_std(humantime::parse_duration( + &store_runtime_config.interval, + )?)?, + command: store_runtime_config + .command + .split(' ') + .map(|s| s.to_string()) + .collect(), + } + } else { + let runtime_config = RuntimeConfig { + interval: cli.interval, + command: cli.command.clone(), + }; + + let interval = + humantime::format_duration(cli.interval.to_std().unwrap_or_default()).to_string(); + let command = cli.command.join(" "); + store.set_runtime_config(StoreRuntimeConfig { interval, command })?; + + runtime_config }; + let diff_mode = match (cli.is_diff, cli.is_deletion_diff) { (true, false) => Some(DiffMode::Add), (false, true) => Some(DiffMode::Delete), @@ -81,6 +117,7 @@ impl App { )) }; + let timemachine_mode = false; let home = Home::new( config.clone(), runtime_config.clone(), @@ -88,13 +125,14 @@ impl App { diff_mode, cli.is_bell, cli.is_no_title, + read_only, + timemachine_mode, ); let mut components: Vec> = vec![Box::new(home)]; if cli.is_debug { components.push(Box::new(FpsCounter::new())); } - log::debug!("{:?}", config.general); let default_skip_empty_diffs = config.general.skip_empty_diffs.unwrap_or_default(); let is_skip_empty_diffs = cli.is_skip_empty_diffs || default_skip_empty_diffs; @@ -104,12 +142,13 @@ impl App { frame_rate: 20.0, components, should_quit: false, + read_only, should_suspend: false, config, runtime_config, mode: Mode::All, last_tick_key_events: Vec::new(), - timemachine_mode: false, + timemachine_mode, search_query: None, is_precise: cli.is_precise, is_bell: cli.is_bell, @@ -130,7 +169,27 @@ impl App { pub async fn run(&mut self) -> Result<()> { let (action_tx, mut action_rx) = mpsc::unbounded_channel(); - let executor_handle = if self.is_precise { + let records = self.store.get_records()?; + for r in records { + action_tx.send(Action::StartExecution(r.id, r.start_time))?; + action_tx.send(Action::FinishExecution( + r.id, + r.end_time, + r.diff, + r.exit_code, + ))?; + } + if self.read_only { + action_tx.send(Action::SetTimemachineMode(true))?; + } + + let executor_handle = if self.read_only { + tokio::spawn(async move { + loop { + tokio::time::sleep(Duration::seconds(1).to_std().unwrap()).await; + } + }) + } else if self.is_precise { tokio::spawn(run_executor_precise( action_tx.clone(), self.store.clone(), @@ -262,7 +321,7 @@ impl App { action_tx.send(Action::UpdateHistoryResult(id, diff, exit_code))?; } - if !self.timemachine_mode { + if !self.timemachine_mode && !self.read_only { action_tx.send(Action::ShowExecution(id, id))?; } } @@ -274,7 +333,7 @@ impl App { Action::ShowExecution(id, end_id) => { let style = termtext::convert_to_anstyle(self.config.get_style("background")); - let record = self.store.get_record(id); + let record = self.store.get_record(id)?; let mut string = "".to_string(); if let Some(record) = record { action_tx.send(Action::SetClock(record.start_time))?; @@ -292,7 +351,7 @@ impl App { string = result.plain_text(); if let Some(diff_mode) = self.diff_mode { if let Some(previous_id) = record.previous_id { - let previous_record = self.store.get_record(previous_id); + let previous_record = self.store.get_record(previous_id)?; if let Some(previous_record) = previous_record { let previous_result = termtext::Converter::new(style) .convert(&previous_record.stdout); @@ -336,10 +395,9 @@ impl App { } Action::SetTimemachineMode(timemachine_mode) => { self.timemachine_mode = timemachine_mode; - if !timemachine_mode { - if let Some(latest_id) = self.store.get_latest_id() { - action_tx.send(Action::ShowExecution(latest_id, latest_id))?; - } + if let Some(latest_id) = self.store.get_latest_id()? { + log::debug!("Latest ID: {latest_id}"); + action_tx.send(Action::ShowExecution(latest_id, latest_id))?; } } Action::ExecuteSearch => { @@ -425,6 +483,12 @@ impl App { }; } } + + if executor_handle.is_finished() { + tui.stop()?; + break; + } + if self.should_suspend { tui.suspend()?; action_tx.send(Action::Resume)?; @@ -438,8 +502,14 @@ impl App { break; } } - executor_handle.abort(); tui.exit()?; - Ok(()) + + if !executor_handle.is_finished() { + log::debug!("Waiting for executor to finish"); + executor_handle.abort(); + return Ok(()); + } + + executor_handle.await? } } diff --git a/src/cli.rs b/src/cli.rs index 514a854..5b84dbb 100644 --- a/src/cli.rs +++ b/src/cli.rs @@ -84,7 +84,7 @@ pub struct Cli { )] pub is_bell: bool, - #[arg(value_name = "COMMAND", num_args(1..), required = true, allow_hyphen_values = true, help = "Command to run")] + #[arg(value_name = "COMMAND", num_args(0..), allow_hyphen_values = true, help = "Command to run")] pub command: Vec, #[arg( @@ -97,6 +97,30 @@ pub struct Cli { #[arg(long = "debug")] pub is_debug: bool, + + #[arg( + long = "save", + value_name = "FILE", + help = "Path to the backup file. If not provided, a temporary file will be created", + conflicts_with_all = ["disable_auto_save", "load"] + )] + pub save: Option, + + #[arg( + long = "disable_auto_save", + help = "Disable to save automatically", + conflicts_with_all = ["save", "load"] + )] + pub disable_auto_save: bool, + + #[arg( + long = "load", + alias = "lookback", + value_name = "FILE", + help = "Path to the backup file", + conflicts_with_all = ["save", "disable_auto_save", "shell", "shell_options", "is_exec", "is_bell", "is_precise", "interval"] + )] + pub load: Option, } fn parse_duration_from_str(s: &str) -> Result { diff --git a/src/components/execution_result.rs b/src/components/execution_result.rs index f236e06..13f4825 100644 --- a/src/components/execution_result.rs +++ b/src/components/execution_result.rs @@ -207,6 +207,18 @@ impl Component for ExecutionResult { } } + if y_scrollable > 0 { + body.width = area.width.saturating_sub(1); + let scrollbar = Scrollbar::new(ScrollbarOrientation::VerticalRight) + .symbols(scrollbar::VERTICAL) + .style(scroll_style) + .thumb_symbol("║"); + f.render_stateful_widget(scrollbar, area, &mut self.y_state); + if x_max > body.width as usize { + x_scrollable = x_scrollable.saturating_add(1); + } + } + self.y_state = self.y_state.content_length(y_scrollable); self.x_state = self.x_state.content_length(x_scrollable); diff --git a/src/components/home.rs b/src/components/home.rs index 53b2f7b..2c43cb1 100644 --- a/src/components/home.rs +++ b/src/components/home.rs @@ -1,3 +1,4 @@ +use core::time; use std::{collections::HashMap, time::Duration}; use color_eyre::{eyre::Result, owo_colors::OwoColorize}; @@ -14,11 +15,14 @@ use crate::{ action::{Action, DiffMode}, config::{Config, KeyBindings, RuntimeConfig}, mode::Mode, + store::Record, + widget::history_item::HisotryItem, }; pub struct Home { command_tx: Option>, config: Config, + runtime_config: RuntimeConfig, is_no_title: bool, mode: Mode, @@ -34,6 +38,7 @@ pub struct Home { } impl Home { + #[allow(clippy::too_many_arguments)] pub fn new( config: Config, runtime_config: RuntimeConfig, @@ -41,8 +46,11 @@ impl Home { diff_mode: Option, is_bell: bool, is_no_title: bool, + read_only: bool, + timemachine_mode: bool, ) -> Self { Self { + runtime_config: runtime_config.clone(), command_tx: None, config: config.clone(), is_no_title, @@ -53,9 +61,9 @@ impl Home { execution_result_component: ExecutionResult::new(is_fold), history_component: History::new(runtime_config.clone()), prompt_component: Prompt::new(), - status_component: Status::new(is_fold, diff_mode, is_bell), + status_component: Status::new(is_fold, diff_mode, is_bell, read_only), help_component: Help::new(config), - timemachine_mode: false, + timemachine_mode, } } diff --git a/src/components/interval.rs b/src/components/interval.rs index 98202e2..44f5c78 100644 --- a/src/components/interval.rs +++ b/src/components/interval.rs @@ -48,7 +48,6 @@ impl Component for Interval { .title("Every") .borders(Borders::ALL) .border_style(self.config.get_style("border")) - // .border_style(Color::Indexed(90)) .title_style(self.config.get_style("title")); let text = humantime::format_duration(self.runtime_config.interval.to_std().unwrap_or_default()) diff --git a/src/components/status.rs b/src/components/status.rs index d209ae5..cd0eae9 100644 --- a/src/components/status.rs +++ b/src/components/status.rs @@ -21,10 +21,11 @@ pub struct Status { diff_mode: Option, is_suspend: bool, is_bell: bool, + read_only: bool, } impl Status { - pub fn new(is_fold: bool, diff_mode: Option, is_bell: bool) -> Self { + pub fn new(is_fold: bool, diff_mode: Option, is_bell: bool, read_only: bool) -> Self { Self { command_tx: None, config: Config::new().unwrap(), @@ -32,6 +33,7 @@ impl Status { diff_mode, is_suspend: false, is_bell, + read_only, } } } @@ -87,22 +89,30 @@ impl Component for Status { } else { status.push(Span::styled(" [D]iff±", disabled_style)); }; - status.push(Span::styled( - " [S]uspend", - if self.is_suspend { - enabled_style - } else { - disabled_style - }, - )); - status.push(Span::styled( - " [B]ell", - if self.is_bell { - enabled_style - } else { - disabled_style - }, - )); + + if !self.read_only { + status.push(Span::styled( + " [S]uspend", + if self.is_suspend { + enabled_style + } else { + disabled_style + }, + )); + status.push(Span::styled( + " [B]ell", + if self.is_bell { + enabled_style + } else { + disabled_style + }, + )); + } else { + status.push(Span::styled( + " Read-only", + self.config.get_style("readonly"), + )); + } let line = Line::raw("").spans(status); let paragraph = Paragraph::new(line).alignment(Alignment::Right); diff --git a/src/config.rs b/src/config.rs index 1b50649..36cc860 100644 --- a/src/config.rs +++ b/src/config.rs @@ -18,6 +18,12 @@ use crate::{ old_config::{General as OldGeneral, OldConfig}, }; +#[derive(Clone, Debug)] +pub struct RuntimeConfig { + pub interval: Duration, + pub command: Vec, +} + const CONFIG: &str = include_str!("../.config/config.json5"); #[derive(Clone, Debug, Deserialize, Default)] @@ -63,12 +69,6 @@ pub struct Config { pub general: General, } -#[derive(Clone, Debug)] -pub struct RuntimeConfig { - pub interval: Duration, - pub command: Vec, -} - impl Config { pub fn new() -> Result { let data_dir = crate::utils::get_data_dir(); diff --git a/src/main.rs b/src/main.rs index 1c49eff..c67750c 100644 --- a/src/main.rs +++ b/src/main.rs @@ -21,10 +21,15 @@ mod types; pub mod utils; mod widget; +use std::path::PathBuf; + use chrono::Duration; use clap::Parser; use cli::Cli; -use color_eyre::eyre::Result; +use color_eyre::eyre::{eyre, Result}; +use directories::ProjectDirs; +use store::Store; +use tempdir::TempDir; use crate::{ app::App, @@ -38,9 +43,40 @@ async fn tokio_main() -> Result<()> { let args = Cli::parse(); let interval = Duration::from(args.interval); - let store = store::memory::MemoryStore::new(); - let mut app = App::new(args, store)?; - app.run().await?; + + if args.load.is_none() && args.command.is_empty() { + return Err(eyre!("No command provided")); + } + if args.load.is_some() && args.command.len() > 1 { + return Err(eyre!("Can not use --load with command")); + } + + if args.disable_auto_save { + let store = store::memory::MemoryStore::new(); + let mut app = App::new(args, store, false)?; + app.run().await?; + } else if let Some(l) = &args.load { + let store = store::sqlite::SQLiteStore::new(l.clone(), false)?; + let mut app = App::new(args.clone(), store, true)?; + app.run().await?; + } else if let Some(b) = &args.save { + let store = store::sqlite::SQLiteStore::new(b.clone(), true)?; + let mut app = App::new(args.clone(), store, false)?; + app.run().await?; + } else { + let tmp_dir = TempDir::new("viddy")?; + let tmp_path = tmp_dir.into_path(); + let file_path = tmp_path.join("backup.sqlite"); + let store = store::sqlite::SQLiteStore::new(file_path.clone(), true)?; + let mut app = App::new(args.clone(), store, false)?; + app.run().await?; + + println!("Backup saved at {}", file_path.to_str().unwrap()); + println!( + "Run `viddy --lookback {}` to load backup", + file_path.to_str().unwrap() + ); + } Ok(()) } diff --git a/src/runner.rs b/src/runner.rs index f076930..1ed8c32 100644 --- a/src/runner.rs +++ b/src/runner.rs @@ -1,3 +1,4 @@ +use color_eyre::Result; use std::{ops::Sub, sync::Arc}; use dissimilar::{diff, Chunk}; @@ -14,15 +15,17 @@ use crate::{ types::ExecutionId, }; -pub async fn run_executor( +pub async fn run_executor( actions: mpsc::UnboundedSender, mut store: S, runtime_config: RuntimeConfig, shell: Option<(String, Vec)>, is_suspend: Arc>, -) { - let mut counter = 0; +) -> Result<()> { + let latest_id = store.get_latest_id()?; + let mut counter = latest_id.map(|id| id.0 + 1).unwrap_or(0); loop { + counter += 1; if *is_suspend.lock().await { tokio::time::sleep(std::time::Duration::from_secs(1)).await; continue; @@ -46,9 +49,9 @@ pub async fn run_executor( let utf8_stderr = String::from_utf8_lossy(&stderr).to_string(); let end_time = chrono::Local::now(); - let latest_id = store.get_latest_id(); + let latest_id = store.get_latest_id()?; let diff = if let Some(latest_id) = latest_id { - if let Some(record) = store.get_record(latest_id) { + if let Some(record) = store.get_record(latest_id)? { let old_stdout = String::from_utf8_lossy(&record.stdout).to_string(); Some(count_diff(&old_stdout, &utf8_stdout)) } else { @@ -76,27 +79,27 @@ pub async fn run_executor( diff, previous_id: latest_id, }; - store.add_record(record); + store.add_record(record)?; if let Err(e) = actions.send(Action::FinishExecution(id, start_time, diff, exit_code)) { eprintln!("Failed to send result: {:?}", e); } - counter += 1; - tokio::time::sleep(runtime_config.interval.to_std().unwrap()).await; } } -pub async fn run_executor_precise( +pub async fn run_executor_precise( actions: mpsc::UnboundedSender, mut store: S, runtime_config: RuntimeConfig, shell: Option<(String, Vec)>, is_suspend: Arc>, -) { - let mut counter = 0; +) -> Result<()> { + let latest_id = store.get_latest_id()?; + let mut counter = latest_id.map(|id| id.0 + 1).unwrap_or(0); loop { + counter += 1; let start_time = chrono::Local::now(); if *is_suspend.lock().await { tokio::time::sleep(std::time::Duration::from_secs(1)).await; @@ -120,9 +123,9 @@ pub async fn run_executor_precise( let utf8_stderr = String::from_utf8_lossy(&stderr).to_string(); let end_time = chrono::Local::now(); - let latest_id = store.get_latest_id(); + let latest_id = store.get_latest_id()?; let diff = if let Some(latest_id) = latest_id { - if let Some(record) = store.get_record(latest_id) { + if let Some(record) = store.get_record(latest_id)? { let old_stdout = String::from_utf8_lossy(&record.stdout).to_string(); Some(count_diff(&old_stdout, &utf8_stdout)) } else { @@ -150,14 +153,12 @@ pub async fn run_executor_precise( diff, previous_id: latest_id, }; - store.add_record(record); + store.add_record(record)?; if let Err(e) = actions.send(Action::FinishExecution(id, start_time, diff, exit_code)) { eprintln!("Failed to send result: {:?}", e); } - counter += 1; - let elapased = chrono::Local::now().signed_duration_since(start_time); let sleep_time = runtime_config.interval.sub(elapased); if let Ok(sleep_time) = sleep_time.to_std() { diff --git a/src/store.rs b/src/store.rs index 8d49bcd..cb21760 100644 --- a/src/store.rs +++ b/src/store.rs @@ -1,5 +1,7 @@ pub mod memory; +pub mod sqlite; +use color_eyre::eyre::Result; use std::{ collections::HashMap, sync::{Arc, RwLock}, @@ -9,10 +11,13 @@ use chrono::{DateTime, Local}; use crate::types::ExecutionId; -pub trait Store { - fn add_record(&mut self, record: Record); - fn get_record(&self, id: ExecutionId) -> Option; - fn get_latest_id(&self) -> Option; +pub trait Store: Clone + Send + Sync + 'static { + fn add_record(&mut self, record: Record) -> Result<()>; + fn get_record(&self, id: ExecutionId) -> Result>; + fn get_latest_id(&self) -> Result>; + fn get_records(&self) -> Result>; + fn get_runtime_config(&self) -> Result>; + fn set_runtime_config(&mut self, config: RuntimeConfig) -> Result<()>; } #[derive(Debug, Clone)] @@ -26,3 +31,9 @@ pub struct Record { pub diff: Option<(u32, u32)>, pub previous_id: Option, } + +#[derive(Debug, Clone, Default)] +pub struct RuntimeConfig { + pub interval: String, + pub command: String, +} diff --git a/src/store/memory.rs b/src/store/memory.rs index 70b7d5d..adfd6de 100644 --- a/src/store/memory.rs +++ b/src/store/memory.rs @@ -1,18 +1,19 @@ +use crate::store::{Record, RuntimeConfig, Store}; +use crate::types::ExecutionId; +use color_eyre::eyre::Result; use std::collections::HashMap; use std::sync::{Arc, RwLock}; -use crate::types::ExecutionId; -use crate::store::{Record, Store}; - #[derive(Debug)] struct MemoryStoreData { records: HashMap, latest_id: Option, } -#[derive(Clone, Debug)] +#[derive(Debug, Clone)] pub struct MemoryStore { data: Arc>, + runtime_config: Arc>>, } impl MemoryStore { @@ -22,31 +23,56 @@ impl MemoryStore { records: HashMap::new(), latest_id: None, })), + runtime_config: Arc::new(RwLock::new(None)), } } } impl Store for MemoryStore { - fn add_record(&mut self, record: Record) { + fn add_record(&mut self, record: Record) -> Result<()> { if let Ok(mut data) = self.data.write() { data.latest_id = Some(record.id); data.records.insert(record.id, record); } + Ok(()) } - fn get_record(&self, id: ExecutionId) -> Option { - if let Ok(data) = self.data.read() { + fn get_record(&self, id: ExecutionId) -> Result> { + Ok(if let Ok(data) = self.data.read() { data.records.get(&id).cloned() } else { None - } + }) } - fn get_latest_id(&self) -> Option { - if let Ok(data) = self.data.read() { + fn get_latest_id(&self) -> Result> { + Ok(if let Ok(data) = self.data.read() { data.latest_id } else { None + }) + } + + fn get_records(&self) -> Result> { + Ok(if let Ok(data) = self.data.read() { + data.records.values().cloned().collect() + } else { + vec![] + }) + } + + fn get_runtime_config(&self) -> Result> { + Ok(if let Ok(runtime_config) = self.runtime_config.read() { + runtime_config.clone() + } else { + None + }) + } + + fn set_runtime_config(&mut self, config: RuntimeConfig) -> Result<()> { + if let Ok(mut runtime_config) = self.runtime_config.write() { + *runtime_config = Some(config); } + Ok(()) } } diff --git a/src/store/sqlite.rs b/src/store/sqlite.rs new file mode 100644 index 0000000..2054a0f --- /dev/null +++ b/src/store/sqlite.rs @@ -0,0 +1,194 @@ +use chrono::{DateTime, Local, Utc}; +use color_eyre::Result; +use rusqlite::Connection; +use std::collections::HashMap; +use std::path::PathBuf; +use std::sync::{Arc, Mutex}; + +use crate::store::{Record, Store}; +use crate::types::ExecutionId; +use crate::widget::history_item::HisotryItem; + +#[derive(Debug, Clone)] +pub struct SQLiteStore { + conn: Arc>, +} + +impl SQLiteStore { + pub fn new(path: PathBuf, init: bool) -> Result { + if init && path.exists() { + std::fs::remove_file(&path)?; + } + + let conn = Connection::open_with_flags( + path, + rusqlite::OpenFlags::SQLITE_OPEN_READ_WRITE | rusqlite::OpenFlags::SQLITE_OPEN_CREATE, + )?; + + if init { + conn.execute( + "CREATE TABLE record ( + id INTEGER PRIMARY KEY, + start_time TEXT NOT NULL, + stdout BLOB NOT NULL, + stderr BLOB NOT NULL, + end_time TEXT NOT NULL, + exit_code INTEGER NOT NULL, + diff_add INTEGER, + diff_delete INTEGER, + previous_id INTEGER + )", + (), + )?; + + conn.execute( + "CREATE TABLE runtime_config ( + interval INTEGER NOT NULL, + command TEXT NOT NULL + )", + (), + )?; + } + + Ok(Self { + conn: Arc::new(Mutex::new(conn)), + }) + } +} + +impl Store for SQLiteStore { + fn add_record(&mut self, record: Record) -> Result<()> { + if let Ok(conn) = self.conn.lock() { + conn.execute( + "INSERT INTO record ( + id, start_time, stdout, stderr, end_time, exit_code, diff_add, diff_delete, previous_id + ) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8, ?9)", + ( + record.id, + record.start_time.to_utc().to_rfc3339(), + record.stdout, + record.stderr, + record.end_time.to_utc().to_rfc3339(), + record.exit_code, + record.diff.map(|(add, delete)| add as i64), + record.diff.map(|(add, delete)| delete as i64), + record.previous_id, + ), + )?; + Ok(()) + } else { + color_eyre::eyre::bail!("Failed to get connection") + } + } + + fn get_record(&self, id: ExecutionId) -> Result> { + if let Ok(conn) = self.conn.lock() { + let r = conn.query_row("SELECT * FROM record WHERE id = ?1", [id], |row| { + let start_time = row.get::<_, DateTime>(1)?; + let end_time = row.get::<_, DateTime>(4)?; + let diff_add: Option = row.get(6)?; + let diff_delete: Option = row.get(7)?; + let diff = diff_add.zip(diff_delete); + Ok(Record { + id: row.get(0)?, + start_time: start_time.with_timezone(&Local), + stdout: row.get(2)?, + stderr: row.get(3)?, + end_time: end_time.with_timezone(&Local), + exit_code: row.get(5)?, + diff, + previous_id: row.get(8)?, + }) + }); + + match r { + Ok(record) => Ok(Some(record)), + Err(rusqlite::Error::QueryReturnedNoRows) => Ok(None), + Err(e) => Err(e.into()), + } + } else { + color_eyre::eyre::bail!("Failed to get connection") + } + } + + fn get_latest_id(&self) -> Result> { + if let Ok(conn) = self.conn.lock() { + let r = conn.query_row( + "SELECT id FROM record ORDER BY id DESC LIMIT 1", + [], + |row| row.get(0), + ); + + match r { + Ok(id) => Ok(Some(id)), + Err(rusqlite::Error::QueryReturnedNoRows) => Ok(None), + Err(e) => Err(e.into()), + } + } else { + color_eyre::eyre::bail!("Failed to get connection") + } + } + + fn get_records(&self) -> Result> { + if let Ok(conn) = self.conn.lock() { + let mut stmt = conn.prepare("SELECT * FROM record")?; + let records = stmt + .query_map([], |row| { + let start_time = row.get::<_, DateTime>(1)?; + let end_time = row.get::<_, DateTime>(4)?; + let diff_add: Option = row.get(6)?; + let diff_delete: Option = row.get(7)?; + let diff = diff_add.zip(diff_delete); + Ok(Record { + id: row.get(0)?, + start_time: start_time.with_timezone(&Local), + stdout: row.get(2)?, + stderr: row.get(3)?, + end_time: end_time.with_timezone(&Local), + exit_code: row.get(5)?, + diff, + previous_id: row.get(8)?, + }) + })? + .collect::>>()?; + Ok(records) + } else { + color_eyre::eyre::bail!("Failed to get connection") + } + } + + fn get_runtime_config(&self) -> Result> { + if let Ok(conn) = self.conn.lock() { + let r = conn.query_row( + "SELECT * FROM runtime_config ORDER BY ROWID DESC LIMIT 1", + [], + |row| { + Ok(crate::store::RuntimeConfig { + interval: row.get(0)?, + command: row.get(1)?, + }) + }, + ); + + match r { + Ok(config) => Ok(Some(config)), + Err(rusqlite::Error::QueryReturnedNoRows) => Ok(None), + Err(e) => Err(e.into()), + } + } else { + color_eyre::eyre::bail!("Failed to get connection") + } + } + + fn set_runtime_config(&mut self, config: crate::store::RuntimeConfig) -> Result<()> { + if let Ok(conn) = self.conn.lock() { + conn.execute( + "INSERT INTO runtime_config (interval, command) VALUES (?1, ?2)", + (config.interval, config.command), + )?; + Ok(()) + } else { + color_eyre::eyre::bail!("Failed to get connection") + } + } +} diff --git a/src/types.rs b/src/types.rs index f0dac64..3bf29ec 100644 --- a/src/types.rs +++ b/src/types.rs @@ -1,5 +1,9 @@ use std::fmt::{self, Display, Formatter}; +use rusqlite::{ + types::{FromSql, ToSql, ToSqlOutput}, + Result, +}; use serde::{Deserialize, Serialize}; #[derive(Debug, Copy, PartialEq, Clone, Eq, Hash, Serialize, Deserialize)] @@ -10,3 +14,17 @@ impl Display for ExecutionId { write!(f, "{}", self.0) } } + +impl ToSql for ExecutionId { + fn to_sql(&self) -> Result> { + Ok(ToSqlOutput::Owned(rusqlite::types::Value::Integer( + i64::from(self.0), + ))) + } +} + +impl FromSql for ExecutionId { + fn column_result(value: rusqlite::types::ValueRef<'_>) -> rusqlite::types::FromSqlResult { + value.as_i64().map(|v| ExecutionId(v as u32)) + } +}