use std::sync::LazyLock; use log::trace; use regex::Regex; use crate::{error, file}; use super::api::{NewShareRequest, Uri}; pub trait Client { fn get_file_id(uri: &str) -> error::Result<&str> { /// Pattern breakdown: /// - `^([^:/?#]+)://` – scheme (anything but `:/?#`) + `"://"` /// - `([^/?#]+)` – authority/host (anything but `/?#`) /// - `/api/v2/alias/upload/` – literal path segment /// - `([^/]+)` – capture SID (one or more non-slash chars) /// - `/files/tus/` – literal path segment /// - `(?P[^/]+)` – capture FID (one or more non-slash chars) /// - `$` – end of string static UPLOAD_URL_RE: LazyLock = LazyLock::new(|| { trace!("compiling UPLOAD_URL_RE"); Regex::new( r"^([^:/?#]+)://([^/?#]+)/api/v2/alias/upload/[^/]+/files/tus/(?P[^/]+)$", ) .expect("Regex compilation failed") }); if let Some(fid) = UPLOAD_URL_RE .captures(uri) .and_then(|caps| caps.name("fid").map(|m| m.as_str())) { Ok(fid) } else { Err(error::Error::unknown(format!( "Could not extract File ID from {uri:?}" ))) } } fn share_create( &self, uri: &Uri, alias_id: &str, data: NewShareRequest, ) -> error::Result; fn share_notify(&self, uri: &Uri, alias_id: &str, share_id: &str) -> error::Result<()>; fn file_create( &self, uri: &Uri, alias_id: &str, share_id: &str, file: &file::Checked, ) -> error::Result; fn file_patch( &self, uri: &Uri, alias_id: &str, share_id: &str, chunk: &file::Chunk, ) -> error::Result<()>; } // TODO move into tests subdir // #[cfg(test)] // mod tests { // use super::*; // #[test] // fn test_get_file_id() { // let good = "https://example.com/api/v2/alias/upload/SID123/files/tus/FID456"; // let good = Client::get_file_id(good); // assert!(good.is_ok()); // assert_eq!(good.unwrap(), "FID456"); // let bad = "https://example.com/api/v2/alias/upload//files/tus/FID456"; // missing SID // assert!(Client::get_file_id(bad).is_err()); // let bad: &'static str = "https://example.com/api/v2/alias/upload/SID123/files/tus/"; // missing FID // assert!(Client::get_file_id(bad).is_err()); // } // }