From 3d3d56440a4a55bb63cbec7805ea109db892bcb6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Tristan=20Dani=C3=ABl=20Maat?= <tm@tlater.net> Date: Wed, 26 Mar 2025 02:27:39 +0800 Subject: [PATCH] feat(card_db): Only fetch database when upstream has updated --- .cargo/config.toml | 2 + Cargo.lock | 52 ++++++++++++++++ Cargo.toml | 6 +- flake.nix | 2 + src/bin/database_worker.rs | 1 + src/database.rs | 20 +----- src/database/worker.rs | 67 ++++++++++++++++++-- src/utils.rs | 122 +++++++++++++++++++++++++++++++++++++ 8 files changed, 249 insertions(+), 23 deletions(-) create mode 100644 .cargo/config.toml diff --git a/.cargo/config.toml b/.cargo/config.toml new file mode 100644 index 0000000..f4e8c00 --- /dev/null +++ b/.cargo/config.toml @@ -0,0 +1,2 @@ +[build] +target = "wasm32-unknown-unknown" diff --git a/Cargo.lock b/Cargo.lock index 4a72acf..128cf41 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -155,6 +155,15 @@ version = "1.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8b96ec4966b5813e2c0507c1f86115c8c5abaadc3980879c3424042a02fd1ad3" +[[package]] +name = "cc" +version = "1.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fcb57c740ae1daf453ae85f16e37396f672b039e00d9d866e07ddb24e328e3a" +dependencies = [ + "shlex", +] + [[package]] name = "cfg-if" version = "1.0.0" @@ -427,7 +436,9 @@ dependencies = [ "url", "wasm-bindgen", "wasm-bindgen-futures", + "wasm-bindgen-test", "wasm-logger", + "web-sys", ] [[package]] @@ -1328,6 +1339,16 @@ dependencies = [ "quote", ] +[[package]] +name = "minicov" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f27fe9f1cc3c22e1687f9446c2083c4c5fc7f0bcf1c7a86bdbded14985895b4b" +dependencies = [ + "cc", + "walkdir", +] + [[package]] name = "minimal-lexical" version = "0.2.1" @@ -1893,6 +1914,12 @@ dependencies = [ "syn", ] +[[package]] +name = "shlex" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" + [[package]] name = "slab" version = "0.4.9" @@ -2093,6 +2120,7 @@ checksum = "dad298b01a40a23aac4580b67e3dbedb7cc8402f3592d7f49469de2ea4aecdd8" dependencies = [ "deranged", "itoa", + "js-sys", "num-conv", "powerfmt", "serde", @@ -2366,6 +2394,30 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "wasm-bindgen-test" +version = "0.3.50" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "66c8d5e33ca3b6d9fa3b4676d774c5778031d27a578c2b007f905acf816152c3" +dependencies = [ + "js-sys", + "minicov", + "wasm-bindgen", + "wasm-bindgen-futures", + "wasm-bindgen-test-macro", +] + +[[package]] +name = "wasm-bindgen-test-macro" +version = "0.3.50" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "17d5042cc5fa009658f9a7333ef24291b1291a25b6382dd68862a7f3b969f69b" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "wasm-logger" version = "0.2.0" diff --git a/Cargo.toml b/Cargo.toml index 7a33316..aa1f690 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -18,7 +18,7 @@ wasm-bindgen = "0.2.100" wasm-bindgen-futures = { version = "0.4.50", features = ["futures-core-03-stream"] } diesel-derive-enum = { version = "2.1.0", features = ["sqlite"] } diesel_migrations = { git = "https://github.com/diesel-rs/diesel.git", features = ["sqlite"] } -time = { version = "0.3.39", features = ["parsing"] } +time = { version = "0.3.39", features = ["formatting", "parsing", "wasm-bindgen"] } wasm-logger = "0.2.0" log = "0.4.26" sqlite-wasm-rs = { version = ">=0.3.0, <0.4.0" , default-features = false, features = ["precompiled"]} @@ -26,3 +26,7 @@ thiserror-ext = "0.2.1" futures-util = "0.3.31" gloo = { version = "0.11.0", features = ["futures"] } gloo-net = { version = "0.6.0" } +web-sys = { version = "0.3.77", features = ["FileSystemWritableFileStream", "WorkerGlobalScope"] } + +[target.'cfg(all(any(target_arch = "wasm32", target_arch = "wasm64"), target_os = "unknown"))'.dev-dependencies] +wasm-bindgen-test = "0.3.50" diff --git a/flake.nix b/flake.nix index dc2446d..eb79fea 100644 --- a/flake.nix +++ b/flake.nix @@ -33,6 +33,8 @@ leptosfmt rust-analyzer trunk + wasm-pack + nodePackages.npm lld diff --git a/src/bin/database_worker.rs b/src/bin/database_worker.rs index 4cecf83..e857e2d 100644 --- a/src/bin/database_worker.rs +++ b/src/bin/database_worker.rs @@ -9,6 +9,7 @@ fn main() { if let Err(e) = a { log::error!("{:?}", e); + panic!("{}", e); } }) } diff --git a/src/database.rs b/src/database.rs index a4c652c..64f0dd9 100644 --- a/src/database.rs +++ b/src/database.rs @@ -4,11 +4,7 @@ pub mod models; pub mod schema; mod worker; -use sqlite_wasm_rs::export::{install_opfs_sahpool, OpfsSAHError}; -use thiserror_ext::AsReport; -use worker::{DatabaseWorker, Query}; - -use crate::utils::github_fetch_file; +use worker::{BrowserDatabaseError, DatabaseWorker, Query}; pub struct BrowserDatabase { worker: WorkerBridge<DatabaseWorker>, @@ -34,19 +30,9 @@ impl BrowserDatabase { /// Initialize the database worker. /// /// Must be executed in a worker script, not on the main thread. - pub async fn init_worker() -> Result<(), OpfsSAHError> { + pub async fn init_worker() -> Result<(), BrowserDatabaseError> { log::debug!("Initializing database worker..."); - let sah_pool_util = install_opfs_sahpool(None, false).await?; - - let babel_cdb = match github_fetch_file("ProjectIgnis", "BabelCDB", "cards.cdb").await { - Ok(cdb) => cdb, - Err(err) => { - log::error!("{}", err.to_report_string()); - panic!("{:?}", err); - } - }; - - sah_pool_util.import_db("/cards.db", &babel_cdb)?; + DatabaseWorker::initialize_db().await?; DatabaseWorker::registrar().register(); log::debug!("Database worker set up!"); Ok(()) diff --git a/src/database/worker.rs b/src/database/worker.rs index 9d6916c..7ee7eb2 100644 --- a/src/database/worker.rs +++ b/src/database/worker.rs @@ -2,16 +2,22 @@ use diesel::{prelude::*, Connection}; use futures::TryFutureExt; use gloo::worker::{HandlerId, Worker, WorkerScope}; use serde::{Deserialize, Serialize}; +use sqlite_wasm_rs::export::{install_opfs_sahpool, OpfsSAHPoolUtil}; use thiserror::Error; use thiserror_ext::AsReport; +use time::{format_description::well_known::Rfc3339, OffsetDateTime}; use wasm_bindgen_futures::spawn_local; +use crate::utils::{ + self, github_fetch_file, github_fetch_last_updated, read_opfs_file, write_opfs_file, +}; + use super::{ models::CardDetails, schema::{datas, texts}, }; -const DB_URL: &str = "file:/cards.db?vfs=opfs-sahpool&mode=ro"; +const DB_URL: &str = "file:/cards.cdb?vfs=opfs-sahpool&mode=ro"; #[derive(Serialize, Deserialize)] pub enum Query { @@ -20,6 +26,7 @@ pub enum Query { #[derive(Serialize, Deserialize, Debug)] pub enum Response { + Update, ByPassword(Option<CardDetails>), Error(String), } @@ -27,6 +34,13 @@ pub enum Response { pub struct DatabaseWorker {} impl DatabaseWorker { + pub async fn initialize_db() -> Result<()> { + let sah_pool_util = install_opfs_sahpool(None, false).await?; + update_card_db(&sah_pool_util).await?; + + Ok(()) + } + async fn run_query(query: &Query) -> Result<Response> { let res = match query { Query::ByPassword(pw) => Response::ByPassword(Self::get_card(*pw).await?), @@ -126,15 +140,58 @@ impl Worker for DatabaseWorker { } } +async fn should_update_card_db() -> Result<bool> { + let last_update = { + let ts = read_opfs_file("last_update.txt").await?; + if !ts.is_empty() { + OffsetDateTime::parse(&ts, &Rfc3339).map_err(BrowserDatabaseError::from) + } else { + Ok(OffsetDateTime::UNIX_EPOCH) + } + }?; + + log::debug!("Last database update: {}", last_update); + + let latest_version = github_fetch_last_updated("ProjectIgnis", "BabelCDB", "cards.cdb").await?; + + log::debug!("Latest database version: {}", latest_version); + + Ok(last_update < latest_version) +} + +async fn update_card_db(sah_pool_util: &OpfsSAHPoolUtil) -> Result<()> { + if should_update_card_db().await? { + log::info!("Updating card database..."); + let db = github_fetch_file("ProjectIgnis", "BabelCDB", "cards.cdb").await?; + sah_pool_util.import_db("/cards.cdb", &db)?; + write_opfs_file( + "last_update.txt", + &OffsetDateTime::now_utc().format(&Rfc3339)?, + ) + .await?; + } else { + log::debug!("Skipping card database updte"); + } + + Ok(()) +} + #[derive(Debug, Error)] pub enum BrowserDatabaseError { - // #[error("failed to set up browser database storage")] - // VFSError(#[from] sqlite_wasm_rs::export::OpfsSAHError), - // #[error("could not fetch updated card database")] - // CardDbFetchError(#[from] GithubError), #[error("failed to connect to database")] DieselConnectionError(#[from] diesel::ConnectionError), #[error("failed to query database")] DieselQueryError(#[from] diesel::result::Error), + + #[error("failed to write database to local storage")] + FileStorage(#[from] sqlite_wasm_rs::export::OpfsSAHError), + #[error("failed to fetch Babel DB")] + GitHubAPI(#[from] crate::utils::GithubError), + #[error("failed to retrieve last Babel DB update date")] + LocalStorage(#[from] utils::OpfsError), + #[error("failed to retrieve last Babel DB update date")] + TimeParseError(#[from] time::error::Parse), + #[error("failed to write last Babel DB update date")] + TimeFormatError(#[from] time::error::Format), } type Result<T> = std::result::Result<T, BrowserDatabaseError>; diff --git a/src/utils.rs b/src/utils.rs index 31e459d..2be9eaa 100644 --- a/src/utils.rs +++ b/src/utils.rs @@ -2,6 +2,13 @@ use gloo::net::http; use thiserror::Error; use time::{format_description::well_known::Rfc3339, OffsetDateTime}; +use wasm_bindgen::{JsCast, JsValue}; +use wasm_bindgen_futures::{js_sys, JsFuture}; +use web_sys::{ + js_sys::Object, FileSystemDirectoryHandle, FileSystemFileHandle, FileSystemGetFileOptions, + FileSystemWritableFileStream, WorkerGlobalScope, +}; + const GITHUB_API: &str = "https://api.github.com"; pub async fn github_fetch_last_updated( @@ -58,3 +65,118 @@ pub enum GithubError { #[error("date/time format reported for GitHub commit is invalid")] InvalidDateTimeError(#[from] time::error::Parse), } + +/// Read a file from the in-browser OPFS filesystem, creating the file +/// if it does not exist. +pub async fn read_opfs_file(path: &str) -> Result<String, OpfsError> { + let dir: FileSystemDirectoryHandle = JsFuture::from( + js_sys::global() + .dyn_into::<WorkerGlobalScope>() + .map_err(OpfsError::NotSupported)? + .navigator() + .storage() + .get_directory(), + ) + .await + .map_err(OpfsError::HandleAccess)? + .into(); + + let opt = FileSystemGetFileOptions::new(); + opt.set_create(true); + let file_handle: FileSystemFileHandle = + JsFuture::from(dir.get_file_handle_with_options(path, &opt)) + .await + .map_err(OpfsError::HandleAccess)? + .into(); + + let file: web_sys::File = JsFuture::from(file_handle.get_file()) + .await + .map_err(OpfsError::HandleAccess)? + .into(); + + Ok(gloo::file::futures::read_as_text(&gloo::file::Blob::from(file)).await?) +} + +/// Write contents to a file in the in-browser OPFS, creating the file +/// if it does not exist. +pub async fn write_opfs_file(path: &str, contents: &str) -> Result<(), OpfsError> { + let dir: FileSystemDirectoryHandle = JsFuture::from( + js_sys::global() + .dyn_into::<WorkerGlobalScope>() + .map_err(OpfsError::NotSupported)? + .navigator() + .storage() + .get_directory(), + ) + .await + .map_err(OpfsError::HandleAccess)? + .into(); + + let opt = FileSystemGetFileOptions::new(); + opt.set_create(true); + let file_handle: FileSystemFileHandle = + JsFuture::from(dir.get_file_handle_with_options(path, &opt)) + .await + .map_err(OpfsError::HandleAccess)? + .into(); + + let write_handle: FileSystemWritableFileStream = JsFuture::from(file_handle.create_writable()) + .await + .map_err(OpfsError::HandleAccess)? + .into(); + + JsFuture::from( + write_handle + .write_with_str(contents) + .map_err(OpfsError::WriteFile)?, + ) + .await + .map_err(OpfsError::WriteFile)?; + + JsFuture::from(write_handle.close()) + .await + .map_err(OpfsError::WriteFile)?; + + Ok(()) +} + +#[derive(Debug, Error)] +pub enum OpfsError { + #[error("OPFS is not supported by this browser: {0:?}")] + NotSupported(Object), + #[error("could not get OPFS directory path: {0:?}")] + HandleAccess(JsValue), + #[error("could not read OPFS file")] + ReadFile(#[from] gloo::file::FileReadError), + #[error("could not write OPFS file: {0:?}")] + WriteFile(JsValue), +} + +#[cfg(all( + any(target_arch = "wasm32", target_arch = "wasm64"), + target_os = "unknown", + test +))] +mod worker_tests { + use wasm_bindgen_test::*; + wasm_bindgen_test_configure!(run_in_dedicated_worker); + + use super::{read_opfs_file, write_opfs_file}; + + #[wasm_bindgen_test] + async fn test_read_write_opfs() { + let empty = read_opfs_file("test.txt") + .await + .expect("file read must succeed"); + assert_eq!(empty, ""); + + write_opfs_file("test.txt", "contents") + .await + .expect("file write must succeed"); + + let nonempty = read_opfs_file("test.txt") + .await + .expect("file read must succeed"); + assert_eq!(nonempty, "contents"); + } +}