use crate::{ archetype::ArchetypeComponentId, component::ComponentId, query::Access, system::{IntoSystem, System}, world::World, }; use std::{any::TypeId, borrow::Cow}; use super::ReadOnlySystem; /// A [`System`] created by piping the output of the first system into the input of the second. /// /// This can be repeated indefinitely, but system pipes cannot branch: the output is consumed by the receiving system. /// /// Given two systems `A` and `B`, A may be piped into `B` as `A.pipe(B)` if the output type of `A` is /// equal to the input type of `B`. /// /// Note that for [`FunctionSystem`](crate::system::FunctionSystem)s the output is the return value /// of the function and the input is the first [`SystemParam`](crate::system::SystemParam) if it is /// tagged with [`In`](crate::system::In) or `()` if the function has no designated input parameter. /// /// # Examples /// /// ``` /// use std::num::ParseIntError; /// /// use bevy_ecs::prelude::*; /// /// fn main() { /// let mut world = World::default(); /// world.insert_resource(Message("42".to_string())); /// /// // pipe the `parse_message_system`'s output into the `filter_system`s input /// let mut piped_system = parse_message_system.pipe(filter_system); /// piped_system.initialize(&mut world); /// assert_eq!(piped_system.run((), &mut world), Some(42)); /// } /// /// #[derive(Resource)] /// struct Message(String); /// /// fn parse_message_system(message: Res) -> Result { /// message.0.parse::() /// } /// /// fn filter_system(In(result): In>) -> Option { /// result.ok().filter(|&n| n < 100) /// } /// ``` pub struct PipeSystem { system_a: SystemA, system_b: SystemB, name: Cow<'static, str>, component_access: Access, archetype_component_access: Access, } impl PipeSystem { /// Manual constructor for creating a [`PipeSystem`]. /// This should only be used when [`IntoPipeSystem::pipe`] cannot be used, /// such as in `const` contexts. pub const fn new(system_a: SystemA, system_b: SystemB, name: Cow<'static, str>) -> Self { Self { system_a, system_b, name, component_access: Access::new(), archetype_component_access: Access::new(), } } } impl> System for PipeSystem { type In = SystemA::In; type Out = SystemB::Out; fn name(&self) -> Cow<'static, str> { self.name.clone() } fn type_id(&self) -> TypeId { TypeId::of::<(SystemA, SystemB)>() } fn archetype_component_access(&self) -> &Access { &self.archetype_component_access } fn component_access(&self) -> &Access { &self.component_access } fn is_send(&self) -> bool { self.system_a.is_send() && self.system_b.is_send() } fn is_exclusive(&self) -> bool { self.system_a.is_exclusive() || self.system_b.is_exclusive() } unsafe fn run_unsafe(&mut self, input: Self::In, world: &World) -> Self::Out { let out = self.system_a.run_unsafe(input, world); self.system_b.run_unsafe(out, world) } // needed to make exclusive systems work fn run(&mut self, input: Self::In, world: &mut World) -> Self::Out { let out = self.system_a.run(input, world); self.system_b.run(out, world) } fn apply_buffers(&mut self, world: &mut World) { self.system_a.apply_buffers(world); self.system_b.apply_buffers(world); } fn initialize(&mut self, world: &mut World) { self.system_a.initialize(world); self.system_b.initialize(world); self.component_access .extend(self.system_a.component_access()); self.component_access .extend(self.system_b.component_access()); } fn update_archetype_component_access(&mut self, world: &World) { self.system_a.update_archetype_component_access(world); self.system_b.update_archetype_component_access(world); self.archetype_component_access .extend(self.system_a.archetype_component_access()); self.archetype_component_access .extend(self.system_b.archetype_component_access()); } fn check_change_tick(&mut self, change_tick: u32) { self.system_a.check_change_tick(change_tick); self.system_b.check_change_tick(change_tick); } fn get_last_change_tick(&self) -> u32 { self.system_a.get_last_change_tick() } fn set_last_change_tick(&mut self, last_change_tick: u32) { self.system_a.set_last_change_tick(last_change_tick); self.system_b.set_last_change_tick(last_change_tick); } fn default_system_sets(&self) -> Vec> { let mut system_sets = self.system_a.default_system_sets(); system_sets.extend_from_slice(&self.system_b.default_system_sets()); system_sets } } /// SAFETY: Both systems are read-only, so piping them together will only read from the world. unsafe impl> ReadOnlySystem for PipeSystem where SystemA: ReadOnlySystem, SystemB: ReadOnlySystem, { } /// An extension trait providing the [`IntoPipeSystem::pipe`] method to pass input from one system into the next. /// /// The first system must have return type `T` /// and the second system must have [`In`](crate::system::In) as its first system parameter. /// /// This trait is blanket implemented for all system pairs that fulfill the type requirements. /// /// See [`PipeSystem`]. pub trait IntoPipeSystem: IntoSystem<(), Payload, ParamA> + Sized where SystemB: IntoSystem, { /// Pass the output of this system `A` into a second system `B`, creating a new compound system. fn pipe(self, system: SystemB) -> PipeSystem; } impl IntoPipeSystem for SystemA where SystemA: IntoSystem<(), Payload, ParamA>, SystemB: IntoSystem, { fn pipe(self, system: SystemB) -> PipeSystem { let system_a = IntoSystem::into_system(self); let system_b = IntoSystem::into_system(system); let name = format!("Pipe({}, {})", system_a.name(), system_b.name()); PipeSystem::new(system_a, system_b, Cow::Owned(name)) } } /// A collection of common adapters for [piping](super::PipeSystem) the result of a system. pub mod adapter { use crate::system::In; use bevy_utils::tracing; use std::fmt::Debug; /// Converts a regular function into a system adapter. /// /// # Examples /// ``` /// use bevy_ecs::prelude::*; /// /// fn return1() -> u64 { 1 } /// /// return1 /// .pipe(system_adapter::new(u32::try_from)) /// .pipe(system_adapter::unwrap) /// .pipe(print); /// /// fn print(In(x): In) { /// println!("{x:?}"); /// } /// ``` pub fn new(mut f: impl FnMut(T) -> U) -> impl FnMut(In) -> U { move |In(x)| f(x) } /// System adapter that unwraps the `Ok` variant of a [`Result`]. /// This is useful for fallible systems that should panic in the case of an error. /// /// There is no equivalent adapter for [`Option`]. Instead, it's best to provide /// an error message and convert to a `Result` using `ok_or{_else}`. /// /// # Examples /// /// Panicking on error /// /// ``` /// use bevy_ecs::prelude::*; /// /// // Building a new schedule/app... /// let mut sched = Schedule::default(); /// sched.add_system( /// // Panic if the load system returns an error. /// load_save_system.pipe(system_adapter::unwrap) /// ) /// // ... /// # ; /// # let mut world = World::new(); /// # sched.run(&mut world); /// /// // A system which may fail irreparably. /// fn load_save_system() -> Result<(), std::io::Error> { /// let save_file = open_file("my_save.json")?; /// dbg!(save_file); /// Ok(()) /// } /// # fn open_file(name: &str) -> Result<&'static str, std::io::Error> /// # { Ok("hello world") } /// ``` pub fn unwrap(In(res): In>) -> T { res.unwrap() } /// System adapter that utilizes the [`bevy_utils::tracing::info!`] macro to print system information. /// /// # Examples /// /// ``` /// use bevy_ecs::prelude::*; /// /// // Building a new schedule/app... /// let mut sched = Schedule::default(); /// sched.add_system( /// // Prints system information. /// data_pipe_system.pipe(system_adapter::info) /// ) /// // ... /// # ; /// # let mut world = World::new(); /// # sched.run(&mut world); /// /// // A system that returns a String output. /// fn data_pipe_system() -> String { /// "42".to_string() /// } /// ``` pub fn info(In(data): In) { tracing::info!("{:?}", data); } /// System adapter that utilizes the [`bevy_utils::tracing::debug!`] macro to print the output of a system. /// /// # Examples /// /// ``` /// use bevy_ecs::prelude::*; /// /// // Building a new schedule/app... /// let mut sched = Schedule::default(); /// sched.add_system( /// // Prints debug data from system. /// parse_message_system.pipe(system_adapter::dbg) /// ) /// // ... /// # ; /// # let mut world = World::new(); /// # sched.run(&mut world); /// /// // A system that returns a Result output. /// fn parse_message_system() -> Result { /// Ok("42".parse()?) /// } /// ``` pub fn dbg(In(data): In) { tracing::debug!("{:?}", data); } /// System adapter that utilizes the [`bevy_utils::tracing::warn!`] macro to print the output of a system. /// /// # Examples /// /// ``` /// use bevy_ecs::prelude::*; /// /// // Building a new schedule/app... /// # let mut sched = Schedule::default(); /// sched.add_system( /// // Prints system warning if system returns an error. /// warning_pipe_system.pipe(system_adapter::warn) /// ) /// // ... /// # ; /// # let mut world = World::new(); /// # sched.run(&mut world); /// /// // A system that returns a Result<(), String> output. /// fn warning_pipe_system() -> Result<(), String> { /// Err("Got to rusty?".to_string()) /// } /// ``` pub fn warn(In(res): In>) { if let Err(warn) = res { tracing::warn!("{:?}", warn); } } /// System adapter that utilizes the [`bevy_utils::tracing::error!`] macro to print the output of a system. /// /// # Examples /// /// ``` /// use bevy_ecs::prelude::*; /// // Building a new schedule/app... /// let mut sched = Schedule::default(); /// sched.add_system( /// // Prints system error if system fails. /// parse_error_message_system.pipe(system_adapter::error) /// ) /// // ... /// # ; /// # let mut world = World::new(); /// # sched.run(&mut world); /// /// // A system that returns a Result<())> output. /// fn parse_error_message_system() -> Result<(), String> { /// Err("Some error".to_owned()) /// } /// ``` pub fn error(In(res): In>) { if let Err(error) = res { tracing::error!("{:?}", error); } } /// System adapter that ignores the output of the previous system in a pipe. /// This is useful for fallible systems that should simply return early in case of an `Err`/`None`. /// /// # Examples /// /// Returning early /// /// ``` /// use bevy_ecs::prelude::*; /// /// // Marker component for an enemy entity. /// #[derive(Component)] /// struct Monster; /// /// // Building a new schedule/app... /// # let mut sched = Schedule::default(); sched /// .add_system( /// // If the system fails, just move on and try again next frame. /// fallible_system.pipe(system_adapter::ignore) /// ) /// // ... /// # ; /// # let mut world = World::new(); /// # sched.run(&mut world); /// /// // A system which may return early. It's more convenient to use the `?` operator for this. /// fn fallible_system( /// q: Query> /// ) -> Option<()> { /// let monster_id = q.iter().next()?; /// println!("Monster entity is {monster_id:?}"); /// Some(()) /// } /// ``` pub fn ignore(In(_): In) {} #[cfg(test)] #[test] fn assert_systems() { use std::str::FromStr; use crate::{prelude::*, system::assert_is_system}; /// Mocks a system that returns a value of type `T`. fn returning() -> T { unimplemented!() } /// Mocks an exclusive system that takes an input and returns an output. fn exclusive_in_out(_: In, _: &mut World) -> B { unimplemented!() } fn not(In(val): In) -> bool { !val } assert_is_system(returning::>.pipe(unwrap)); assert_is_system(returning::>.pipe(ignore)); assert_is_system(returning::<&str>.pipe(new(u64::from_str)).pipe(unwrap)); assert_is_system(exclusive_in_out::<(), Result<(), std::io::Error>>.pipe(error)); assert_is_system(returning::.pipe(exclusive_in_out::)); returning::<()>.run_if(returning::.pipe(not)); } }