Add FromHandler command

This commit is contained in:
Joscha 2024-12-29 00:45:52 +01:00
parent 3e93e1192b
commit 3571a97fb2
2 changed files with 59 additions and 26 deletions

View file

@ -1,29 +1,16 @@
use std::time::Duration;
use async_trait::async_trait;
use euphoxide::api::Message;
use euphoxide_bot::{
basic::FromHandler,
botrulez::{FullHelp, Ping, ShortHelp},
Command, CommandExt, Commands, Context, Info, Propagate,
CommandExt, Commands, Context, Propagate,
};
use euphoxide_client::MultiClient;
use log::error;
use tokio::sync::mpsc;
struct Pyramid;
#[async_trait]
impl Command for Pyramid {
fn info(&self, _ctx: &Context) -> Info {
Info::new().with_description("build a pyramid")
}
async fn execute(
&self,
_arg: &str,
msg: &Message,
ctx: &Context,
) -> euphoxide::Result<Propagate> {
async fn pyramid(_arg: &str, msg: &Message, ctx: &Context) -> euphoxide::Result<Propagate> {
let mut parent = msg.id;
for _ in 0..3 {
@ -36,7 +23,6 @@ impl Command for Pyramid {
ctx.reply_only(parent, "brick").await?;
Ok(Propagate::No)
}
}
#[tokio::main]
async fn main() {
@ -56,7 +42,12 @@ async fn main() {
.specific("help")
.hidden(),
)
.then(Pyramid.general("pyramid"))
.then(
FromHandler::new(pyramid)
.described()
.with_description("build a pyramid")
.general("pyramid"),
)
.build();
let clients = MultiClient::new(event_tx);

View file

@ -1,5 +1,7 @@
//! Basic command wrappers.
use std::future::Future;
use async_trait::async_trait;
use euphoxide::api::Message;
@ -97,3 +99,43 @@ where
}
}
}
// Black type magic, thanks a lot to https://github.com/kpreid and the
// async_fn_traits crate!
// TODO Simplify all this once AsyncFn becomes stable
pub trait HandlerFn<'a0, 'a1, 'a2, E>:
Fn(&'a0 str, &'a1 Message, &'a2 Context<E>) -> Self::Future
where
E: 'a2,
{
type Future: Future<Output = Result<Propagate, E>> + Send;
}
impl<'a0, 'a1, 'a2, E, F, Fut> HandlerFn<'a0, 'a1, 'a2, E> for F
where
E: 'a2,
F: Fn(&'a0 str, &'a1 Message, &'a2 Context<E>) -> Fut + ?Sized,
Fut: Future<Output = Result<Propagate, E>> + Send,
{
type Future = Fut;
}
pub struct FromHandler<F>(F);
impl<F> FromHandler<F> {
pub fn new(f: F) -> Self {
Self(f)
}
}
#[async_trait]
impl<E, F> Command<E> for FromHandler<F>
where
F: for<'a0, 'a1, 'a2> HandlerFn<'a0, 'a1, 'a2, E> + Sync,
{
async fn execute(&self, arg: &str, msg: &Message, ctx: &Context<E>) -> Result<Propagate, E> {
(self.0)(arg, msg, ctx).await
}
}