From c54e1abbb727f78d6cfa5c92495b27e2cd3554b6 Mon Sep 17 00:00:00 2001 From: Juha Kukkonen Date: Tue, 21 May 2024 18:12:10 +0300 Subject: [PATCH] Enhance utoipa-swagger-ui build Use `curl` to download Swagger UI by default and allow using `reqwest` optionally with `reqwest` feature flag. Reqwest will be automatically enabled on Windows targets. Change the re-build parameters as follows. * For file protocol the re-build should happen when the upstream file changes * For http protocol the re-build should happen when the environment variable holding the downloadd URL will change. --- utoipa-swagger-ui/Cargo.toml | 23 +++++++- utoipa-swagger-ui/README.md | 11 +++- utoipa-swagger-ui/build.rs | 111 +++++++++++++++++++++++++++-------- 3 files changed, 116 insertions(+), 29 deletions(-) diff --git a/utoipa-swagger-ui/Cargo.toml b/utoipa-swagger-ui/Cargo.toml index c21c720b..e80f7459 100644 --- a/utoipa-swagger-ui/Cargo.toml +++ b/utoipa-swagger-ui/Cargo.toml @@ -12,15 +12,20 @@ authors = ["Juha Kukkonen "] rust-version.workspace = true [features] +default = ["url"] debug = [] debug-embed = ["rust-embed/debug-embed"] +reqwest = ["dep:reqwest"] +url = ["dep:url"] [dependencies] rust-embed = { version = "8" } mime_guess = { version = "2.0" } actix-web = { version = "4", optional = true, default-features = false } rocket = { version = "0.5", features = ["json"], optional = true } -axum = { version = "0.7", default-features = false, features = ["json"], optional = true } +axum = { version = "0.7", default-features = false, features = [ + "json", +], optional = true } utoipa = { version = "4", path = "../utoipa" } serde = { version = "1.0", features = ["derive"] } serde_json = { version = "1.0" } @@ -35,4 +40,18 @@ rustdoc-args = ["--cfg", "doc_cfg"] [build-dependencies] zip = { version = "1", default-features = false, features = ["deflate"] } regex = "1.7" -reqwest = { version = "0.12", features = ["blocking", "rustls-tls"], default-features = false } + +# enabled optionally to allow rust only build with expense of bigger dependency tree and platform +# independant build. By default `curl` system package is tried for downloading the Swagger UI. +reqwest = { version = "0.12", features = [ + "blocking", + "rustls-tls", +], default-features = false, optional = true } +url = { version = "2.5", optional = true } + +# Windows is forced to use reqwest to download the Swagger UI. +[target.'cfg(windows)'.build-dependencies] +reqwest = { version = "0.12", features = [ + "blocking", + "rustls-tls", +], default-features = false } diff --git a/utoipa-swagger-ui/README.md b/utoipa-swagger-ui/README.md index 5cf74783..561f9c1b 100644 --- a/utoipa-swagger-ui/README.md +++ b/utoipa-swagger-ui/README.md @@ -30,6 +30,9 @@ more details at [serve](https://docs.rs/utoipa-swagger-ui/latest/utoipa_swagger_ hassle free. * **debug-embed** Enables `debug-embed` feature on `rust_embed` crate to allow embedding files in debug builds as well. +* **reqwest** Use `reqwest` for downloading Swagger UI accoring to the `SWAGGER_UI_DOWNLOAD_URL` environment + variable. This is only enabled by default on _Windows_. +* **url** Enabled by default for parsing and encoding the download URL. ## Install @@ -49,7 +52,13 @@ utoipa-swagger-ui = { version = "7", features = ["actix-web"] } **Note!** Also remember that you already have defined `utoipa` dependency in your `Cargo.toml` -## Config +## Build Config + +_`utoipa-swagger-ui` crate will by default try to use system `curl` package for downloading the Swagger UI. It +can optionally be downloaded with `reqwest` by enabling `reqwest` feature. On Windows the `reqwest` feature +is enabled by default. Reqwest can be useful for platform independent builds however bringing quite a few +unnecessary dependencies just to download a file. If the `SWAGGER_UI_DOWNLOAD_URL` is a file path then no +downloading will happen._ The following configuration env variables are available at build time: diff --git a/utoipa-swagger-ui/build.rs b/utoipa-swagger-ui/build.rs index 23c55e09..75d5fbde 100644 --- a/utoipa-swagger-ui/build.rs +++ b/utoipa-swagger-ui/build.rs @@ -1,10 +1,10 @@ -use reqwest::Url; use std::{ env, error::Error, fs::{self, File}, - io::{self, Read}, - path::PathBuf, + io, + path::{Path, PathBuf}, + process::Command, }; use regex::Regex; @@ -22,34 +22,40 @@ use zip::{result::ZipError, ZipArchive}; const SWAGGER_UI_DOWNLOAD_URL_DEFAULT: &str = "https://github.com/swagger-api/swagger-ui/archive/refs/tags/v5.17.3.zip"; +const SWAGGER_UI_DOWNLOAD_URL: &str = "SWAGGER_UI_DOWNLOAD_URL"; +const SWAGGER_UI_OVERWRITE_FOLDER: &str = "SWAGGER_UI_OVERWRITE_FOLDER"; + fn main() { let target_dir = env::var("OUT_DIR").unwrap(); - println!("OUT_DIR: {}", target_dir); + println!("OUT_DIR: {target_dir}"); let url = - env::var("SWAGGER_UI_DOWNLOAD_URL").unwrap_or(SWAGGER_UI_DOWNLOAD_URL_DEFAULT.to_string()); + env::var(SWAGGER_UI_DOWNLOAD_URL).unwrap_or(SWAGGER_UI_DOWNLOAD_URL_DEFAULT.to_string()); - println!("SWAGGER_UI_DOWNLOAD_URL: {}", url); + println!("{SWAGGER_UI_DOWNLOAD_URL}: {url}"); let zip_filename = url.split('/').last().unwrap().to_string(); let zip_path = [&target_dir, &zip_filename].iter().collect::(); - if !zip_path.exists() { - if url.starts_with("http://") || url.starts_with("https://") { - println!("start download to : {:?}", zip_path); - download_file(&url, zip_path.clone()).unwrap(); - } else if url.starts_with("file://") { - let file_path = Url::parse(&url).unwrap().to_file_path().unwrap(); - println!("start copy to : {:?}", zip_path); - fs::copy(file_path, zip_path.clone()).unwrap(); - } else { - panic!("invalid SWAGGER_UI_DOWNLOAD_URL: {} -> must start with http:// | https:// | file://", url); - } + if url.starts_with("file:") { + let mut file_path = url::Url::parse(&url).unwrap().to_file_path().unwrap(); + file_path = fs::canonicalize(file_path).expect("swagger ui download path should exists"); + + // with file protocol utoipa swagger ui should compile when file changes + println!("cargo:rerun-if-changed={:?}", file_path); + + println!("start copy to : {:?}", zip_path); + fs::copy(file_path, zip_path.clone()).unwrap(); + } else if url.starts_with("http://") || url.starts_with("https://") { + println!("start download to : {:?}", zip_path); + + // with http protocol we update when the 'SWAGGER_UI_DOWNLOAD_URL' changes + println!("cargo:rerun-if-env-changed={SWAGGER_UI_DOWNLOAD_URL}"); + + download_file(&url, zip_path.clone()).unwrap(); } else { - println!("already downloaded or copied: {:?}", zip_path); + panic!("invalid {SWAGGER_UI_DOWNLOAD_URL}: {url} -> must start with http:// | https:// | file:"); } - println!("cargo:rerun-if-changed={:?}", zip_path.clone()); - let swagger_ui_zip = File::open([&target_dir, &zip_filename].iter().collect::()).unwrap(); @@ -63,10 +69,10 @@ fn main() { write_embed_code(&target_dir, &zip_top_level_folder); let overwrite_folder = - PathBuf::from(env::var("SWAGGER_UI_OVERWRITE_FOLDER").unwrap_or("overwrite".to_string())); + PathBuf::from(env::var(SWAGGER_UI_OVERWRITE_FOLDER).unwrap_or("overwrite".to_string())); if overwrite_folder.exists() { - println!("SWAGGER_UI_OVERWRITE_FOLDER: {:?}", overwrite_folder); + println!("{SWAGGER_UI_OVERWRITE_FOLDER}: {overwrite_folder:?}"); for entry in fs::read_dir(overwrite_folder).unwrap() { let entry = entry.unwrap(); @@ -75,10 +81,7 @@ fn main() { overwrite_target_file(&target_dir, &zip_top_level_folder, path_in); } } else { - println!( - "SWAGGER_UI_OVERWRITE_FOLDER not found: {:?}", - overwrite_folder - ); + println!("{SWAGGER_UI_OVERWRITE_FOLDER} not found: {overwrite_folder:?}"); } } @@ -170,6 +173,22 @@ struct SwaggerUiDist; } fn download_file(url: &str, path: PathBuf) -> Result<(), Box> { + let reqwest_feature = env::var("CARGO_FEATURE_REQWEST"); + println!("reqwest feature: {reqwest_feature:?}"); + if reqwest_feature.is_ok() { + #[cfg(any(feature = "reqwest", target_os = "windows"))] + { + download_file_reqwest(url, path)?; + } + Ok(()) + } else { + println!("trying to download using `curl` system package"); + download_file_curl(url, path.as_path()) + } +} + +#[cfg(any(feature = "reqwest", target_os = "windows"))] +fn download_file_reqwest(url: &str, path: PathBuf) -> Result<(), Box> { let mut client_builder = reqwest::blocking::Client::builder(); if let Ok(cainfo) = env::var("CARGO_HTTP_CAINFO") { @@ -189,13 +208,53 @@ fn download_file(url: &str, path: PathBuf) -> Result<(), Box> { Ok(()) } +#[cfg(any(feature = "reqwest", target_os = "windows"))] fn parse_ca_file(path: &str) -> Result> { let mut buf = Vec::new(); + use io::Read; File::open(path)?.read_to_end(&mut buf)?; let cert = reqwest::Certificate::from_pem(&buf)?; Ok(cert) } +fn download_file_curl>(url: &str, target_dir: T) -> Result<(), Box> { + let url = url::Url::parse(url)?; + + let mut args = Vec::with_capacity(6); + args.extend([ + "-sSL", + "-o", + target_dir + .as_ref() + .as_os_str() + .to_str() + .expect("target dir should be valid utf-8"), + url.as_str(), + ]); + let cacert = env::var("CARGO_HTTP_CAINFO").unwrap_or_default(); + if !cacert.is_empty() { + args.extend(["--cacert", &cacert]); + } + + let download = Command::new("curl") + .args(args) + .spawn() + .and_then(|mut child| child.wait()); + + Ok(download + .and_then(|status| { + if status.success() { + Ok(()) + } else { + Err(std::io::Error::new( + io::ErrorKind::Other, + format!("curl download file exited with error status: {status}"), + )) + } + }) + .map_err(Box::new)?) +} + fn overwrite_target_file(target_dir: &str, swagger_ui_dist_zip: &str, path_in: PathBuf) { let filename = path_in.file_name().unwrap().to_str().unwrap(); println!("overwrite file: {:?}", path_in.file_name().unwrap());