error_handling/
error_handling.rs1use bevy::ecs::{entity::SpawnError, error::warn, world::DeferredWorld};
5use bevy::math::sampling::UniformMeshSampler;
6use bevy::prelude::*;
7
8use chacha20::ChaCha8Rng;
9use rand::distr::Distribution;
10use rand::SeedableRng;
11
12fn main() {
13 let mut app = App::new();
14 app.set_error_handler(warn);
23
24 app.add_plugins(DefaultPlugins);
25
26 #[cfg(feature = "mesh_picking")]
27 app.add_plugins(MeshPickingPlugin);
28
29 app.add_systems(Startup, setup);
33
34 app.add_systems(Startup, failing_commands);
37
38 app.add_systems(
40 PostStartup,
41 failing_system.pipe(|result: In<Result>| {
42 let _ = result.0.inspect_err(|err| info!("captured error: {err}"));
43 }),
44 );
45
46 app.add_observer(fallible_observer);
48
49 app.run();
55}
56
57fn setup(
61 mut commands: Commands,
62 mut meshes: ResMut<Assets<Mesh>>,
63 mut materials: ResMut<Assets<StandardMaterial>>,
64) -> Result {
65 let mut seeded_rng = ChaCha8Rng::seed_from_u64(19878367467712);
66
67 commands.spawn((
69 Mesh3d(meshes.add(Plane3d::default().mesh().size(12.0, 12.0))),
70 MeshMaterial3d(materials.add(Color::srgb(0.3, 0.5, 0.3))),
71 Transform::from_xyz(0.0, -2.5, 0.0),
72 ));
73
74 commands.spawn((
76 PointLight {
77 shadow_maps_enabled: true,
78 ..default()
79 },
80 Transform::from_xyz(4.0, 8.0, 4.0),
81 ));
82
83 commands.spawn((
85 Camera3d::default(),
86 Transform::from_xyz(-2.0, 3.0, 5.0).looking_at(Vec3::ZERO, Vec3::Y),
87 ));
88
89 let mut sphere_mesh = Sphere::new(1.0).mesh().ico(7)?;
91 sphere_mesh.generate_tangents()?;
92
93 let mut sphere = commands.spawn((
95 Mesh3d(meshes.add(sphere_mesh.clone())),
96 MeshMaterial3d(materials.add(StandardMaterial::default())),
97 Transform::from_xyz(-1.0, 1.0, 0.0),
98 ));
99
100 let triangles = sphere_mesh.triangles()?;
102 let distribution = UniformMeshSampler::try_new(triangles)?;
103
104 let point_mesh = meshes.add(Sphere::new(0.01).mesh().ico(3)?);
106 let point_material = materials.add(StandardMaterial {
107 base_color: Srgba::RED.into(),
108 emissive: LinearRgba::rgb(1.0, 0.0, 0.0),
109 ..default()
110 });
111
112 for point in distribution.sample_iter(&mut seeded_rng).take(10000) {
114 sphere.with_child((
115 Mesh3d(point_mesh.clone()),
116 MeshMaterial3d(point_material.clone()),
117 Transform::from_translation(point),
118 ));
119 }
120
121 Ok(())
123}
124
125fn fallible_observer(
127 pointer_move: On<Pointer<Move>>,
128 mut world: DeferredWorld,
129 mut step: Local<f32>,
130) -> Result {
131 let mut transform = world
132 .get_mut::<Transform>(pointer_move.entity)
133 .ok_or("No transform found.")?;
134
135 *step = if transform.translation.x > 3. {
136 -0.1
137 } else if transform.translation.x < -3. || *step == 0. {
138 0.1
139 } else {
140 *step
141 };
142
143 transform.translation.x += *step;
144
145 Ok(())
146}
147
148#[derive(Resource)]
149struct UninitializedResource;
150
151fn failing_system(world: &mut World) -> Result {
152 world
153 .get_resource::<UninitializedResource>()
156 .ok_or("Resource not initialized")
158 .with_severity(Severity::Warning)?;
161
162 world
163 .spawn_empty_at(Entity::from_raw_u32(12345678).unwrap())
165 .map_severity(|e| match e {
166 SpawnError::AlreadySpawned => Severity::Debug,
168 SpawnError::Invalid(_) => Severity::Error,
170 })?;
171
172 Ok(())
173}
174
175fn failing_commands(mut commands: Commands) {
176 commands
177 .entity(Entity::from_raw_u32(12345678).unwrap())
179 .insert(Transform::default());
183
184 commands.queue_handled(
187 |world: &mut World| -> Result {
188 world
189 .get_resource::<UninitializedResource>()
190 .ok_or("Resource not initialized when accessed in a command")?;
191
192 Ok(())
193 },
194 |error, context| {
195 error!("{error}, {context}");
196 },
197 );
198}