logging + more irc things

This commit is contained in:
wrk 2023-05-29 17:11:50 +02:00
parent 9972c1d294
commit 6b66e862d5
4 changed files with 114 additions and 30 deletions

View File

@ -10,3 +10,4 @@ tokio = { version = "1.28.2", features = ["full"] }
async-native-tls = { version = "0.5.0", default-features = false, features = [ "runtime-tokio" ] } async-native-tls = { version = "0.5.0", default-features = false, features = [ "runtime-tokio" ] }
serde = { version = "1.0.163", features = ["derive"] } serde = { version = "1.0.163", features = ["derive"] }
serde_yaml = "0.9.21" serde_yaml = "0.9.21"
log = "0.4.18"

View File

@ -1,29 +1,33 @@
use std::time::Duration; use std::time::Duration;
use log::{debug, info, warn};
use crate::{Irc, IrcPrefix}; use crate::{Irc, IrcPrefix};
impl Irc { impl Irc {
pub(crate) fn event_ping(&mut self, ping_token: &str) { pub(crate) fn event_ping(&mut self, ping_token: &str) {
debug!("PING {}", ping_token);
self.queue(&format!("PONG {}", ping_token)); self.queue(&format!("PONG {}", ping_token));
} }
pub(crate) fn event_welcome(&mut self) { pub(crate) fn event_welcome(&mut self, welcome_msg: &str) {
debug!("{welcome_msg}");
// self.identify(); // self.identify();
self.join_config_channels(); self.join_config_channels();
} }
pub(crate) fn event_nicknameinuse(&mut self) { pub(crate) fn event_nicknameinuse(&mut self) {
self.update_nick(&format!("{}_", &self.config.nick)) let new_nick = &format!("{}_", &self.config.nick);
warn!("Nick already in use., switching to {}", new_nick);
self.update_nick(new_nick)
} }
pub(crate) fn event_kick(&mut self, channel: &str, nick: &str, message: &str) { pub(crate) fn event_kick(&mut self, channel: &str, nick: &str, kicker: &str, reason: &str) {
if nick != &self.config.nick { if nick != &self.config.nick {
return; return;
} }
println!("we got kicked!"); warn!("We got kicked from {} by {}! ({})", channel, kicker, reason);
println!("{message}");
self.join(channel); self.join(channel);
} }
@ -32,14 +36,15 @@ impl Irc {
return; return;
} }
println!("need to reconnect."); warn!("We quit. We'll reconnect in {} seconds.", 15);
std::thread::sleep(Duration::from_secs(15)); std::thread::sleep(Duration::from_secs(15));
self.connect().await.unwrap(); self.connect().await.unwrap();
self.register(); self.register();
} }
pub(crate) fn event_invite(&mut self, prefix: &IrcPrefix, channel: &str) { pub(crate) fn event_invite(&mut self, prefix: &IrcPrefix, channel: &str) {
println!("{} invited us to {}", prefix.nick, channel); info!("{} invited us to {}", prefix.nick, channel);
self.join(channel);
} }
pub(crate) fn event_notice( pub(crate) fn event_notice(
@ -66,6 +71,15 @@ impl Irc {
if self.is_flood(channel) { if self.is_flood(channel) {
return; return;
} }
self.run_system(prefix, sys_name);
let response = self.run_system(prefix, sys_name);
if response.0.is_none() {
return;
}
for line in response.0.unwrap() {
self.privmsg(channel, &line)
}
} }
} }

View File

@ -6,7 +6,7 @@ pub mod system_params;
use std::{ use std::{
any::TypeId, any::TypeId,
collections::{HashMap, VecDeque}, collections::{HashMap, HashSet, VecDeque},
io::ErrorKind, io::ErrorKind,
net::ToSocketAddrs, net::ToSocketAddrs,
path::Path, path::Path,
@ -16,8 +16,9 @@ use std::{
use async_native_tls::TlsStream; use async_native_tls::TlsStream;
use factory::Factory; use factory::Factory;
use irc_command::IrcCommand; use irc_command::IrcCommand;
use log::{debug, info, trace, warn};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use system::{IntoSystem, StoredSystem, System}; use system::{IntoSystem, Response, StoredSystem, System};
use tokio::{ use tokio::{
fs::File, fs::File,
io::{AsyncReadExt, AsyncWriteExt}, io::{AsyncReadExt, AsyncWriteExt},
@ -141,7 +142,7 @@ pub struct IrcConfig {
host: String, host: String,
port: u16, port: u16,
ssl: bool, ssl: bool,
channels: Vec<String>, channels: HashSet<String>,
nick: String, nick: String,
user: String, user: String,
real: String, real: String,
@ -204,14 +205,16 @@ impl Irc {
self self
} }
pub fn run_system<'a>(&mut self, prefix: &'a IrcPrefix, name: &str) { pub fn run_system<'a>(&mut self, prefix: &'a IrcPrefix, name: &str) -> Response {
let system = self.systems.get_mut(name).unwrap(); let system = self.systems.get_mut(name).unwrap();
system.run(prefix, &mut self.factory); system.run(prefix, &mut self.factory)
} }
pub async fn connect(&mut self) -> std::io::Result<()> { pub async fn connect(&mut self) -> std::io::Result<()> {
let domain = format!("{}:{}", self.config.host, self.config.port); let domain = format!("{}:{}", self.config.host, self.config.port);
info!("Connecting to {}", domain);
let mut addrs = domain let mut addrs = domain
.to_socket_addrs() .to_socket_addrs()
.expect("Unable to get addrs from domain {domain}"); .expect("Unable to get addrs from domain {domain}");
@ -235,6 +238,10 @@ impl Irc {
} }
pub fn register(&mut self) { pub fn register(&mut self) {
info!(
"Registering as {}!{} ({})",
self.config.nick, self.config.user, self.config.real
);
self.queue(&format!( self.queue(&format!(
"USER {} 0 * {}", "USER {} 0 * {}",
self.config.user, self.config.real self.config.user, self.config.real
@ -279,6 +286,7 @@ impl Irc {
while self.send_queue.len() > 0 { while self.send_queue.len() > 0 {
let msg = self.send_queue.pop_front().unwrap(); let msg = self.send_queue.pop_front().unwrap();
trace!(">> {}", msg.replace("\r\n", ""));
let bytes_written = match self.stream.write(msg.as_bytes()).await { let bytes_written = match self.stream.write(msg.as_bytes()).await {
Ok(bytes_written) => bytes_written, Ok(bytes_written) => bytes_written,
Err(err) => match err.kind() { Err(err) => match err.kind() {
@ -306,13 +314,11 @@ impl Irc {
let max = (MAX_MSG_LEN - "\r\n".len()).min(msg[i..].len()); let max = (MAX_MSG_LEN - "\r\n".len()).min(msg[i..].len());
let mut m = msg[i..(i + max)].to_owned(); let mut m = msg[i..(i + max)].to_owned();
println!(">> {:?}", m);
m = m + "\r\n"; m = m + "\r\n";
self.send_queue.push_back(m); self.send_queue.push_back(m);
i += MAX_MSG_LEN - "\r\n".len() i += MAX_MSG_LEN - "\r\n".len()
} }
} else { } else {
println!(">> {:?}", msg);
msg = msg + "\r\n"; msg = msg + "\r\n";
self.send_queue.push_back(msg); self.send_queue.push_back(msg);
} }
@ -320,8 +326,8 @@ impl Irc {
pub async fn update(&mut self) -> std::io::Result<()> { pub async fn update(&mut self) -> std::io::Result<()> {
self.recv().await?; self.recv().await?;
self.send().await?;
self.handle_commands().await; self.handle_commands().await;
self.send().await?;
Ok(()) Ok(())
} }
@ -330,7 +336,7 @@ impl Irc {
let owned_line = self.recv_queue.pop_front().unwrap(); let owned_line = self.recv_queue.pop_front().unwrap();
let line = owned_line.as_str(); let line = owned_line.as_str();
println!("<< {:?}", line); trace!("<< {:?}", line);
let mut message: IrcMessage = line.into(); let mut message: IrcMessage = line.into();
@ -372,12 +378,15 @@ impl Irc {
} }
fn join(&mut self, channel: &str) { fn join(&mut self, channel: &str) {
self.queue(&format!("JOIN {}", channel)) info!("Joining {channel}");
self.queue(&format!("JOIN {}", channel));
self.config.channels.insert(channel.to_owned());
} }
fn join_config_channels(&mut self) { fn join_config_channels(&mut self) {
for i in 0..self.config.channels.len() { for i in 0..self.config.channels.len() {
let channel = &self.config.channels[i]; let channel = self.config.channels.iter().nth(i).unwrap();
info!("Joining {channel}");
self.queue(&format!("JOIN {}", channel)) self.queue(&format!("JOIN {}", channel))
} }
} }
@ -390,14 +399,18 @@ impl Irc {
fn is_flood(&mut self, channel: &str) -> bool { fn is_flood(&mut self, channel: &str) -> bool {
let mut flood_control = match self.flood_controls.entry(channel.to_owned()) { let mut flood_control = match self.flood_controls.entry(channel.to_owned()) {
std::collections::hash_map::Entry::Occupied(o) => o.into_mut(), std::collections::hash_map::Entry::Occupied(o) => o.into_mut(),
std::collections::hash_map::Entry::Vacant(v) => v.insert(FloodControl { std::collections::hash_map::Entry::Vacant(v) => {
last_cmd: SystemTime::now(), v.insert(FloodControl {
}), last_cmd: SystemTime::now(),
});
return false;
}
}; };
let elapsed = flood_control.last_cmd.elapsed().unwrap(); let elapsed = flood_control.last_cmd.elapsed().unwrap();
if elapsed.as_secs_f32() < self.config.flood_interval { if elapsed.as_secs_f32() < self.config.flood_interval {
warn!("they be floodin @ {channel}!");
return true; return true;
} }
@ -406,12 +419,14 @@ impl Irc {
} }
pub fn privmsg(&mut self, channel: &str, message: &str) { pub fn privmsg(&mut self, channel: &str, message: &str) {
debug!("sending privmsg to {} : {}", channel, message);
self.queue(&format!("PRIVMSG {} :{}", channel, message)); self.queue(&format!("PRIVMSG {} :{}", channel, message));
} }
pub fn privmsg_all(&mut self, message: &str) { pub fn privmsg_all(&mut self, message: &str) {
for i in 0..self.config.channels.len() { for i in 0..self.config.channels.len() {
let channel = &self.config.channels[i]; let channel = self.config.channels.iter().nth(i).unwrap();
debug!("sending privmsg to {} : {}", channel, message);
self.queue(&format!("PRIVMSG {} :{}", channel, message)); self.queue(&format!("PRIVMSG {} :{}", channel, message));
} }
} }
@ -419,17 +434,18 @@ impl Irc {
async fn handle_message<'a>(&mut self, message: &'a IrcMessage<'a>) { async fn handle_message<'a>(&mut self, message: &'a IrcMessage<'a>) {
match message.command { match message.command {
IrcCommand::PING => self.event_ping(&message.parameters[0]), IrcCommand::PING => self.event_ping(&message.parameters[0]),
IrcCommand::RPL_WELCOME => self.event_welcome(), IrcCommand::RPL_WELCOME => self.event_welcome(&message.parameters[1..].join(" ")),
IrcCommand::ERR_NICKNAMEINUSE => self.event_nicknameinuse(), IrcCommand::ERR_NICKNAMEINUSE => self.event_nicknameinuse(),
IrcCommand::KICK => self.event_kick( IrcCommand::KICK => self.event_kick(
message.parameters[0], &message.parameters[0],
message.parameters[1], &message.parameters[1],
&message.parameters[3..].join(" "), &message.prefix.as_ref().unwrap().nick,
&message.parameters[2..].join(" "),
), ),
IrcCommand::QUIT => self.event_quit(message.prefix.as_ref().unwrap()).await, IrcCommand::QUIT => self.event_quit(message.prefix.as_ref().unwrap()).await,
IrcCommand::INVITE => self.event_invite( IrcCommand::INVITE => self.event_invite(
message.prefix.as_ref().unwrap(), message.prefix.as_ref().unwrap(),
&message.parameters[0][1..], &message.parameters[1][1..],
), ),
IrcCommand::PRIVMSG => self.event_privmsg( IrcCommand::PRIVMSG => self.event_privmsg(
message.prefix.as_ref().unwrap(), message.prefix.as_ref().unwrap(),

View File

@ -100,3 +100,56 @@ impl IntoResponse for () {
Response(None) Response(None)
} }
} }
impl IntoResponse for String {
fn response(self) -> Response {
Response(Some(vec![self]))
}
}
impl IntoResponse for &str {
fn response(self) -> Response {
Response(Some(vec![self.to_owned()]))
}
}
impl IntoResponse for Vec<String> {
fn response(self) -> Response {
Response(Some(self))
}
}
impl IntoResponse for Vec<&str> {
fn response(self) -> Response {
Response(Some(
self.iter().map(|elem| elem.to_string()).collect::<Vec<_>>(),
))
}
}
macro_rules! impl_into_response_for_primitives {
($param:ident) => {
impl IntoResponse for $param {
fn response(self) -> Response {
Response(Some(vec![self.to_string()]))
}
}
};
}
impl_into_response_for_primitives!(u8);
impl_into_response_for_primitives!(u16);
impl_into_response_for_primitives!(u32);
impl_into_response_for_primitives!(usize);
impl_into_response_for_primitives!(u64);
impl_into_response_for_primitives!(u128);
impl_into_response_for_primitives!(i8);
impl_into_response_for_primitives!(i16);
impl_into_response_for_primitives!(i32);
impl_into_response_for_primitives!(isize);
impl_into_response_for_primitives!(i64);
impl_into_response_for_primitives!(i128);
impl_into_response_for_primitives!(f32);
impl_into_response_for_primitives!(f64);