Skip to content

Commit

Permalink
[assets] set LoadState properly and more testing! (bevyengine#2226)
Browse files Browse the repository at this point in the history
1) Sets `LoadState` properly on all failing cases in `AssetServer::load_async`
2) Adds more tests for sad and happy paths of asset loading

_Note_: this brings in the `tempfile` crate.
  • Loading branch information
NathanSWard committed Jun 8, 2021
1 parent c2722f7 commit fe32a60
Show file tree
Hide file tree
Showing 3 changed files with 215 additions and 26 deletions.
4 changes: 4 additions & 0 deletions crates/bevy_asset/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -45,3 +45,7 @@ js-sys = "0.3"

[target.'cfg(target_os = "android")'.dependencies]
ndk-glue = { version = "0.2" }

[dev-dependencies]
futures-lite = "1.4.0"
tempfile = "3.2.0"
230 changes: 204 additions & 26 deletions crates/bevy_asset/src/asset_server.rs
Original file line number Diff line number Diff line change
Expand Up @@ -205,6 +205,7 @@ impl AssetServer {
}
LoadState::Failed => return LoadState::Failed,
LoadState::NotLoaded => return LoadState::NotLoaded,
LoadState::Unloaded => return LoadState::Unloaded,
},
HandleId::Id(_, _) => return LoadState::NotLoaded,
}
Expand All @@ -230,7 +231,6 @@ impl AssetServer {
self.load_untyped(path).typed()
}

// TODO: properly set failed LoadState in all failure cases
async fn load_async(
&self,
asset_path: AssetPath<'_>,
Expand Down Expand Up @@ -272,15 +272,19 @@ impl AssetServer {
source_info.version
};

let set_asset_failed = || {
let mut asset_sources = self.server.asset_sources.write();
let source_info = asset_sources
.get_mut(&asset_path_id.source_path_id())
.expect("`AssetSource` should exist at this point.");
source_info.load_state = LoadState::Failed;
};

// load the asset bytes
let bytes = match self.server.asset_io.load_path(asset_path.path()).await {
Ok(bytes) => bytes,
Err(err) => {
let mut asset_sources = self.server.asset_sources.write();
let source_info = asset_sources
.get_mut(&asset_path_id.source_path_id())
.expect("`AssetSource` should exist at this point.");
source_info.load_state = LoadState::Failed;
set_asset_failed();
return Err(AssetServerError::AssetIoError(err));
}
};
Expand All @@ -293,10 +297,15 @@ impl AssetServer {
version,
&self.server.task_pool,
);
asset_loader

if let Err(err) = asset_loader
.load(&bytes, &mut load_context)
.await
.map_err(AssetServerError::AssetLoaderError)?;
.map_err(AssetServerError::AssetLoaderError)
{
set_asset_failed();
return Err(err);
}

// if version has changed since we loaded and grabbed a lock, return. theres is a newer
// version being loaded
Expand Down Expand Up @@ -500,9 +509,7 @@ impl AssetServer {
.get_or_insert_with(|| self.server.asset_sources.write());
if let Some(source_info) = asset_sources.get_mut(&id.source_path_id()) {
source_info.committed_assets.remove(&id.label_id());
if source_info.is_loaded() {
source_info.load_state = LoadState::Loaded;
}
source_info.load_state = LoadState::Unloaded;
}
}
assets.remove(handle_id);
Expand All @@ -516,23 +523,35 @@ impl AssetServer {
}
}

pub fn free_unused_assets_system(asset_server: Res<AssetServer>) {
fn free_unused_assets_system_impl(asset_server: &AssetServer) {
asset_server.free_unused_assets();
asset_server.mark_unused_assets();
}

pub fn free_unused_assets_system(asset_server: Res<AssetServer>) {
free_unused_assets_system_impl(&asset_server);
}

#[cfg(test)]
mod test {
use super::*;
use crate::{loader::LoadedAsset, update_asset_storage_system};
use bevy_ecs::prelude::*;
use bevy_reflect::TypeUuid;
use bevy_utils::BoxedFuture;

#[derive(Debug, TypeUuid)]
#[uuid = "a5189b72-0572-4290-a2e0-96f73a491c44"]
struct PngAsset;

struct FakePngLoader;
impl AssetLoader for FakePngLoader {
fn load<'a>(
&'a self,
_: &'a [u8],
_: &'a mut LoadContext,
ctx: &'a mut LoadContext,
) -> BoxedFuture<'a, Result<(), anyhow::Error>> {
ctx.set_default_asset(LoadedAsset::new(PngAsset));
Box::pin(async move { Ok(()) })
}

Expand All @@ -541,6 +560,21 @@ mod test {
}
}

struct FailingLoader;
impl AssetLoader for FailingLoader {
fn load<'a>(
&'a self,
_: &'a [u8],
_: &'a mut LoadContext,
) -> BoxedFuture<'a, Result<(), anyhow::Error>> {
Box::pin(async { anyhow::bail!("failed") })
}

fn extensions(&self) -> &[&str] {
&["fail"]
}
}

struct FakeMultipleDotLoader;
impl AssetLoader for FakeMultipleDotLoader {
fn load<'a>(
Expand All @@ -556,10 +590,10 @@ mod test {
}
}

fn setup() -> AssetServer {
fn setup(asset_path: impl AsRef<Path>) -> AssetServer {
use crate::FileAssetIo;

let asset_server = AssetServer {
AssetServer {
server: Arc::new(AssetServerInternal {
loaders: Default::default(),
extension_to_loader_index: Default::default(),
Expand All @@ -568,38 +602,39 @@ mod test {
handle_to_path: Default::default(),
asset_lifecycles: Default::default(),
task_pool: Default::default(),
asset_io: Box::new(FileAssetIo::new(&".")),
asset_io: Box::new(FileAssetIo::new(asset_path)),
}),
};
asset_server.add_loader::<FakePngLoader>(FakePngLoader);
asset_server.add_loader::<FakeMultipleDotLoader>(FakeMultipleDotLoader);
asset_server
}
}

#[test]
fn extensions() {
let asset_server = setup();
let asset_server = setup(".");
asset_server.add_loader(FakePngLoader);

let t = asset_server.get_path_asset_loader("test.png");
assert_eq!(t.unwrap().extensions()[0], "png");
}

#[test]
fn case_insensitive_extensions() {
let asset_server = setup();
let asset_server = setup(".");
asset_server.add_loader(FakePngLoader);

let t = asset_server.get_path_asset_loader("test.PNG");
assert_eq!(t.unwrap().extensions()[0], "png");
}

#[test]
fn no_loader() {
let asset_server = setup();
let asset_server = setup(".");
let t = asset_server.get_path_asset_loader("test.pong");
assert!(t.is_err());
}

#[test]
fn multiple_extensions_no_loader() {
let asset_server = setup();
let asset_server = setup(".");

assert!(
match asset_server.get_path_asset_loader("test.v1.2.3.pong") {
Expand Down Expand Up @@ -634,15 +669,158 @@ mod test {

#[test]
fn filename_with_dots() {
let asset_server = setup();
let asset_server = setup(".");
asset_server.add_loader(FakePngLoader);

let t = asset_server.get_path_asset_loader("test-v1.2.3.png");
assert_eq!(t.unwrap().extensions()[0], "png");
}

#[test]
fn multiple_extensions() {
let asset_server = setup();
let asset_server = setup(".");
asset_server.add_loader(FakeMultipleDotLoader);

let t = asset_server.get_path_asset_loader("test.test.png");
assert_eq!(t.unwrap().extensions()[0], "test.png");
}

fn create_dir_and_file(file: impl AsRef<Path>) -> tempfile::TempDir {
let asset_dir = tempfile::tempdir().unwrap();
std::fs::write(asset_dir.path().join(file), &[]).unwrap();
asset_dir
}

#[test]
fn test_missing_loader() {
let dir = create_dir_and_file("file.not-a-real-extension");
let asset_server = setup(dir.path());

let path: AssetPath = "file.not-a-real-extension".into();
let handle = asset_server.get_handle_untyped(path.get_id());

let err = futures_lite::future::block_on(asset_server.load_async(path.clone(), true))
.unwrap_err();
assert!(match err {
AssetServerError::MissingAssetLoader { extensions } => {
extensions == ["not-a-real-extension"]
}
_ => false,
});

assert_eq!(asset_server.get_load_state(handle), LoadState::NotLoaded);
}

#[test]
fn test_invalid_asset_path() {
let asset_server = setup(".");
asset_server.add_loader(FakePngLoader);

let path: AssetPath = "an/invalid/path.png".into();
let handle = asset_server.get_handle_untyped(path.get_id());

let err = futures_lite::future::block_on(asset_server.load_async(path.clone(), true))
.unwrap_err();
assert!(matches!(err, AssetServerError::AssetIoError(_)));

assert_eq!(asset_server.get_load_state(handle), LoadState::Failed);
}

#[test]
fn test_failing_loader() {
let dir = create_dir_and_file("fake.fail");
let asset_server = setup(dir.path());
asset_server.add_loader(FailingLoader);

let path: AssetPath = "fake.fail".into();
let handle = asset_server.get_handle_untyped(path.get_id());

let err = futures_lite::future::block_on(asset_server.load_async(path.clone(), true))
.unwrap_err();
assert!(matches!(err, AssetServerError::AssetLoaderError(_)));

assert_eq!(asset_server.get_load_state(handle), LoadState::Failed);
}

#[test]
fn test_asset_lifecycle() {
let dir = create_dir_and_file("fake.png");
let asset_server = setup(dir.path());
asset_server.add_loader(FakePngLoader);
let assets = asset_server.register_asset_type::<PngAsset>();

let mut world = World::new();
world.insert_resource(assets);
world.insert_resource(asset_server);

let mut tick = {
let mut free_unused_assets_system = free_unused_assets_system.system();
free_unused_assets_system.initialize(&mut world);
let mut update_asset_storage_system = update_asset_storage_system::<PngAsset>.system();
update_asset_storage_system.initialize(&mut world);

move |world: &mut World| {
free_unused_assets_system.run((), world);
update_asset_storage_system.run((), world);
}
};

fn load_asset(path: AssetPath, world: &World) -> HandleUntyped {
let asset_server = world.get_resource::<AssetServer>().unwrap();
let id = futures_lite::future::block_on(asset_server.load_async(path.clone(), true))
.unwrap();
asset_server.get_handle_untyped(id)
}

fn get_asset(id: impl Into<HandleId>, world: &World) -> Option<&PngAsset> {
world
.get_resource::<Assets<PngAsset>>()
.unwrap()
.get(id.into())
}

fn get_load_state(id: impl Into<HandleId>, world: &World) -> LoadState {
world
.get_resource::<AssetServer>()
.unwrap()
.get_load_state(id.into())
}

// ---
// Start of the actual lifecycle test
// ---

let path: AssetPath = "fake.png".into();
assert_eq!(LoadState::NotLoaded, get_load_state(path.get_id(), &world));

// load the asset
let handle = load_asset(path.clone(), &world);
let weak_handle = handle.clone_weak();

// asset is loading
assert_eq!(LoadState::Loading, get_load_state(&handle, &world));

tick(&mut world);
// asset should exist and be loaded at this point
assert_eq!(LoadState::Loaded, get_load_state(&handle, &world));
assert!(get_asset(&handle, &world).is_some());

// after dropping the handle, next call to `tick` will prepare the assets for removal.
drop(handle);
tick(&mut world);
assert_eq!(LoadState::Loaded, get_load_state(&weak_handle, &world));
assert!(get_asset(&weak_handle, &world).is_some());

// second call to tick will actually remove the asset.
tick(&mut world);
assert_eq!(LoadState::Unloaded, get_load_state(&weak_handle, &world));
assert!(get_asset(&weak_handle, &world).is_none());

// finally, reload the asset
let handle = load_asset(path.clone(), &world);
assert_eq!(LoadState::Loading, get_load_state(&handle, &world));
tick(&mut world);
assert_eq!(LoadState::Loaded, get_load_state(&handle, &world));
assert!(get_asset(&handle, &world).is_some());
}
}
7 changes: 7 additions & 0 deletions crates/bevy_asset/src/info.rs
Original file line number Diff line number Diff line change
Expand Up @@ -41,8 +41,15 @@ impl SourceInfo {
/// The load state of an asset
#[derive(Copy, Clone, Debug, Eq, PartialEq, Ord, PartialOrd)]
pub enum LoadState {
/// The asset has not be loaded.
NotLoaded,
/// The asset in the the process of loading.
Loading,
/// The asset has loaded and is living inside an [`Assets`](crate::Assets) collection.
Loaded,
/// The asset failed to load.
Failed,
/// The asset was previously loaded, however all handles were dropped and
/// the asset was removed from the [`Assets`](crate::Assets) collection.
Unloaded,
}

0 comments on commit fe32a60

Please sign in to comment.