diff --git a/.gitignore b/.gitignore index d3b84d9590bb8..95d6c4b3ac6c3 100644 --- a/.gitignore +++ b/.gitignore @@ -4,6 +4,7 @@ crates/**/target benches/**/target tools/**/target **/*.rs.bk +rustc-ice-* # Cargo Cargo.lock @@ -21,6 +22,7 @@ dxil.dll assets/**/*.meta crates/bevy_asset/imported_assets imported_assets +.http-asset-cache # Bevy Examples example_showcase_config.ron diff --git a/Cargo.toml b/Cargo.toml index fd61ba79853ba..404638207e01d 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -436,6 +436,12 @@ file_watcher = ["bevy_internal/file_watcher"] # Enables watching in memory asset providers for Bevy Asset hot-reloading embedded_watcher = ["bevy_internal/embedded_watcher"] +# Enables using assets from HTTP sources +http_source = ["bevy_internal/http_source"] + +# Assets downloaded from HTTP sources are cached +http_source_cache = ["bevy_internal/http_source_cache"] + # Enable stepping-based debugging of Bevy systems bevy_debug_stepping = ["bevy_internal/bevy_debug_stepping"] @@ -496,7 +502,7 @@ accesskit = "0.17" smol = "2" smol-macros = "0.1" smol-hyper = "0.1" -ureq = { version = "2.10.1", features = ["json"] } +ureq = { version = "2.10", default-features = false, features = ["json", "tls"] } [target.'cfg(target_arch = "wasm32")'.dev-dependencies] wasm-bindgen = { version = "0.2" } @@ -1640,12 +1646,24 @@ path = "examples/asset/extra_source.rs" doc-scrape-examples = true [package.metadata.example.extra_asset_source] -name = "Extra asset source" +name = "Extra Asset Source" description = "Load an asset from a non-standard asset source" category = "Assets" # Uses non-standard asset path wasm = false +[[example]] +name = "http_source" +path = "examples/asset/http_source.rs" +doc-scrape-examples = true +required-features = ["http_source"] + +[package.metadata.example.http_source] +name = "HTTP Asset Source" +description = "Load an asset from a http source" +category = "Assets" +wasm = true + [[example]] name = "hot_asset_reloading" path = "examples/asset/hot_asset_reloading.rs" diff --git a/crates/bevy_asset/Cargo.toml b/crates/bevy_asset/Cargo.toml index e1d62ad2a2169..64571b2bca25e 100644 --- a/crates/bevy_asset/Cargo.toml +++ b/crates/bevy_asset/Cargo.toml @@ -14,6 +14,8 @@ keywords = ["bevy"] file_watcher = ["notify-debouncer-full", "watch"] embedded_watcher = ["file_watcher"] multi_threaded = ["bevy_tasks/multi_threaded"] +http_source = ["ureq", "rustls", "once_cell"] +http_source_cache = [] asset_processor = [] watch = [] trace = [] @@ -63,6 +65,19 @@ js-sys = "0.3" [target.'cfg(not(target_arch = "wasm32"))'.dependencies] notify-debouncer-full = { version = "0.4.0", optional = true } +# note that while ureq is semver stable rustls is not, meaning possible breaking changes on minor releases. https://github.com/bevyengine/bevy/pull/16366#issuecomment-2572890794 +ureq = { git = "https://github.com/algesten/ureq", rev = "423aa8f05b30799129080e1bbe821ef067b50ed2", optional = true, default-features = false, features = [ + "rustls-no-provider", + "gzip", + "json", +] } +once_cell = { version = "1.20", optional = true } +rustls = { version = "0.23", optional = true, default-features = false, features = [ + "aws_lc_rs", + "logging", + "std", + "tls12", +] } [dev-dependencies] bevy_core = { path = "../bevy_core", version = "0.15.0-dev" } diff --git a/crates/bevy_asset/src/http_source.rs b/crates/bevy_asset/src/http_source.rs new file mode 100644 index 0000000000000..dfeec60f329db --- /dev/null +++ b/crates/bevy_asset/src/http_source.rs @@ -0,0 +1,275 @@ +use crate::io::{AssetReader, AssetReaderError, Reader}; +use crate::io::{AssetSource, PathStream}; +use crate::AssetApp; +use bevy_app::App; +use bevy_tasks::ConditionalSendFuture; +use std::path::{Path, PathBuf}; + +/// Adds the `http` and `https` asset sources to the app. +/// Any asset path that begins with `http` or `https` will be loaded from the web +/// via `fetch`(wasm) or `ureq`(native). +/// +/// Due to [licensing complexities](https://github.com/briansmith/ring/issues/1827) +/// secure `https` requests are disabled by default in non-wasm builds. +/// To enable add this to your dependencies in Cargo.toml: +/// ```toml +/// ureq = { version = "*", features = ["tls"] } +/// ``` +pub fn http_source_plugin(app: &mut App) { + app.register_asset_source( + "http", + AssetSource::build().with_reader(|| Box::new(HttpSourceAssetReader::Http)), + ); + app.register_asset_source( + "https", + AssetSource::build().with_reader(|| Box::new(HttpSourceAssetReader::Https)), + ); +} + +/// Asset reader that treats paths as urls to load assets from. +pub enum HttpSourceAssetReader { + /// Unencrypted connections. + Http, + /// Use TLS for setting up connections. + Https, +} + +impl HttpSourceAssetReader { + fn make_uri(&self, path: &Path) -> PathBuf { + PathBuf::from(match self { + Self::Http => "http://", + Self::Https => "https://", + }) + .join(path) + } + + /// See [`crate::io::get_meta_path`] + fn make_meta_uri(&self, path: &Path) -> Option { + let mut uri = self.make_uri(path); + let mut extension = path.extension()?.to_os_string(); + extension.push(".meta"); + uri.set_extension(extension); + Some(uri) + } +} + +#[cfg(target_arch = "wasm32")] +async fn get<'a>(path: PathBuf) -> Result, AssetReaderError> { + use crate::io::wasm::HttpWasmAssetReader; + + HttpWasmAssetReader::new("") + .fetch_bytes(path) + .await + .map(|r| Box::new(r) as Box) +} + +#[cfg(not(target_arch = "wasm32"))] +async fn get<'a>(path: PathBuf) -> Result, AssetReaderError> { + use crate::io::VecReader; + use std::io::{self, BufReader, Read}; + + let str_path = path.to_str().ok_or_else(|| { + AssetReaderError::Io( + io::Error::new( + io::ErrorKind::Other, + format!("non-utf8 path: {}", path.display()), + ) + .into(), + ) + })?; + + #[cfg(feature = "http_source_cache")] + if let Some(data) = http_asset_cache::try_load_from_cache(str_path)? { + return Ok(Box::new(VecReader::new(data))); + } + use once_cell::sync::Lazy; + use ureq::Agent; + + static AGENT: Lazy = Lazy::new(|| { + use alloc::sync::Arc; + use ureq::{ + tls::{TlsConfig, TlsProvider}, + Agent, + }; + + let crypto = Arc::new(rustls::crypto::aws_lc_rs::default_provider()); + Agent::config_builder() + .tls_config( + TlsConfig::builder() + .provider(TlsProvider::Rustls) + // requires rustls or rustls-no-provider feature + .unversioned_rustls_crypto_provider(crypto) + .build(), + ) + .build() + .new_agent() + }); + + match AGENT.get(str_path).call() { + Ok(mut response) => { + // let mut reader = response.into_reader(); + let mut reader = BufReader::new(response.body_mut().with_config().reader()); + + let mut buffer = Vec::new(); + reader.read_to_end(&mut buffer)?; + + #[cfg(feature = "http_source_cache")] + http_asset_cache::save_to_cache(str_path, &buffer)?; + + Ok(Box::new(VecReader::new(buffer))) + } + // ureq considers all >=400 status codes as errors + Err(ureq::Error::StatusCode(code)) => { + if code == 404 { + Err(AssetReaderError::NotFound(path)) + } else { + Err(AssetReaderError::HttpError(code)) + } + } + Err(err) => Err(AssetReaderError::Io( + io::Error::new( + io::ErrorKind::Other, + format!( + "unexpected error while loading asset {}: {}", + path.display(), + err + ), + ) + .into(), + )), + } +} + +impl AssetReader for HttpSourceAssetReader { + fn read<'a>( + &'a self, + path: &'a Path, + ) -> impl ConditionalSendFuture, AssetReaderError>> { + get(self.make_uri(path)) + } + + async fn read_meta<'a>(&'a self, path: &'a Path) -> Result, AssetReaderError> { + match self.make_meta_uri(path) { + Some(uri) => get(uri).await, + None => Err(AssetReaderError::NotFound( + "source path has no extension".into(), + )), + } + } + + async fn is_directory<'a>(&'a self, _path: &'a Path) -> Result { + Ok(false) + } + + async fn read_directory<'a>( + &'a self, + path: &'a Path, + ) -> Result, AssetReaderError> { + Err(AssetReaderError::NotFound(self.make_uri(path))) + } +} + +/// A naive implementation of an HTTP asset cache that never invalidates. +/// `ureq` currently does not support caching, so this is a simple workaround. +/// It should eventually be replaced by `http-cache` or similar, see [tracking issue](https://github.com/06chaynes/http-cache/issues/91) +#[cfg(feature = "http_source_cache")] +mod http_asset_cache { + use core::hash::{Hash, Hasher}; + use std::collections::hash_map::DefaultHasher; + use std::fs::{self, File}; + use std::io::{self, Read, Write}; + use std::path::PathBuf; + + const CACHE_DIR: &str = ".http-asset-cache"; + + fn url_to_hash(url: &str) -> String { + let mut hasher = DefaultHasher::new(); + url.hash(&mut hasher); + format!("{:x}", hasher.finish()) + } + + pub fn try_load_from_cache(url: &str) -> Result>, io::Error> { + let filename = url_to_hash(url); + let cache_path = PathBuf::from(CACHE_DIR).join(&filename); + + if cache_path.exists() { + let mut file = File::open(&cache_path)?; + let mut buffer = Vec::new(); + file.read_to_end(&mut buffer)?; + Ok(Some(buffer)) + } else { + Ok(None) + } + } + + pub fn save_to_cache(url: &str, data: &[u8]) -> Result<(), io::Error> { + let filename = url_to_hash(url); + let cache_path = PathBuf::from(CACHE_DIR).join(&filename); + + fs::create_dir_all(CACHE_DIR).ok(); + + let mut cache_file = File::create(&cache_path)?; + cache_file.write_all(data)?; + + Ok(()) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn make_http_uri() { + assert_eq!( + HttpSourceAssetReader::Http + .make_uri(Path::new("example.com/favicon.png")) + .to_str() + .unwrap(), + "http://example.com/favicon.png" + ); + } + + #[test] + fn make_https_uri() { + assert_eq!( + HttpSourceAssetReader::Https + .make_uri(Path::new("example.com/favicon.png")) + .to_str() + .unwrap(), + "https://example.com/favicon.png" + ); + } + + #[test] + fn make_http_meta_uri() { + assert_eq!( + HttpSourceAssetReader::Http + .make_meta_uri(Path::new("example.com/favicon.png")) + .expect("cannot create meta uri") + .to_str() + .unwrap(), + "http://example.com/favicon.png.meta" + ); + } + + #[test] + fn make_https_meta_uri() { + assert_eq!( + HttpSourceAssetReader::Https + .make_meta_uri(Path::new("example.com/favicon.png")) + .expect("cannot create meta uri") + .to_str() + .unwrap(), + "https://example.com/favicon.png.meta" + ); + } + + #[test] + fn make_https_without_extension_meta_uri() { + assert_eq!( + HttpSourceAssetReader::Https.make_meta_uri(Path::new("example.com/favicon")), + None + ); + } +} diff --git a/crates/bevy_asset/src/io/mod.rs b/crates/bevy_asset/src/io/mod.rs index 0c4c0b1f00356..2674813f85d36 100644 --- a/crates/bevy_asset/src/io/mod.rs +++ b/crates/bevy_asset/src/io/mod.rs @@ -46,7 +46,8 @@ pub enum AssetReaderError { Io(Arc), /// The HTTP request completed but returned an unhandled [HTTP response status code](https://developer.mozilla.org/en-US/docs/Web/HTTP/Status). - /// If the request fails before getting a status code (e.g. request timeout, interrupted connection, etc), expect [`AssetReaderError::Io`]. + /// - If the request returns a 404 error, expect [`AssetReaderError::NotFound`]. + /// - If the request fails before getting a status code (e.g. request timeout, interrupted connection, etc), expect [`AssetReaderError::Io`]. #[error("Encountered HTTP status {0:?} when loading asset")] HttpError(u16), } diff --git a/crates/bevy_asset/src/io/wasm.rs b/crates/bevy_asset/src/io/wasm.rs index 25a5d223cbb0b..982d5d22f7540 100644 --- a/crates/bevy_asset/src/io/wasm.rs +++ b/crates/bevy_asset/src/io/wasm.rs @@ -51,7 +51,11 @@ fn js_value_to_err(context: &str) -> impl FnOnce(JsValue) -> std::io::Error + '_ } impl HttpWasmAssetReader { - async fn fetch_bytes<'a>(&self, path: PathBuf) -> Result { + // Also used by HttpSourceAssetReader + pub(crate) async fn fetch_bytes<'a>( + &self, + path: PathBuf, + ) -> Result { // The JS global scope includes a self-reference via a specializing name, which can be used to determine the type of global context available. let global: Global = js_sys::global().unchecked_into(); let promise = if !global.window().is_undefined() { diff --git a/crates/bevy_asset/src/lib.rs b/crates/bevy_asset/src/lib.rs index d252946c3c186..44e07308ff908 100644 --- a/crates/bevy_asset/src/lib.rs +++ b/crates/bevy_asset/src/lib.rs @@ -179,6 +179,9 @@ mod reflect; mod render_asset; mod server; +#[cfg(feature = "http_source")] +mod http_source; + pub use assets::*; pub use bevy_asset_macros::Asset; pub use direct_access_ext::DirectAssetAccessExt; @@ -310,6 +313,9 @@ impl AssetPlugin { impl Plugin for AssetPlugin { fn build(&self, app: &mut App) { + #[cfg(feature = "http_source")] + app.add_plugins(http_source::http_source_plugin); + let embedded = EmbeddedAssetRegistry::default(); { let mut sources = app diff --git a/crates/bevy_internal/Cargo.toml b/crates/bevy_internal/Cargo.toml index 3acc57d16cbf0..f75d59a72afd5 100644 --- a/crates/bevy_internal/Cargo.toml +++ b/crates/bevy_internal/Cargo.toml @@ -190,6 +190,12 @@ debug_glam_assert = ["bevy_math/debug_glam_assert"] default_font = ["bevy_text?/default_font"] +# Enables using assets from HTTP sources +http_source = ["bevy_asset?/http_source"] + +# Assets downloaded from HTTP sources are cached +http_source_cache = ["bevy_asset?/http_source_cache"] + # Enables the built-in asset processor for processed assets. asset_processor = ["bevy_asset?/asset_processor"] diff --git a/deny.toml b/deny.toml index f8114fed1d1a7..530715041f683 100644 --- a/deny.toml +++ b/deny.toml @@ -21,7 +21,10 @@ allow = [ "ISC", "MIT", "MIT-0", + "MPL-2.0", "Unlicense", + "OpenSSL", + "Unicode-3.0", "Zlib", ] diff --git a/docs/cargo_features.md b/docs/cargo_features.md index c07158ef2e1d4..aea223995cc1b 100644 --- a/docs/cargo_features.md +++ b/docs/cargo_features.md @@ -77,6 +77,8 @@ The default feature set enables most of the expected features of a game engine, |ghost_nodes|Experimental support for nodes that are ignored for UI layouting| |gif|GIF image format support| |glam_assert|Enable assertions to check the validity of parameters passed to glam| +|http_source|Enables using assets from HTTP sources| +|http_source_cache|Assets downloaded from HTTP sources are cached| |ico|ICO image format support| |ios_simulator|Enable support for the ios_simulator by downgrading some rendering capabilities| |jpeg|JPEG image format support| diff --git a/examples/README.md b/examples/README.md index bf4e04d0f5010..cbd479ee383a7 100644 --- a/examples/README.md +++ b/examples/README.md @@ -239,7 +239,8 @@ Example | Description [Custom Asset](../examples/asset/custom_asset.rs) | Implements a custom asset loader [Custom Asset IO](../examples/asset/custom_asset_reader.rs) | Implements a custom AssetReader [Embedded Asset](../examples/asset/embedded_asset.rs) | Embed an asset in the application binary and load it -[Extra asset source](../examples/asset/extra_source.rs) | Load an asset from a non-standard asset source +[Extra Asset Source](../examples/asset/extra_source.rs) | Load an asset from a non-standard asset source +[HTTP Asset Source](../examples/asset/http_source.rs) | Load an asset from a http source [Hot Reloading of Assets](../examples/asset/hot_asset_reloading.rs) | Demonstrates automatic reloading of assets when modified on disk [Mult-asset synchronization](../examples/asset/multi_asset_sync.rs) | Demonstrates how to wait for multiple assets to be loaded. [Repeated texture configuration](../examples/asset/repeated_texture.rs) | How to configure the texture to repeat instead of the default clamp to edges diff --git a/examples/asset/http_source.rs b/examples/asset/http_source.rs new file mode 100644 index 0000000000000..2529e5240c971 --- /dev/null +++ b/examples/asset/http_source.rs @@ -0,0 +1,25 @@ +//! Example usage of the `http` asset source to load assets from the web. +//! +//! Due to [licensing complexities](https://github.com/briansmith/ring/issues/1827) +//! secure `https` requests are disabled by default in non-wasm builds. +//! To enable add this to your dependencies in Cargo.toml: +//! ```toml +//! ureq = { version = "*", features = ["tls"] } +//! ``` +use bevy::prelude::*; + +fn main() { + App::new() + .add_plugins(DefaultPlugins) + .add_systems(Startup, setup) + .run(); +} + +fn setup(mut commands: Commands, asset_server: Res) { + commands.spawn(Camera2d); + + commands.spawn( + // Simply use a url where you would normally use an asset folder relative path + Sprite::from_image(asset_server.load("https://raw.githubusercontent.com/bevyengine/bevy/refs/heads/main/assets/branding/bevy_bird_dark.png")) + ); +}