Asyncronous Parallel Entity Component System
apecs
is an entity-component system written in Rust that supports traditional syncronous
systems as well as asyncronous systems that can evolve over time. This makes it great for
general applications, quick game prototypes, DIY engines and any simulation that has discrete
steps.
specs
, bevy_ecs
, hecs
, legion
, shipyard
, planck_ecs
rust
use apecs::{anyhow, Write, world::*, system::*};
let mut world = World::default();
world
.with_system("demo", |mut u32_number: Write<u32>| -> anyhow::Result<ShouldContinue> {
*u32_number += 1;
if *u32_number == 3 {
end()
} else {
ok()
}
})
.unwrap();
world.run();
assert_eq!(3, *world.resource::<u32>().unwrap());
Write
and Read
will create default resources
during fetching, and WriteExpect
and ReadExpect
will not.rust
use apecs::{anyhow, Write, world::*};
async fn demo(mut facade: Facade) -> anyhow::Result<()> {
loop {
let mut u32_number: Write<u32> = facade.fetch().await?;
*u32_number += 1;
if *u32_number > 5 {
break;
}
}
Ok(())
}
let mut world = World::default();
world
.with_async_system("demo", demo);
world.run();
assert_eq!(6, *world.resource::<u32>().unwrap());
rust
use apecs::world::World;
let mut world = World::default();
world
.with_async(async {
log::trace!("hello");
});
world.run();
fetch data (system data) derive macros ```rust use apecs::{CanFetch, Read, Write, world::*};
struct MyData {
entities: Read
let mut world = World::default(); let mut mydata: MyData = world.fetch().unwrap(); *mydata.u32_number = 1; ```
system scheduling
fn one(mut u32number: Write
fn two(mut u32number: Write
fn thrice(mut f32number: Write
fn lastly((u32number, f32number): (Read
let mut world = World::default(); world .withsystem("one", one).unwrap() .withsystemwithdependencies("two", two, &["one"]).unwrap() .withsystem("thrice", thrice).unwrap() .withsystembarrier() .withsystem("lastly", lastly).unwrap();
asserteq!( vec![ vec!["one", "thrice"], vec!["two"], vec!["lastly"], ], world.getsyncschedulenames() );
world.tick(); asserteq!( vec![ vec!["thrice"], vec!["lastly"], ], world.getsyncschedulenames() );
world.tick(); world.tick(); assert!(world.getsyncschedulenames().isempty()); ```
component storage
// Make a type for tracking changes
struct MyTracker(u64);
// Entities and ArchetypeSet (which stores components) are default
// resources
let mut world = World::default();
world
.withsystem("create", |mut entities: Write
asserteq!( vec![ vec!["create"], vec!["progress"], vec!["sync"], ], world.getsyncschedulenames() );
world.tick(); // entities are created, components applied lazily world.tick(); // f32s are modified, u32s and strings are synced world.tick(); // f32s are modified, u32s and strings are synced
let qbundle: Query<(&f32, &u32, &String)> = world.fetch().unwrap(); asserteq!( (2.0f32, 2u32, "13:2".tostring()), qbundle.query().findone(13).map(|(f, u, s)| (*f, *u, s.tostring())).unwrap() ); ```
rust
use apecs::{system::*, world::*, Write, Read};
let mut world = World::default();
world
.with_system("one", |mut f32_number: Write<f32>| {
*f32_number += 1.0;
ok()
}).unwrap()
.with_system("two", |f32_number: Read<f32>| {
println!("system two reads {}", *f32_number);
ok()
}).unwrap()
.with_system("three", |f32_number: Read<f32>| {
println!("system three reads {}", *f32_number);
ok()
}).unwrap()
.with_parallelism(Parallelism::Automatic);
world.tick();
plugins (groups of systems, resources and sub-plugins) ```rust use apecs::{plugins::, storage::, system::, world::, CanFetch, Write};
struct MyTracker(u64);
struct SyncData {
qdirtyf32s: Query<(&'static f32, &'static mut String)>,
tracker: Write
fn create(
(mut entities, mut archeset): (Write
fn progress(q: Query<&mut f32>) -> anyhow::Result
fn sync(mut data: SyncData) -> anyhow::Result
// now we can package it all up into a plugin let myplugin = Plugin::default() .withsystem("create", create, &[]) .withsystem("progress", progress, &["create"]) .withsystem("sync", sync, &[]);
let mut world = World::default(); world.withplugin(myplugin).unwrap(); world.tick(); ```
bash
cargo test
wasm-pack test --firefox crates/apecs
I like firefox, but you can use different browsers for the wasm tests. For the most part they're there just to make sure apecs works on wasm.
The apecs
benchmarks measure itself against my favorite ECS libs:
specs
, bevy
, hecs
, legion
, shipyard
and planck_ecs
.
bash
cargo bench -p benchmarks
apecs
uses generic associated types. This means it can only be compiled with nightly.