mirror of
https://github.com/ayywrk/fuckircd.git
synced 2024-11-21 22:46:38 +00:00
inital commit
This commit is contained in:
commit
ccec62de1c
4
.gitignore
vendored
Normal file
4
.gitignore
vendored
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
/target
|
||||||
|
Cargo.lock
|
||||||
|
config.yaml
|
||||||
|
reporting_config.yaml
|
15
Cargo.toml
Normal file
15
Cargo.toml
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
[package]
|
||||||
|
name = "fuckircd"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
ircie = { path = "../ircie" }
|
||||||
|
tokio = { version = "1.28.2", features = ["full"] }
|
||||||
|
rand = "0.8.5"
|
||||||
|
log = "0.4.19"
|
||||||
|
env_logger = "0.10.0"
|
||||||
|
serde = { version = "1.0.163", features = ["derive"] }
|
||||||
|
serde_yaml = "0.9.21"
|
8
config.example.yaml
Normal file
8
config.example.yaml
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
host: 127.0.0.1
|
||||||
|
hostname: you.super.domain.net
|
||||||
|
port: 6667
|
||||||
|
|
||||||
|
ircd_name: ircd-ratbox-3.0.10
|
||||||
|
banner_filepath: your_banner.txt
|
||||||
|
chan_name: HIE
|
||||||
|
privmsg_line: LOLOLOL
|
21
reporting_config.example.yaml
Normal file
21
reporting_config.example.yaml
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
host: irc.somewhere.org
|
||||||
|
port: 6697
|
||||||
|
ssl: true
|
||||||
|
|
||||||
|
channels:
|
||||||
|
- "#chats"
|
||||||
|
|
||||||
|
nick: ERIS
|
||||||
|
user: ERIS
|
||||||
|
real: eris.port6667.net/6667
|
||||||
|
|
||||||
|
nickserv_pass: REDACTED
|
||||||
|
nickserv_email: REDACTED
|
||||||
|
|
||||||
|
cmdkey: .
|
||||||
|
|
||||||
|
flood_interval: 1
|
||||||
|
|
||||||
|
owner: "*!*@change.me"
|
||||||
|
|
||||||
|
admins: []
|
706
src/main.rs
Normal file
706
src/main.rs
Normal file
@ -0,0 +1,706 @@
|
|||||||
|
use std::{
|
||||||
|
collections::VecDeque,
|
||||||
|
io::{ErrorKind, Result},
|
||||||
|
net::SocketAddr,
|
||||||
|
ops::{Deref, DerefMut},
|
||||||
|
sync::Arc,
|
||||||
|
time::Duration,
|
||||||
|
};
|
||||||
|
|
||||||
|
use env_logger::Env;
|
||||||
|
use ircie::{
|
||||||
|
system_params::{Context, ResMut},
|
||||||
|
Irc,
|
||||||
|
};
|
||||||
|
|
||||||
|
use log::{debug, error, info, trace};
|
||||||
|
use rand::{
|
||||||
|
distributions::{Alphanumeric, DistString},
|
||||||
|
rngs::StdRng,
|
||||||
|
Rng, SeedableRng,
|
||||||
|
};
|
||||||
|
use serde::Deserialize;
|
||||||
|
use tokio::{
|
||||||
|
fs::File,
|
||||||
|
io::{AsyncReadExt, AsyncWriteExt},
|
||||||
|
net::TcpListener,
|
||||||
|
sync::{
|
||||||
|
mpsc::{self},
|
||||||
|
RwLock,
|
||||||
|
},
|
||||||
|
task::JoinHandle,
|
||||||
|
};
|
||||||
|
|
||||||
|
const MAX_MSG_LEN: usize = 512;
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
struct Config {
|
||||||
|
host: String,
|
||||||
|
hostname: String,
|
||||||
|
port: u16,
|
||||||
|
ircd_name: String,
|
||||||
|
banner_filepath: String,
|
||||||
|
chan_name: String,
|
||||||
|
privmsg_line: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Config {
|
||||||
|
pub async fn load(path: &str) -> Result<Self> {
|
||||||
|
let mut file = File::open(path).await?;
|
||||||
|
let mut content = String::new();
|
||||||
|
file.read_to_string(&mut content).await?;
|
||||||
|
|
||||||
|
let config: Self = serde_yaml::from_str(&content).unwrap();
|
||||||
|
|
||||||
|
Ok(config)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct Ircd {
|
||||||
|
host: String,
|
||||||
|
port: u16,
|
||||||
|
hostname: String,
|
||||||
|
name: String,
|
||||||
|
motd: Vec<String>,
|
||||||
|
chan_name: String,
|
||||||
|
privmsg_line: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Nerd(mpsc::Receiver<Nick>);
|
||||||
|
|
||||||
|
struct IrcClient<T> {
|
||||||
|
addr: SocketAddr,
|
||||||
|
read_task: JoinHandle<T>,
|
||||||
|
write_task: JoinHandle<T>,
|
||||||
|
rx: mpsc::Receiver<bool>,
|
||||||
|
reporting_tx: mpsc::Sender<Nick>,
|
||||||
|
send_queue: Arc<RwLock<VecDeque<String>>>,
|
||||||
|
recv_queue: Arc<RwLock<VecDeque<String>>>,
|
||||||
|
ircd: Ircd,
|
||||||
|
server_created_on: String,
|
||||||
|
global_users: u32,
|
||||||
|
global_invisibles: u32,
|
||||||
|
global_servers: u32,
|
||||||
|
total_ircops: u32,
|
||||||
|
total_unknown_conns: u32,
|
||||||
|
total_channels: u32,
|
||||||
|
local_users: u32,
|
||||||
|
local_servers: u32,
|
||||||
|
max_local_users: u32,
|
||||||
|
max_global_users: u32,
|
||||||
|
highest_conn_count: u32,
|
||||||
|
total_conns_received: u32,
|
||||||
|
|
||||||
|
nick: Option<Nick>,
|
||||||
|
user: Option<User>,
|
||||||
|
welcomed: bool,
|
||||||
|
quit: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct Nick(pub String);
|
||||||
|
|
||||||
|
impl Deref for Nick {
|
||||||
|
type Target = String;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DerefMut for Nick {
|
||||||
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
||||||
|
&mut self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub struct User {
|
||||||
|
username: String,
|
||||||
|
hostname: String,
|
||||||
|
servername: String,
|
||||||
|
realname: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> IrcClient<T> {
|
||||||
|
pub fn new(
|
||||||
|
addr: SocketAddr,
|
||||||
|
send_queue: Arc<RwLock<VecDeque<String>>>,
|
||||||
|
recv_queue: Arc<RwLock<VecDeque<String>>>,
|
||||||
|
ircd: Ircd,
|
||||||
|
read_task: JoinHandle<T>,
|
||||||
|
write_task: JoinHandle<T>,
|
||||||
|
rx: mpsc::Receiver<bool>,
|
||||||
|
reporting_tx: mpsc::Sender<Nick>,
|
||||||
|
) -> Self {
|
||||||
|
let mut rng = rand::thread_rng();
|
||||||
|
|
||||||
|
let global_users = rng.gen_range(200..8000);
|
||||||
|
let global_invisibles = rng.gen_range(100..global_users);
|
||||||
|
let global_servers = rng.gen_range(2..20);
|
||||||
|
let total_ircops = rng.gen_range(10..(global_users / 10));
|
||||||
|
let total_unknown_conns = rng.gen_range(1..10);
|
||||||
|
let total_channels =
|
||||||
|
rng.gen_range((global_users / 30)..((global_users as f32 * 1.5) as u32));
|
||||||
|
let local_users = rng.gen_range((global_users / 20)..(global_users / 2));
|
||||||
|
let local_servers = rng.gen_range(1..global_servers);
|
||||||
|
let max_local_users = rng.gen_range(local_users..(local_users * 2));
|
||||||
|
let max_global_users = rng.gen_range(max_local_users..(max_local_users * 2));
|
||||||
|
let highest_conn_count = rng.gen_range((max_local_users - 10)..max_local_users + 10);
|
||||||
|
let total_conns_received =
|
||||||
|
rng.gen_range((highest_conn_count * 100)..(highest_conn_count * 1000));
|
||||||
|
|
||||||
|
Self {
|
||||||
|
addr,
|
||||||
|
read_task,
|
||||||
|
write_task,
|
||||||
|
rx,
|
||||||
|
reporting_tx,
|
||||||
|
send_queue: send_queue,
|
||||||
|
recv_queue: recv_queue,
|
||||||
|
ircd,
|
||||||
|
server_created_on: "Sun May 24 2020 at 03:31:19 UTC".to_owned(),
|
||||||
|
global_users,
|
||||||
|
global_invisibles,
|
||||||
|
global_servers,
|
||||||
|
total_ircops,
|
||||||
|
total_unknown_conns,
|
||||||
|
total_channels,
|
||||||
|
local_users,
|
||||||
|
local_servers,
|
||||||
|
max_local_users,
|
||||||
|
max_global_users,
|
||||||
|
highest_conn_count,
|
||||||
|
total_conns_received,
|
||||||
|
nick: None,
|
||||||
|
user: None,
|
||||||
|
welcomed: false,
|
||||||
|
quit: false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn queue(&mut self, data: impl AsRef<str>) {
|
||||||
|
// do not use this trace if you have get_fucked() executing
|
||||||
|
//trace!(">> {}", data.as_ref());
|
||||||
|
self.send_queue
|
||||||
|
.write()
|
||||||
|
.await
|
||||||
|
.push_back(format!("{}\r\n", data.as_ref()));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn process(&mut self) -> Result<()> {
|
||||||
|
self.connection().await?;
|
||||||
|
|
||||||
|
loop {
|
||||||
|
if self.quit {
|
||||||
|
self.quit();
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Ok(should_quit) = self.rx.try_recv() {
|
||||||
|
if should_quit {
|
||||||
|
self.quit();
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
self.process_line().await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn process_line(&mut self) {
|
||||||
|
let Some(line) = self.recv_queue.write().await.pop_front() else { return; };
|
||||||
|
trace!("<< {}", line);
|
||||||
|
self.handle(&line).await;
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn connection(&mut self) -> Result<()> {
|
||||||
|
self.queue("NOTICE AUTH :*** Looking up your hostname...")
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue("NOTICE AUTH :*** Checking ident").await;
|
||||||
|
self.queue("NOTICE AUTH :*** Found your hostname").await;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn handle(&mut self, line: &str) {
|
||||||
|
let args = line.split_whitespace().collect::<Vec<_>>();
|
||||||
|
|
||||||
|
match args[0].to_uppercase().as_str() {
|
||||||
|
"USER" => self.user(args[1], args[2], args[3], args[4]).await,
|
||||||
|
"NICK" => self.nick(args[1]).await,
|
||||||
|
"CAP" => self.cap_ls_302(args[1]).await,
|
||||||
|
"PING" => self.pong(args[1]).await,
|
||||||
|
"QUIT" => self.quit = true,
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn quit(&mut self) {
|
||||||
|
info!(
|
||||||
|
"{}:{} -> connection closed.",
|
||||||
|
self.addr.ip().to_string(),
|
||||||
|
self.addr.port()
|
||||||
|
);
|
||||||
|
self.read_task.abort();
|
||||||
|
self.write_task.abort();
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn pong(&mut self, val: &str) {
|
||||||
|
self.queue(format!(
|
||||||
|
":{} PONG {} :{}",
|
||||||
|
self.ircd.hostname, self.ircd.hostname, val
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn cap_ls_302(&mut self, cmd: &str) {
|
||||||
|
if cmd == "LS" {
|
||||||
|
self.queue(format!(":{} CAP * LS :multi-prefix", self.ircd.hostname))
|
||||||
|
.await;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if cmd == "REQ" {
|
||||||
|
self.queue(format!(":{} CAP * ACK :multi-prefix", self.ircd.hostname))
|
||||||
|
.await;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn user(&mut self, username: &str, hostname: &str, servername: &str, realname: &str) {
|
||||||
|
self.user = Some(User {
|
||||||
|
username: username.to_owned(),
|
||||||
|
hostname: hostname.to_owned(),
|
||||||
|
servername: servername.to_owned(),
|
||||||
|
realname: realname.to_owned(),
|
||||||
|
});
|
||||||
|
self.maybe_welcome().await;
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn nick(&mut self, nick: &str) {
|
||||||
|
self.nick = Some(Nick(nick.to_owned()));
|
||||||
|
self.maybe_welcome().await;
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn maybe_welcome(&mut self) {
|
||||||
|
if self.nick.is_none() || self.user.is_none() || self.welcomed {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.queue("NOTICE AUTH :*** No Ident response").await;
|
||||||
|
|
||||||
|
self.queue(format!(
|
||||||
|
":{} 001 {} :Welcome to the EFnet Internet Relay Chat Network bob",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(format!(
|
||||||
|
":{} 002 {} :Your host is {}[{}/{}], running version {}",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.ircd.host,
|
||||||
|
self.ircd.port,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.ircd.name
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 003 {} :This server was created {}",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.server_created_on
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 004 {} :{} {} oiwszcrkfydnxbauglZCD biklmnopstveIrS bkloveI",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.ircd.name
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(
|
||||||
|
&format!(
|
||||||
|
":{} 005 {} CHANTYPES=&# EXCEPTS INVEX CHANMODES=eIb,k,l,imnpstS CHANLIMIT=&#:75 PREFIX=(ov)@+ MAXLIST=beI:100 MODES=4 NETWORK=EFnet KNOCK STATUSMSG=@+ CALLERID=g :are supported by this server",
|
||||||
|
self.ircd.hostname, self.nick.as_ref().unwrap().0
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(
|
||||||
|
&format!(
|
||||||
|
":{} 005 {} SAFELIST ELIST=U CASEMAPPING=rfc1459 CHARSET=ascii NICKLEN=9 CHANNELLEN=50 TOPICLEN=160 ETRACE CPRIVMSG CNOTICE DEAF=D MONITOR=60 :are supported by this server",
|
||||||
|
self.ircd.hostname, self.nick.as_ref().unwrap().0
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(
|
||||||
|
&format!(
|
||||||
|
":{} 005 {} FNC ACCEPT=20 MAP TARGMAX=NAMES:1,LIST:1,KICK:1,WHOIS:1,PRIVMSG:4,NOTICE:4,ACCEPT:,MONITOR: :are supported by this server",
|
||||||
|
self.ircd.hostname, self.nick.as_ref().unwrap().0
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 251 {} :There are {} users and {} invisible on {} servers",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.global_users,
|
||||||
|
self.global_invisibles,
|
||||||
|
self.global_servers
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 252 {} {} :IRC Operators online",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.total_ircops
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 253 {} {} :Unknown connection(s)",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.total_unknown_conns
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 254 {} {} :channels formed",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.total_channels
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 255 {} :I have {} clients and {} servers",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.local_users,
|
||||||
|
self.local_servers
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 265 {} {} {} :Current local users {}, max {}",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.local_users,
|
||||||
|
self.max_local_users,
|
||||||
|
self.local_users,
|
||||||
|
self.max_local_users
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 266 {} {} {} :Current global users {}, max {}",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.global_users + self.global_invisibles,
|
||||||
|
self.max_global_users,
|
||||||
|
self.global_users + self.global_invisibles,
|
||||||
|
self.max_global_users
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 250 {} :Highest connection count: {} ({} clients) ({} connections received)",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.highest_conn_count,
|
||||||
|
self.max_local_users,
|
||||||
|
self.total_conns_received
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.welcomed = true;
|
||||||
|
self.get_fucked().await;
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn get_fucked(&mut self) {
|
||||||
|
let mut rng = StdRng::from_entropy();
|
||||||
|
|
||||||
|
info!(
|
||||||
|
"I'm fucking the shit out of {} ({}:{}) right now...",
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.addr.ip().to_string(),
|
||||||
|
self.addr.port()
|
||||||
|
);
|
||||||
|
|
||||||
|
self.reporting_tx
|
||||||
|
.send(self.nick.clone().unwrap())
|
||||||
|
.await
|
||||||
|
.ok();
|
||||||
|
|
||||||
|
loop {
|
||||||
|
if let Ok(should_quit) = self.rx.try_recv() {
|
||||||
|
if should_quit {
|
||||||
|
self.quit();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if rng.gen_bool(1. / 5.) {
|
||||||
|
self.motd().await;
|
||||||
|
}
|
||||||
|
|
||||||
|
let chan = Alphanumeric.sample_string(&mut rng, 6)
|
||||||
|
+ "-"
|
||||||
|
+ &self.ircd.chan_name
|
||||||
|
+ "-"
|
||||||
|
+ &Alphanumeric.sample_string(&mut rng, 6);
|
||||||
|
|
||||||
|
self.queue(format!(
|
||||||
|
":{}!{}@{} JOIN #{}",
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.user.as_ref().unwrap().username,
|
||||||
|
self.user.as_ref().unwrap().hostname,
|
||||||
|
chan
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
let msg = format!(
|
||||||
|
":FUCKYOU{}!B@STARD PRIVMSG #{} :\x03{} {} {}",
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
chan,
|
||||||
|
rng.gen_range(2..14),
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.ircd.privmsg_line
|
||||||
|
);
|
||||||
|
|
||||||
|
self.queue(msg).await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn motd(&mut self) {
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 375 {} :- {} Message of the Day -",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
self.ircd.hostname
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
for line in self.ircd.motd.clone() {
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 372 {} : {}",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0,
|
||||||
|
line
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.queue(&format!(
|
||||||
|
":{} 376 {} :End of /MOTD command.",
|
||||||
|
self.ircd.hostname,
|
||||||
|
self.nick.as_ref().unwrap().0
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::main]
|
||||||
|
async fn main() -> std::io::Result<()> {
|
||||||
|
env_logger::Builder::from_env(Env::default().default_filter_or("info")).init();
|
||||||
|
|
||||||
|
info!("fuckircd - made by wrk.");
|
||||||
|
|
||||||
|
let config = Config::load("config.yaml").await?;
|
||||||
|
|
||||||
|
if !std::path::Path::new(&config.banner_filepath).exists() {
|
||||||
|
error!(
|
||||||
|
"motd file '{}` doesn't exists. please make it.",
|
||||||
|
config.banner_filepath
|
||||||
|
);
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut banner_file = File::open(&config.banner_filepath).await.unwrap();
|
||||||
|
let mut content = vec![];
|
||||||
|
banner_file.read_to_end(&mut content).await.unwrap();
|
||||||
|
|
||||||
|
let motd = content
|
||||||
|
.split(|&c| c == 0x0a)
|
||||||
|
.map(|line| line.strip_suffix(b"\r").unwrap_or(line))
|
||||||
|
.map(|line| String::from_utf8_lossy(line).to_string())
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
let ircd = Ircd {
|
||||||
|
host: config.host.clone(),
|
||||||
|
port: config.port,
|
||||||
|
hostname: config.hostname.clone(),
|
||||||
|
name: config.ircd_name,
|
||||||
|
motd,
|
||||||
|
chan_name: config.chan_name.clone(),
|
||||||
|
privmsg_line: config.privmsg_line.clone(),
|
||||||
|
};
|
||||||
|
|
||||||
|
let (reporting_tx, reporting_rx) = mpsc::channel(1);
|
||||||
|
|
||||||
|
let mut reporting = Irc::from_config("reporting_config.yaml").await?;
|
||||||
|
|
||||||
|
reporting
|
||||||
|
.add_resource(Nerd(reporting_rx))
|
||||||
|
.await
|
||||||
|
.add_interval_task(Duration::from_millis(50), report_fucker)
|
||||||
|
.await;
|
||||||
|
|
||||||
|
let listener = TcpListener::bind(format!("{}:{}", ircd.host, ircd.port)).await?;
|
||||||
|
|
||||||
|
info!("Starting reporting bot.");
|
||||||
|
tokio::task::spawn(async move { reporting.run().await.unwrap() });
|
||||||
|
|
||||||
|
info!("Listening on {}:{}", ircd.host, ircd.port);
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let (socket, addr) = listener.accept().await?;
|
||||||
|
|
||||||
|
info!(
|
||||||
|
"Got connexion from {}:{}",
|
||||||
|
addr.ip().to_string(),
|
||||||
|
addr.port()
|
||||||
|
);
|
||||||
|
|
||||||
|
let (mut reader, mut writer) = tokio::io::split(socket);
|
||||||
|
let send_queue = Arc::new(RwLock::new(VecDeque::<String>::new()));
|
||||||
|
let recv_queue = Arc::new(RwLock::new(VecDeque::<String>::new()));
|
||||||
|
|
||||||
|
let cloned_send_queue = send_queue.clone();
|
||||||
|
let cloned_recv_queue = recv_queue.clone();
|
||||||
|
|
||||||
|
let (write_tx, rx) = mpsc::channel(1);
|
||||||
|
|
||||||
|
let read_tx = write_tx.clone();
|
||||||
|
|
||||||
|
// WRITE TASK
|
||||||
|
let write_task = tokio::task::spawn(async move {
|
||||||
|
loop {
|
||||||
|
let len;
|
||||||
|
{
|
||||||
|
let queue = cloned_send_queue.read().await;
|
||||||
|
len = queue.len();
|
||||||
|
}
|
||||||
|
if len == 0 {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
let mut queue = cloned_send_queue.write().await;
|
||||||
|
let msg = queue.pop_front().unwrap();
|
||||||
|
|
||||||
|
let bytes_written = match writer.write(msg.as_bytes()).await {
|
||||||
|
Ok(bytes_written) => bytes_written,
|
||||||
|
Err(err) => match err.kind() {
|
||||||
|
ErrorKind::WouldBlock => {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
write_tx.send(true).await.unwrap();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
if bytes_written == 0 {
|
||||||
|
write_tx.send(true).await.unwrap();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if bytes_written < msg.len() {
|
||||||
|
queue.push_front(msg[bytes_written..].to_owned());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// READ TASK
|
||||||
|
let read_task = tokio::task::spawn(async move {
|
||||||
|
let mut partial_line = String::new();
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let mut buf = [0; MAX_MSG_LEN];
|
||||||
|
let mut lines = vec![];
|
||||||
|
let bytes_read = match reader.read(&mut buf).await {
|
||||||
|
Ok(bytes_read) => bytes_read,
|
||||||
|
Err(err) => match err.kind() {
|
||||||
|
ErrorKind::WouldBlock => {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
read_tx.send(true).await.unwrap();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
if bytes_read == 0 {
|
||||||
|
read_tx.send(true).await.unwrap();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
let buf = &buf[..bytes_read];
|
||||||
|
|
||||||
|
let buf_str: String =
|
||||||
|
partial_line + String::from_utf8_lossy(buf).into_owned().as_str();
|
||||||
|
|
||||||
|
partial_line = String::new();
|
||||||
|
|
||||||
|
let new_lines: Vec<&str> = buf_str.split("\n").collect();
|
||||||
|
let len = new_lines.len();
|
||||||
|
|
||||||
|
for (index, line) in new_lines.into_iter().enumerate() {
|
||||||
|
if index == len - 1 && &buf[buf.len() - 1..] != b"\n" {
|
||||||
|
partial_line = line.to_owned();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if line.len() != 0 {
|
||||||
|
lines.push(line.to_owned());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let mut queue = cloned_recv_queue.write().await;
|
||||||
|
queue.append(&mut lines.into());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let cloned_ircd = ircd.clone();
|
||||||
|
let cloned_reporting_tx = reporting_tx.clone();
|
||||||
|
tokio::task::spawn(async move {
|
||||||
|
IrcClient::new(
|
||||||
|
addr,
|
||||||
|
send_queue,
|
||||||
|
recv_queue,
|
||||||
|
cloned_ircd,
|
||||||
|
read_task,
|
||||||
|
write_task,
|
||||||
|
rx,
|
||||||
|
cloned_reporting_tx,
|
||||||
|
)
|
||||||
|
.process()
|
||||||
|
.await
|
||||||
|
.ok();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn report_fucker(mut ctx: Context, mut nerd: ResMut<Nerd>) {
|
||||||
|
//TODO: Add ircie config access to systems.
|
||||||
|
|
||||||
|
if let Ok(nerd_nick) = nerd.0.try_recv() {
|
||||||
|
debug!("Sending report to #dev...");
|
||||||
|
|
||||||
|
ctx.privmsg(
|
||||||
|
"#dev",
|
||||||
|
&format!("I'm fucking with {} right now.", nerd_nick.0),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user