funny crate do funny thing

This commit is contained in:
mzntori 2024-03-09 01:36:47 +01:00
parent 3826ada069
commit 803289d754
7 changed files with 160 additions and 43 deletions

10
Cargo.toml Normal file
View file

@ -0,0 +1,10 @@
[package]
name = "twitchbot-rs"
version = "0.1.0"
edition = "2021"
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
[dependencies]
twitch-irc = "5.0.1"
tokio = { version = "1.33.0", features = ["full"] }

View file

@ -1,7 +1,7 @@
use std::collections::HashMap;
use crate::command::Command;
use std::sync::Arc;
use std::sync::{Arc, Mutex as StdMutex};
use tokio::sync::mpsc::UnboundedReceiver;
use tokio::sync::Mutex;
@ -14,7 +14,6 @@ use twitch_irc::{
};
type IncomingMessages = UnboundedReceiver<ServerMessage>;
pub type Client = TwitchIRCClient<SecureTCPTransport, StaticLoginCredentials>;
/// rename types wrapped in `Arc<Mutex<...>>` to their respective name ending with `AM`
@ -22,16 +21,16 @@ pub type Client = TwitchIRCClient<SecureTCPTransport, StaticLoginCredentials>;
pub type ClientAM = Arc<Mutex<Client>>;
#[derive(Debug)]
pub struct Bot<'a, P> {
username: &'a str,
oauth_token: &'a str,
payload: Arc<Mutex<P>>,
commands: Arc<Mutex<dyn Command<CommandPayLoad=P>>>
payload: Arc<StdMutex<P>>,
commands: Arc<Mutex<HashMap<String, Box<dyn Command<CommandPayLoad=P>>>>>,
}
impl<'a, P> Bot<'a, P>
where P: Send + 'static
{
/// Return a message stream and a client wrapped in `Arc<Mutex<...>>`
fn incoming_messages_and_client(&self) -> (IncomingMessages, ClientAM) {
@ -59,12 +58,12 @@ impl<'a, P> Bot<'a, P>
Bot {
username,
oauth_token,
payload: Arc::new(Mutex::new(payload)),
payload: Arc::new(StdMutex::new(payload)),
commands: Arc::new(Mutex::new(HashMap::new())),
}
}
pub async fn add_command(&mut self, identifier: String, command: impl Command<CommandPayLoad=P>) {
pub async fn add_command(&mut self, identifier: String, command: Box<dyn Command<CommandPayLoad=P>>) {
let mut commands = self.commands.lock().await;
commands.insert(identifier, command);
}
@ -72,6 +71,7 @@ impl<'a, P> Bot<'a, P>
pub async fn run(&'a self) {
// set up client and message stream
let (mut incoming_messages, client_am) = self.incoming_messages_and_client();
let commands = Arc::clone(&self.commands);
let payload = Arc::clone(&self.payload);
let initial_channel = self.username.to_owned();
@ -93,7 +93,13 @@ impl<'a, P> Bot<'a, P>
ServerMessage::Part(_) => {}
ServerMessage::Ping(_) => {}
ServerMessage::Pong(_) => {}
ServerMessage::Privmsg(_) => {}
ServerMessage::Privmsg(msg) => {
let mut cmd = commands.lock().await;
if let Some(command) = cmd.get_mut(&msg.message_text) {
let mut queue = command.execute(Arc::clone(&payload), msg);
queue.execute(Arc::clone(&client_am)).await;
}
}
ServerMessage::Reconnect(_) => {}
ServerMessage::RoomState(_) => {}
ServerMessage::UserNotice(_) => {}

66
src/client_queue.rs Normal file
View file

@ -0,0 +1,66 @@
use twitch_irc::message::ReplyToMessage;
use crate::bot::ClientAM;
pub enum ClientOps {
// channel login
Join(String),
// channel login
Part(String),
// channel login, message
Say(String, String),
// channel login, message
Me(String, String),
// // message
// Reply(String),
// // message
// ReplyMe(String),
}
pub struct ClientQueue {
queue: Vec<ClientOps>,
}
impl ClientQueue {
pub fn new() -> Self {
ClientQueue { queue: vec![] }
}
pub fn join(&mut self, channel: String) {
self.queue.push(ClientOps::Join(channel));
}
pub fn part(&mut self, channel: String) {
self.queue.push(ClientOps::Part(channel));
}
pub fn say(&mut self, channel: String, message: String) {
self.queue.push(ClientOps::Say(channel, message));
}
pub fn me(&mut self, channel: String, message: String) {
self.queue.push(ClientOps::Me(channel, message));
}
// pub fn reply(&mut self, message: String) {
// self.queue.push(ClientOps::Reply(message));
// }
//
// pub fn reply_me(&mut self, message: String) {
// self.queue.push(ClientOps::ReplyMe(message));
// }
pub async fn execute(&self, client_am: ClientAM) {
let client = client_am.lock().await;
for op in self.queue.iter() {
match op {
ClientOps::Join(ch) => { client.join(ch.to_string()).unwrap() }
ClientOps::Part(ch) => { client.part(ch.to_string()) }
ClientOps::Say(ch, msg) => { client.say(ch.to_string(), msg.to_string()).await.unwrap() }
ClientOps::Me(ch, msg) => { client.me(ch.to_string(), msg.to_string()).await.unwrap() }
// ClientOps::Reply(ch) => {}
// ClientOps::ReplyMe(msg) => {}
}
}
}
}

View file

@ -1,14 +1,13 @@
use crate::bot::ClientAM;
use crate::ClientQueue;
use std::sync::Arc;
use std::sync::{Arc, Mutex as StdMutex};
use tokio::sync::Mutex;
use twitch_irc::message::PrivmsgMessage;
pub trait Command {
pub trait Command: Send {
type CommandPayLoad;
async fn execute(&self, pl_am: Arc<Mutex<Self::CommandPayLoad>>, client_am: ClientAM);
fn execute(&self, pl_am: Arc<StdMutex<Self::CommandPayLoad>>, ctx: PrivmsgMessage) -> ClientQueue;
fn help(&self) -> String;
fn info(&self) -> String;
}

0
src/context.rs Normal file
View file

12
src/lib.rs Normal file
View file

@ -0,0 +1,12 @@
pub mod bot;
pub mod command;
pub mod client_queue;
pub mod context;
pub use bot::Bot;
pub use command::Command;
pub use client_queue::ClientQueue;
pub type Context = twitch_irc::message::PrivmsgMessage;

View file

@ -1,6 +1,6 @@
use std::sync::Arc;
use tokio::sync::Mutex;
use twitchbot_rs::{Bot, Command};
use std::sync::{Arc, Mutex as StdMutex};
use twitchbot_rs::{Bot, ClientQueue, Command, Context};
use twitchbot_rs::bot::ClientAM;
@ -9,41 +9,65 @@ pub struct Payload {
}
// pub struct PingCommand;
//
// impl Command for PingCommand {
// type CommandPayLoad = Payload;
//
// async fn execute(&self, pl_am: Arc<Mutex<Self::CommandPayLoad>>, client_am: ClientAM) {
// let client
// }
//
// fn help(&self) -> String {
// todo!()
// }
//
// fn info(&self) -> String {
// todo!()
// }
// }
pub struct PingCommand;
impl Command for PingCommand {
type CommandPayLoad = Payload;
fn execute(&self, pl_am: Arc<StdMutex<Self::CommandPayLoad>>, ctx: Context) -> ClientQueue {
let mut queue = ClientQueue::new();
let pl = pl_am.lock().unwrap();
queue.say("mzntori".to_string(), format!("Pong! {} {}", pl.content, ctx.sender.name));
println!("Pong executed! {}", pl.content);
queue
}
fn help(&self) -> String {
"pongs".to_owned()
}
fn info(&self) -> String {
"do be ponging".to_owned()
}
}
pub struct UrmCommand;
impl Command for UrmCommand {
type CommandPayLoad = Payload;
fn execute(&self, pl_am: Arc<StdMutex<Self::CommandPayLoad>>, ctx: Context) -> ClientQueue {
let mut pl = pl_am.lock().unwrap();
pl.content.push('1');
ClientQueue::new()
}
fn help(&self) -> String {
"pongs".to_owned()
}
fn info(&self) -> String {
"do be ponging".to_owned()
}
}
#[tokio::test]
async fn main() {
// let bot = Bot::new("");
let login = std::env::var("LOGIN").unwrap();
let oauth = std::env::var("OAUTH").unwrap();
let bot = Bot::new(
let mut bot = Bot::new(
login.as_str(),
oauth.as_str(),
Payload { content: String::new() },
);
// bot.add_command()
bot.add_command("!ping".to_owned(), Box::new(PingCommand {})).await;
bot.add_command("!urm".to_owned(), Box::new(UrmCommand {})).await;
bot.run().await;
println!("hello world!")
}