Bevy system error handling

Bevy tracking Latest version Apache 2.0 Documentation

Decorate your bevy system with the [sysfail] macro attribute to handle failure.

Before

```rust use bevy::prelude::*; use bevy::utils::Duration;

use thiserror::Error;

[derive(Error, Debug)]

enum GizmoError { #[error("A Gizmo error")] Error, }

[derive(Debug, PartialEq, Eq, Hash, SystemSet, Clone)]

enum TransformGizmoSystem { Drag, Place }

fn main() { let mut app = App::new(); app.addplugin(bevy::time::TimePlugin) .addsystem( draggizmo .pipe(printgizmoerror) .inset(TransformGizmoSystem::Drag), ) .addsystem( deletegizmo .pipe(|In()| {}) .after(TransformGizmoSystem::Place)) .addsystem( placegizmo .pipe(printgizmoerror) .inset(TransformGizmoSystem::Place) .after(TransformGizmoSystem::Drag), ); app.update(); }

fn printgizmoerror( In(result): In>>, mut lasterroroccurence: Local>, time: Res

fn draggizmo(time: Resseconds()); let _ = Err(GizmoError::Error)?; println!("This will never print"); Ok(()) }

fn place_gizmo() -> Result<(), Box> { let () = Result::<(), &'static str>::Ok(())?; println!("this line should actually show up"); let _ = Err("Ah, some creative use of info logging I see")?; Ok(()) }

fn deletegizmo(time: Resseconds()); let _ = None?; println!("This will never print"); Some(()) } ```

After

```rust use bevy::prelude::; use bevy_mod_sysfail::macros::;

use thiserror::Error;

[derive(Error, Debug)]

enum GizmoError { #[error("A Gizmo error")] Error, }

fn main() { let mut app = App::new(); app.addplugin(bevy::time::TimePlugin) .addsystem(draggizmo) .addsystem(deletegizmo.after(placegizmo)) .addsystem(placegizmo.after(drag_gizmo)); app.update(); }

[sysfail(log)]

fn draggizmo(time: Resseconds()); let _ = Err(GizmoError::Error)?; println!("This will never print"); Ok(()) }

[sysfail(log(level = "info"))]

fn place_gizmo() -> Result<(), &'static str> { let () = Result::<(), &'static str>::Ok(())?; println!("this line should actually show up"); let _ = Err("Ah, some creative use of info logging I see")?; Ok(()) }

[quick_sysfail]

fn deletegizmo(time: Resseconds()); let _ = None?; println!("This will never print"); } ```

sysfail attribute

[sysfail] is an attribute macro you can slap on top of your systems to define the handling of errors. Unlike pipe, this is done directly at the definition site, and not when adding to the app. As a result, it's easy to see at a glance what kind of error handling is happening in the system, it also allows using the system name as a label in system dependency specification.

The [sysfail] attribute can only be used on systems returning a type implementing the [Failure] trait. [Failure] is implemented for Result<(), impl FailureMode> and Option<()>. [sysfail] takes a single argument, it is one of the following:

Note that with log, the macro generates a new system with additional parameters.

quick_sysfail attribute

[quick_sysfail] is like sysfail(ignore) but only works on Option<()>. This attribute, unlike sysfail allows you to elide the final Some(()) and the type signature of the system. It's for the maximally lazy, like me.

```rust use bevymodsysfail::macros::*;

[sysfail(ignore)]

fn place_gizmo() -> Option<()> { // … Some(()) } // equivalent to:

[quick_sysfail]

fn quickplacegizmo() { // … } ```

Traits

How error is handled is not very customizable, but there is a few behaviors controllable by the user, always through traits.

Failure trait

[Failure] is implemented for Result<(), impl FailureMode> and Option<()>.

Systems marked with the [sysfail] attribute must return a type implementing [Failure].

FailureMode trait

[FailureMode] defines how the failure is handled. By implementing the trait on your own error types, you can specify:

[FailureMode] is implemented for Box<dyn Error>, anyhow::Error, () and &'static str.

LogLevelOverride trait

[LogLevelOverride] is an extension trait that allows you to override the log level of a failure. Use the warn, trace, debug, silent, error and info methods to specify the level of logging of a failure.

Change log

Version Matrix

| bevy | latest supporting version | |------|--------| | 0.10 | 2.0.0 | | 0.9 | 1.1.0 | | 0.8 | 0.1.0 |

License

Copyright © 2022 Nicola Papale

This software is licensed under Apache 2.0.