Initial Commit
Signed-off-by: Marcel Müller <neikos@neikos.email>
This commit is contained in:
commit
9ff1cf1bac
12 changed files with 615 additions and 0 deletions
1
.envrc
Normal file
1
.envrc
Normal file
|
|
@ -0,0 +1 @@
|
|||
use flake
|
||||
4
.gitignore
vendored
Normal file
4
.gitignore
vendored
Normal file
|
|
@ -0,0 +1,4 @@
|
|||
.direnv
|
||||
|
||||
/target
|
||||
/result*
|
||||
14
Cargo.lock
generated
Normal file
14
Cargo.lock
generated
Normal file
|
|
@ -0,0 +1,14 @@
|
|||
# This file is automatically @generated by Cargo.
|
||||
# It is not intended for manual editing.
|
||||
version = 4
|
||||
|
||||
[[package]]
|
||||
name = "tytix"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"tytix-core",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tytix-core"
|
||||
version = "0.1.0"
|
||||
11
Cargo.toml
Normal file
11
Cargo.toml
Normal file
|
|
@ -0,0 +1,11 @@
|
|||
[workspace]
|
||||
|
||||
members = ["crates/*"]
|
||||
|
||||
resolver = "3"
|
||||
|
||||
[workspace.package]
|
||||
edition = "2024"
|
||||
version = "0.1.0"
|
||||
license = "EUPL-1.2"
|
||||
authors = ["Marcel Müller <neikos@neikos.email>"]
|
||||
17
crates/tytix-core/Cargo.toml
Normal file
17
crates/tytix-core/Cargo.toml
Normal file
|
|
@ -0,0 +1,17 @@
|
|||
[package]
|
||||
name = "tytix-core"
|
||||
version.workspace = true
|
||||
edition.workspace = true
|
||||
authors.workspace = true
|
||||
|
||||
description = "A clear and simple to use actor framework"
|
||||
repository = "https://github.com/TheNeikos/tytix"
|
||||
license = "EUPL-1.2"
|
||||
keywords = []
|
||||
categories = []
|
||||
|
||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||
|
||||
[dependencies]
|
||||
|
||||
[dev-dependencies]
|
||||
235
crates/tytix-core/src/lib.rs
Normal file
235
crates/tytix-core/src/lib.rs
Normal file
|
|
@ -0,0 +1,235 @@
|
|||
#![feature(const_cmp)]
|
||||
#![feature(const_trait_impl)]
|
||||
|
||||
use std::any::Any;
|
||||
use std::any::TypeId;
|
||||
|
||||
pub trait Message: Send + Any {
|
||||
type Reply: Send + Any;
|
||||
}
|
||||
|
||||
pub trait MessageIdentifier {
|
||||
const IDENT: BundleChain;
|
||||
}
|
||||
|
||||
impl<M: Message> MessageIdentifier for M {
|
||||
const IDENT: BundleChain = BundleChain::of::<M>();
|
||||
}
|
||||
|
||||
pub struct InternalMessage(Box<dyn std::any::Any>);
|
||||
|
||||
impl InternalMessage {
|
||||
pub fn new(message: impl Message) -> InternalMessage {
|
||||
InternalMessage(Box::new(message))
|
||||
}
|
||||
|
||||
pub fn into_inner<M: Message>(self) -> Result<M, InternalMessage> {
|
||||
self.0.downcast().map(|v| *v).map_err(InternalMessage)
|
||||
}
|
||||
}
|
||||
|
||||
pub trait Address<MB>
|
||||
where
|
||||
MB: MessageBundle,
|
||||
{
|
||||
fn send<M: Message>(&mut self, message: M);
|
||||
}
|
||||
|
||||
pub trait InternalMessageHandler {
|
||||
type HandledMessages: MessageBundle;
|
||||
fn handle_message(&mut self, msg: InternalMessage);
|
||||
}
|
||||
|
||||
impl<MB, IMH> Address<MB> for IMH
|
||||
where
|
||||
MB: MessageBundle,
|
||||
IMH: InternalMessageHandler<HandledMessages = MB>,
|
||||
{
|
||||
fn send<M: Message>(&mut self, message: M) {
|
||||
const {
|
||||
let true = <M as IsContainedInBundle<MB>>::IS_CONTAINED else {
|
||||
panic!("Message is not contained in MessageBundle",);
|
||||
};
|
||||
}
|
||||
|
||||
self.handle_message(InternalMessage(Box::new(message)));
|
||||
}
|
||||
}
|
||||
|
||||
pub trait MessageReceiver {}
|
||||
|
||||
pub trait MessageBundle {
|
||||
const IDS: BundleChain;
|
||||
}
|
||||
|
||||
impl MessageBundle for () {
|
||||
const IDS: BundleChain = BundleChain {
|
||||
next: None,
|
||||
op: BundleOp::Remove(TypeId::of::<()>()),
|
||||
};
|
||||
}
|
||||
|
||||
impl<M: Message> MessageBundle for (M,) {
|
||||
const IDS: BundleChain = BundleChain::of::<M>();
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Copy)]
|
||||
pub struct BundleChain {
|
||||
op: BundleOp,
|
||||
next: Option<&'static BundleChain>,
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Copy)]
|
||||
pub enum BundleOp {
|
||||
Add(TypeId),
|
||||
Remove(TypeId),
|
||||
}
|
||||
|
||||
impl BundleChain {
|
||||
pub const fn of<M: Message>() -> BundleChain {
|
||||
BundleChain {
|
||||
op: BundleOp::Add(TypeId::of::<M>()),
|
||||
next: None,
|
||||
}
|
||||
}
|
||||
|
||||
pub const fn with<M: Message>(&'static self) -> BundleChain {
|
||||
add_to_chain(self, TypeId::of::<M>())
|
||||
}
|
||||
|
||||
pub const fn without<M: Message>(&'static self) -> BundleChain {
|
||||
remove_from_chain(self, TypeId::of::<M>())
|
||||
}
|
||||
}
|
||||
|
||||
impl<A, B> MessageBundle for (A, B)
|
||||
where
|
||||
A: Message,
|
||||
B: Message,
|
||||
{
|
||||
const IDS: BundleChain = BundleChain::of::<A>().with::<B>();
|
||||
}
|
||||
|
||||
impl<A, B, C, D> MessageBundle for (A, B, C, D)
|
||||
where
|
||||
A: Message,
|
||||
B: Message,
|
||||
C: Message,
|
||||
D: Message,
|
||||
{
|
||||
const IDS: BundleChain = BundleChain::of::<A>().with::<B>().with::<C>().with::<D>();
|
||||
}
|
||||
|
||||
pub trait IsContainedInBundle<MB> {
|
||||
const IS_CONTAINED: bool;
|
||||
}
|
||||
|
||||
impl<M, MB> IsContainedInBundle<MB> for M
|
||||
where
|
||||
M: Message,
|
||||
MB: MessageBundle,
|
||||
{
|
||||
const IS_CONTAINED: bool = check_is_contained(&MB::IDS, TypeId::of::<M>());
|
||||
}
|
||||
|
||||
const fn check_is_contained(ids: &'static BundleChain, id: TypeId) -> bool {
|
||||
match ids.op {
|
||||
BundleOp::Add(added_id) => {
|
||||
if check_type_id_equal(added_id, id) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
BundleOp::Remove(removed_id) => {
|
||||
if check_type_id_equal(removed_id, id) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if let Some(next) = ids.next {
|
||||
check_is_contained(next, id)
|
||||
} else {
|
||||
false
|
||||
}
|
||||
}
|
||||
|
||||
const fn check_type_id_equal(left: TypeId, right: TypeId) -> bool {
|
||||
left == right
|
||||
}
|
||||
|
||||
const fn add_to_chain(prev: &'static BundleChain, to_add: TypeId) -> BundleChain {
|
||||
BundleChain {
|
||||
op: BundleOp::Add(to_add),
|
||||
next: Some(prev),
|
||||
}
|
||||
}
|
||||
|
||||
const fn remove_from_chain(prev: &'static BundleChain, to_remove: TypeId) -> BundleChain {
|
||||
BundleChain {
|
||||
op: BundleOp::Remove(to_remove),
|
||||
next: Some(prev),
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
|
||||
struct Foo;
|
||||
|
||||
struct Bar;
|
||||
|
||||
impl Message for Foo {
|
||||
type Reply = ();
|
||||
}
|
||||
|
||||
impl Message for Bar {
|
||||
type Reply = ();
|
||||
}
|
||||
|
||||
struct Zap;
|
||||
|
||||
impl Message for Zap {
|
||||
type Reply = ();
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn check_checker() {
|
||||
assert!(check_type_id_equal(
|
||||
TypeId::of::<bool>(),
|
||||
TypeId::of::<bool>()
|
||||
));
|
||||
|
||||
const BUNDLE: BundleChain = BundleChain::of::<Foo>().with::<Zap>();
|
||||
|
||||
assert!(check_is_contained(&BUNDLE, TypeId::of::<Foo>()));
|
||||
assert!(!check_is_contained(&BUNDLE, TypeId::of::<Bar>()));
|
||||
|
||||
const WITHOUT_BUNDLE: BundleChain = BundleChain::of::<Foo>().with::<Zap>().without::<Zap>();
|
||||
|
||||
assert!(!check_is_contained(&WITHOUT_BUNDLE, TypeId::of::<Zap>()));
|
||||
|
||||
const WITHOUT_WITH_BUNDLE: BundleChain =
|
||||
BundleChain::of::<Foo>().without::<Zap>().with::<Zap>();
|
||||
|
||||
assert!(check_is_contained(
|
||||
&WITHOUT_WITH_BUNDLE,
|
||||
TypeId::of::<Zap>()
|
||||
));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn check_sending_messages() {
|
||||
struct Sender;
|
||||
|
||||
impl InternalMessageHandler for Sender {
|
||||
type HandledMessages = (Foo, Bar);
|
||||
|
||||
fn handle_message(&mut self, _msg: InternalMessage) {}
|
||||
}
|
||||
|
||||
let mut s = Sender;
|
||||
|
||||
s.send(Foo);
|
||||
}
|
||||
}
|
||||
12
crates/tytix/Cargo.toml
Normal file
12
crates/tytix/Cargo.toml
Normal file
|
|
@ -0,0 +1,12 @@
|
|||
[package]
|
||||
name = "tytix"
|
||||
version.workspace = true
|
||||
edition.workspace = true
|
||||
authors.workspace = true
|
||||
|
||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||
|
||||
[dependencies]
|
||||
tytix-core = { path = "../tytix-core/" }
|
||||
|
||||
[dev-dependencies]
|
||||
123
crates/tytix/src/lib.rs
Normal file
123
crates/tytix/src/lib.rs
Normal file
|
|
@ -0,0 +1,123 @@
|
|||
use std::marker::PhantomData;
|
||||
|
||||
use tytix_core::Address;
|
||||
use tytix_core::InternalMessageHandler;
|
||||
use tytix_core::IsContainedInBundle;
|
||||
use tytix_core::Message;
|
||||
use tytix_core::MessageBundle;
|
||||
|
||||
pub trait AddressExt<MB> {
|
||||
fn map_before<F, M, R>(self, f: F) -> MappedBeforeAddress<Self, F, M, R>
|
||||
where
|
||||
MB: MessageBundle,
|
||||
F: Fn(M) -> R + 'static,
|
||||
M: Message,
|
||||
R: Message + IsContainedInBundle<MB>,
|
||||
Self: Sized;
|
||||
}
|
||||
|
||||
impl<MB: MessageBundle, A: Address<MB>> AddressExt<MB> for A {
|
||||
fn map_before<F, M, R>(self, f: F) -> MappedBeforeAddress<Self, F, M, R>
|
||||
where
|
||||
MB: MessageBundle,
|
||||
F: Fn(M) -> R + 'static,
|
||||
M: Message,
|
||||
R: Message + IsContainedInBundle<MB>,
|
||||
Self: Sized,
|
||||
{
|
||||
const {
|
||||
let true = <R as IsContainedInBundle<MB>>::IS_CONTAINED else {
|
||||
panic!("Message is not contained in MessageBundle",);
|
||||
};
|
||||
}
|
||||
|
||||
MappedBeforeAddress {
|
||||
address: self,
|
||||
func: f,
|
||||
_pd: PhantomData,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub struct MappedBeforeAddress<A, F, M, R> {
|
||||
address: A,
|
||||
func: F,
|
||||
_pd: PhantomData<fn(M) -> R>,
|
||||
}
|
||||
|
||||
pub struct MappedHandledMessages<MB, M, R> {
|
||||
_pd: PhantomData<fn(MB, M) -> R>,
|
||||
}
|
||||
|
||||
impl<MB, M, R> MessageBundle for MappedHandledMessages<MB, M, R>
|
||||
where
|
||||
MB: MessageBundle,
|
||||
M: Message,
|
||||
R: Message + IsContainedInBundle<MB>,
|
||||
{
|
||||
const IDS: tytix_core::BundleChain = MB::IDS.without::<R>().with::<M>();
|
||||
}
|
||||
|
||||
impl<A, F, M, R> InternalMessageHandler for MappedBeforeAddress<A, F, M, R>
|
||||
where
|
||||
A: InternalMessageHandler,
|
||||
M: Message,
|
||||
R: Message + IsContainedInBundle<A::HandledMessages>,
|
||||
F: FnMut(M) -> R,
|
||||
{
|
||||
type HandledMessages = MappedHandledMessages<A::HandledMessages, M, R>;
|
||||
|
||||
fn handle_message(&mut self, msg: tytix_core::InternalMessage) {
|
||||
match msg.into_inner::<M>() {
|
||||
Ok(removed_message) => {
|
||||
let new_message = (self.func)(removed_message);
|
||||
|
||||
self.address.send(new_message);
|
||||
}
|
||||
Err(other_message) => self.address.handle_message(other_message),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use std::sync::OnceLock;
|
||||
|
||||
use super::*;
|
||||
|
||||
struct Foo;
|
||||
|
||||
impl Message for Foo {
|
||||
type Reply = ();
|
||||
}
|
||||
|
||||
struct Bar;
|
||||
|
||||
impl Message for Bar {
|
||||
type Reply = ();
|
||||
}
|
||||
|
||||
struct SimpleAddress;
|
||||
|
||||
impl InternalMessageHandler for SimpleAddress {
|
||||
type HandledMessages = (Foo,);
|
||||
|
||||
fn handle_message(&mut self, msg: tytix_core::InternalMessage) {
|
||||
drop(msg);
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn check_mapping() {
|
||||
static MSG: OnceLock<bool> = OnceLock::new();
|
||||
|
||||
let mut sa = SimpleAddress.map_before(|_b: Bar| {
|
||||
let _ = MSG.set(true);
|
||||
Foo
|
||||
});
|
||||
|
||||
sa.send(Bar);
|
||||
|
||||
MSG.get().expect("The message was mapped!");
|
||||
}
|
||||
}
|
||||
97
flake.lock
generated
Normal file
97
flake.lock
generated
Normal file
|
|
@ -0,0 +1,97 @@
|
|||
{
|
||||
"nodes": {
|
||||
"crane": {
|
||||
"locked": {
|
||||
"lastModified": 1736101677,
|
||||
"narHash": "sha256-iKOPq86AOWCohuzxwFy/MtC8PcSVGnrxBOvxpjpzrAY=",
|
||||
"owner": "ipetkov",
|
||||
"repo": "crane",
|
||||
"rev": "61ba163d85e5adeddc7b3a69bb174034965965b2",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"owner": "ipetkov",
|
||||
"repo": "crane",
|
||||
"type": "github"
|
||||
}
|
||||
},
|
||||
"flake-utils": {
|
||||
"inputs": {
|
||||
"systems": "systems"
|
||||
},
|
||||
"locked": {
|
||||
"lastModified": 1731533236,
|
||||
"narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=",
|
||||
"owner": "numtide",
|
||||
"repo": "flake-utils",
|
||||
"rev": "11707dc2f618dd54ca8739b309ec4fc024de578b",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"owner": "numtide",
|
||||
"repo": "flake-utils",
|
||||
"type": "github"
|
||||
}
|
||||
},
|
||||
"nixpkgs": {
|
||||
"locked": {
|
||||
"lastModified": 1736200483,
|
||||
"narHash": "sha256-JO+lFN2HsCwSLMUWXHeOad6QUxOuwe9UOAF/iSl1J4I=",
|
||||
"owner": "NixOS",
|
||||
"repo": "nixpkgs",
|
||||
"rev": "3f0a8ac25fb674611b98089ca3a5dd6480175751",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"id": "nixpkgs",
|
||||
"ref": "nixos-24.11",
|
||||
"type": "indirect"
|
||||
}
|
||||
},
|
||||
"root": {
|
||||
"inputs": {
|
||||
"crane": "crane",
|
||||
"flake-utils": "flake-utils",
|
||||
"nixpkgs": "nixpkgs",
|
||||
"rust-overlay": "rust-overlay"
|
||||
}
|
||||
},
|
||||
"rust-overlay": {
|
||||
"inputs": {
|
||||
"nixpkgs": [
|
||||
"nixpkgs"
|
||||
]
|
||||
},
|
||||
"locked": {
|
||||
"lastModified": 1762051177,
|
||||
"narHash": "sha256-pESNTx/m3WnrYx+OujBtDP5Bj0/mAyHa4MgEwzkgkLE=",
|
||||
"owner": "oxalica",
|
||||
"repo": "rust-overlay",
|
||||
"rev": "08c33e87c4829bbdd42b5af247cf7a19e126369f",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"owner": "oxalica",
|
||||
"repo": "rust-overlay",
|
||||
"type": "github"
|
||||
}
|
||||
},
|
||||
"systems": {
|
||||
"locked": {
|
||||
"lastModified": 1681028828,
|
||||
"narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=",
|
||||
"owner": "nix-systems",
|
||||
"repo": "default",
|
||||
"rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"owner": "nix-systems",
|
||||
"repo": "default",
|
||||
"type": "github"
|
||||
}
|
||||
}
|
||||
},
|
||||
"root": "root",
|
||||
"version": 7
|
||||
}
|
||||
96
flake.nix
Normal file
96
flake.nix
Normal file
|
|
@ -0,0 +1,96 @@
|
|||
{
|
||||
description = "The Theater Rust library";
|
||||
inputs = {
|
||||
nixpkgs.url = "nixpkgs/nixos-24.11";
|
||||
flake-utils = {
|
||||
url = "github:numtide/flake-utils";
|
||||
};
|
||||
crane = {
|
||||
url = "github:ipetkov/crane";
|
||||
};
|
||||
rust-overlay = {
|
||||
url = "github:oxalica/rust-overlay";
|
||||
inputs = {
|
||||
nixpkgs.follows = "nixpkgs";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
outputs = { nixpkgs, crane, flake-utils, rust-overlay, ... }:
|
||||
flake-utils.lib.eachDefaultSystem (system:
|
||||
let
|
||||
pkgs = import nixpkgs {
|
||||
inherit system;
|
||||
overlays = [ (import rust-overlay) ];
|
||||
};
|
||||
|
||||
rustTarget = pkgs.rust-bin.fromRustupToolchainFile ./rust-toolchain.toml;
|
||||
unstableRustTarget = pkgs.rust-bin.selectLatestNightlyWith (toolchain: toolchain.default.override {
|
||||
extensions = [ "rust-src" "miri" "rustfmt" ];
|
||||
});
|
||||
craneLib = (crane.mkLib pkgs).overrideToolchain rustTarget;
|
||||
unstableCraneLib = (crane.mkLib pkgs).overrideToolchain unstableRustTarget;
|
||||
|
||||
tomlInfo = craneLib.crateNameFromCargoToml { cargoToml = ./Cargo.toml; };
|
||||
inherit (tomlInfo) version;
|
||||
src = ./.;
|
||||
|
||||
rustfmt' = pkgs.writeShellScriptBin "rustfmt" ''
|
||||
exec "${unstableRustTarget}/bin/rustfmt" "$@"
|
||||
'';
|
||||
|
||||
cargoArtifacts = craneLib.buildDepsOnly {
|
||||
inherit src;
|
||||
cargoExtraArgs = "--all-features --all";
|
||||
};
|
||||
|
||||
cloudmqtt = craneLib.buildPackage {
|
||||
inherit cargoArtifacts src version;
|
||||
cargoExtraArgs = "--all-features --all";
|
||||
};
|
||||
|
||||
in
|
||||
rec {
|
||||
checks = {
|
||||
inherit cloudmqtt;
|
||||
|
||||
cloudmqtt-clippy = craneLib.cargoClippy {
|
||||
inherit cargoArtifacts src;
|
||||
cargoExtraArgs = "--all --all-features";
|
||||
cargoClippyExtraArgs = "-- --deny warnings";
|
||||
};
|
||||
|
||||
cloudmqtt-fmt = unstableCraneLib.cargoFmt {
|
||||
inherit src;
|
||||
};
|
||||
};
|
||||
|
||||
packages.cloudmqtt = cloudmqtt;
|
||||
packages.default = packages.cloudmqtt;
|
||||
|
||||
apps.cloudmqtt = flake-utils.lib.mkApp {
|
||||
name = "cloudmqtt";
|
||||
drv = cloudmqtt;
|
||||
};
|
||||
apps.default = apps.cloudmqtt;
|
||||
|
||||
devShells.default = devShells.cloudmqtt;
|
||||
devShells.cloudmqtt = pkgs.mkShell {
|
||||
buildInputs = [ ];
|
||||
|
||||
nativeBuildInputs = [
|
||||
rustfmt'
|
||||
unstableRustTarget
|
||||
|
||||
pkgs.cargo-msrv
|
||||
pkgs.cargo-deny
|
||||
pkgs.cargo-expand
|
||||
pkgs.cargo-bloat
|
||||
pkgs.cargo-fuzz
|
||||
|
||||
pkgs.gitlint
|
||||
];
|
||||
};
|
||||
}
|
||||
);
|
||||
}
|
||||
2
rust-toolchain.toml
Normal file
2
rust-toolchain.toml
Normal file
|
|
@ -0,0 +1,2 @@
|
|||
[toolchain]
|
||||
channel = "1.91.0"
|
||||
3
rustfmt.toml
Normal file
3
rustfmt.toml
Normal file
|
|
@ -0,0 +1,3 @@
|
|||
imports_granularity = "Item"
|
||||
reorder_imports = true
|
||||
group_imports = "StdExternalCrate"
|
||||
Loading…
Add table
Add a link
Reference in a new issue