logging + more irc things
This commit is contained in:
parent
9972c1d294
commit
6b66e862d5
@ -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"
|
@ -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)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
56
src/lib.rs
56
src/lib.rs
@ -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(),
|
||||||
|
@ -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);
|
||||||
|
Loading…
Reference in New Issue
Block a user