1
mirror of git://git.acid.vegas/random.git synced 2025-01-08 06:26:39 +00:00

for fourone

This commit is contained in:
Dionysus 2020-05-21 21:01:05 -04:00
parent a00b7b4eac
commit e2fcae32d8
Signed by: acidvegas
GPG Key ID: EF4B922DB85DC9DE
8 changed files with 699 additions and 1 deletions

@ -1 +0,0 @@
Subproject commit a2d36f86df617efad0dcace56862f0a8c147d3da

42
irc/bots/ircs/README.md Normal file
View File

@ -0,0 +1,42 @@
###### Information
This project is no longer being maintained & is made available for historical purposes only.
The IRCS project is basically a stripped down version of [Anope](https://www.anope.org/)'s bots all crammed into one & was developed for usage with [UnrealIRCd](https://www.unrealircd.org/) 4.
###### Setup
You will get the lowest ping having the bot connect to localhost on the same box as the IRCd is running.
The bot *will* require network operator privledges in order to work, so make sure you add that into your IRCd configuration.
Edit [`config.py`](https://github.com/acidvegas/ircs/blob/master/ircs/core/config.py) and change the `oper_passwd` and the `admin_host` settings.
###### Commands
| Mode Command | Description | Restriction |
| --- | --- | --- |
| !mode \<chan> | Read all the auto-mode hosts for \<channel>. | *+q only* |
| !mode \<chan> \<mode> | Read all the \<mode> auto-mode hosts for \<channel>. | *+q only* |
| !mode \<chan> \<mode> +\<ident> | Automatically +\<mode> a user matching \<ident>. | *+q only* |
| !mode \<chan> \<mode> -\<ident> | Remove automatic +\<mode> from a user matching \<ident>. | *+q only* |
| !sync \<chan> | Set all the channels stored in the database for \<channel>. | *+q only* |
| Vhost Command | Description | Restriction |
| --- | --- | --- |
| !vhost add \<ident> \<vhost> | Change the host of \<ident> to \<vhost> on connect. | *admin only*|
| !vhost drop \<ident> | Delete the VHOST registered to \<ident>. | *admin only* |
| !vhost list | Return a list of all activated VHOSTs. | *admin only* |
| !vhost on | Turn on your VHOST. | *vhost users only* |
| !vhost off | Turn off your VHOST. | *vhost users only*|
| !vhost sync | Change your current hostmask to your VHOST. | *vhost users only* |
| Admin Command | Description | Restriction |
| --- | --- | --- |
| !husers | List all users connected but not joined to any channel(s). | *admin only* |
| !husers join \<channel> | Force join all hidden users into \<channe>. | *admin only* |
| !husers kill | Kill the connection of all hidden users. | *admin only* |
| !husers gline | G:Line the connection of all hidden users. | *admin only* |
| !husers gzline | GZ:Line the connection of all hidden users. | *admin only* |
###### Mirrors
- [acid.vegas](https://acid.vegas/random) *(main)*
- [GitHub](https://github.com/acidvegas/random)
- [GitLab](https://gitlab.com/acidvegas/random)

View File

@ -0,0 +1,20 @@
#!/usr/bin/env python
# IRC Services (IRCS) - Developed by acidvegas in Python (https://acid.vegas/ircs)
# config.py
# Connection
server = 'localhost'
port = 6667
use_ipv6 = False
use_ssl = False
vhost = None
password = None
# Identity
nickname = 'IRCS'
username = 'ircs'
realname = 'IRC Services Bot'
# Admin
admin_host = 'CHANGEME'
oper_passwd = 'CHANGEME'

View File

@ -0,0 +1,70 @@
#!/usr/bin/env python
# IRC Services (IRCS) - Developed by acidvegas in Python (https://acid.vegas/ircs)
# debug.py
import ctypes
import os
import sys
import time
import string
def check_data(data):
if all(c in string.printable for c in data):
return True
else:
return False
def check_privileges():
if check_windows():
if ctypes.windll.shell32.IsUserAnAdmin() != 0:
return True
else:
return False
else:
if os.getuid() == 0 or os.geteuid() == 0:
return True
else:
return False
def check_version(major):
if sys.version_info.major == major:
return True
else:
return False
def check_windows():
if os.name == 'nt':
return True
else:
return False
def clear():
if check_windows():
os.system('cls')
else:
os.system('clear')
def error(msg, reason=None):
if reason:
print('{0} | [!] - {1} ({2})'.format(get_time(), msg, str(reason)))
else:
print('{0} | [!] - {1}'.format(get_time(), msg))
def error_exit(msg):
raise SystemExit('{0} | [!] - {1}'.format(get_time(), msg))
def get_time():
return time.strftime('%I:%M:%S')
def info():
clear()
print(''.rjust(56, '#'))
print('#{0}#'.format(''.center(54)))
print('#{0}#'.format('IRC Services (IRCS)'.center(54)))
print('#{0}#'.format('Developed by acidvegas in Python'.center(54)))
print('#{0}#'.format('https://acid.vegas/ircs'.center(54)))
print('#{0}#'.format(''.center(54)))
print(''.rjust(56, '#'))
def irc(msg):
print('{0} | [~] - {1}'.format(get_time(), msg))

View File

@ -0,0 +1,113 @@
#!/usr/bin/env python
# IRC Services (IRCS) - Developed by acidvegas in Python (https://acid.vegas/ircs)
# functions.py
import inspect
import os
import sqlite3
import string
# Database
database_dir = os.path.join(os.path.dirname(os.path.realpath(inspect.stack()[-1][1])), 'data')
database_file = os.path.join(database_dir, 'ircs.db')
# Globals
db = sqlite3.connect(database_file)
sql = db.cursor()
class Database:
def check():
tables = sql.execute('SELECT name FROM sqlite_master WHERE type=\'table\'').fetchall()
if len(tables):
return True
else:
return False
def create():
sql.execute('CREATE TABLE CHANSERV (CHANNEL TEXT NOT NULL, IDENT TEXT NOT NULL, MODE TEXT NOT NULL);')
sql.execute('CREATE TABLE HOSTSERV (IDENT TEXT NOT NULL, VHOST TEXT NOT NULL, STATUS TEXT NOT NULL);')
db.commit()
class ChanServ:
def add_mode(chan, ident, mode):
sql.execute('INSERT INTO CHANSERV (CHANNEL,IDENT,MODE) VALUES (?, ?, ?)', (chan, ident, mode))
db.commit()
def channels():
return set(list(item[0] for item in sql.execute('SELECT CHANNEL FROM CHANSERV ORDER BY CHANNEL ASC').fetchall()))
def del_mode(chan, ident):
sql.execute('DELETE FROM CHANSERV WHERE CHANNEL=? AND IDENT=?', (chan, ident))
db.commit()
def drop(chan):
sql.execute('DELETE FROM CHANSERV WHERE CHANNEL=?', (chan,))
db.commit()
def get_mode(chan, ident):
data = sql.execute('SELECT MODE FROM CHANSERV WHERE CHANNEL=? AND IDENT=?', (chan, ident)).fetchone()
if data:
return data[0]
else:
return None
def hosts():
return set(list(item[0].split('@')[1] for item in sql.execute('SELECT IDENT FROM CHANSERV', (channel,)).fetchall()))
def idents(chan, mode=None):
if mode:
return list(item[0] for item in sql.execute('SELECT IDENT FROM CHANSERV WHERE CHANNEL=? AND MODE=?', (channel, mode)).fetchall())
else:
return list(item[0] for item in sql.execute('SELECT IDENT FROM CHANSERV WHERE CHANNEL=?', (channel,)).fetchall())
def read(channel, mode=None):
if mode:
return sql.execute('SELECT IDENT FROM CHANSERV WHERE CHANNEL=? AND MODE=? ORDER BY CHANNEL ASC, MODE ASC, IDENT ASC', (channel, mode)).fetchall()
else:
return sql.execute('SELECT IDENT,MODE FROM CHANSERV WHERE CHANNEL=? ORDER BY CHANNEL ASC, MODE ASC, IDENT ASC', (channel,)).fetchall()
class HostServ:
def add(ident, vhost):
sql.execute('INSERT INTO HOSTSERV (IDENT,VHOST,STATUS) VALUES (?, ?, \'pending\')', (ident, vhost))
db.commit()
def delete(ident):
sql.execute('DELETE FROM HOSTSERV WHERE IDENT=?', (ident,))
db.commit()
def get_vhost(ident, active=False):
data = sql.execute('SELECT VHOST FROM HOSTSERV WHERE IDENT=? AND STATUS=\'on\'', (ident,)).fetchone()
if data:
return data[0]
else:
return None
def get_status(ident):
data = sql.execute('SELECT STATUS FROM HOSTSERV WHERE IDENT=?', (ident,)).fetchone()
if data:
return data[0]
else:
return None
def hosts():
return set(list(item[0].split('@')[1] for item in sql.execute('SELECT IDENT FROM CHANSERV', (channel,)).fetchall()))
def idents():
return list(item[0] for item in sql.execute('SELECT IDENT FROM HOSTSERV').fetchall())
def pending():
return sql.execute('SELECT IDENT,VHOST FROM HOSTSERV WHERE STATUS=\'pending\' ORDER BY IDENT ASC').fetchall()
def read():
return sql.execute('SELECT IDENT,VHOST FROM HOSTSERV ORDER BY IDENT ASC').fetchall()
def set_status(ident, status):
sql.execute('UPDATE HOSTSERV SET STATUS=? WHERE IDENT=?', (status, ident))
db.commit()
def vhosts():
return list(item[0] for item in sql.execute('SELECT VHOST FROM HOSTSERV').fetchall())

View File

@ -0,0 +1,433 @@
#!/usr/bin/env python
# IRC Services (IRCS) - Developed by acidvegas in Python (https://acid.vegas/ircs)
# irc.py
import socket
import ssl
import time
import config
import debug
from functions import Database, ChanServ, HostServ
# Formatting Control Characters / Color Codes
bold = '\x02'
italic = '\x1D'
underline = '\x1F'
reverse = '\x16'
reset = '\x0f'
white = '00'
black = '01'
blue = '02'
green = '03'
red = '04'
brown = '05'
purple = '06'
orange = '07'
yellow = '08'
light_green = '09'
cyan = '10'
light_cyan = '11'
light_blue = '12'
pink = '13'
grey = '14'
light_grey = '15'
class IRC(object):
server = config.server
port = config.port
use_ipv6 = config.use_ipv6
use_ssl = config.use_ssl
vhost = config.vhost
password = config.password
nickname = config.nickname
username = config.username
realname = config.realname
oper_passwd = config.oper_passwd
admin_host = config.admin_host
def __init__(self):
self.husers = list()
self.last = dict()
self.sock = None
def action(self, chan, msg):
self.sendmsg(chan, '\x01ACTION {0}\x01'.format(msg))
def chghost(self, nick, host):
self.raw('CHGHOST {0} {1}'.format(nick, host))
def color(self, msg, foreground, background=None):
if background:
return '\x03{0},{1}{2}{3}'.format(foreground, background, msg, reset)
else:
return '\x03{0}{1}{2}'.format(foreground, msg, reset)
def connect(self):
try:
self.create_socket()
self.sock.connect((self.server, self.port))
if self.password:
self.raw('PASS ' + self.password)
self.raw('USER {0} 0 * :{1}'.format(self.username, self.realname))
self.raw('NICK ' + self.nickname)
except socket.error as ex:
debug.error('Failed to connect to IRC server.', ex)
self.event_disconnect()
else:
self.listen()
def create_socket(self):
if self.use_ipv6:
self.sock = socket.socket(socket.AF_INET6, socket.SOCK_STREAM)
else:
self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
if self.vhost:
self.sock.bind((self.vhost, 0))
if self.use_ssl:
self.sock = ssl.wrap_socket(self.sock)
def error(self, target, msg, reason=None):
if reason:
self.sendmsg(target, '[{0}] {1} {2}'.format(self.color('ERROR', red), msg, self.color('({0})'.format(str(reason)), grey)))
else:
self.sendmsg(target, '[{0}] {1}'.format(self.color('ERROR', red), msg))
def event_connect(self):
self.mode(self.nickname, '+Bd')
self.oper(self.username, self.oper_passwd)
if Database.check():
for channel in ChanServ.channels():
self.join(channel)
else:
Database.create()
def event_connection(self, nick, ident):
vhost = HostServ.get_vhost(ident, True)
if vhost:
self.chghost(nick, vhost)
def event_disconnect(self):
self.sock.close()
time.sleep(10)
self.connect()
def event_end_of_who(self):
if self.last['cmd'] == 'husers':
if self.husers:
self.sendmsg(self.last['nick'], '{0} {1}'.format(self.color('Total:', light_blue), self.color(len(self.husers), grey)))
else:
self.error(self.last['nick'], 'No hidden users found.')
def event_join(self, nick, ident, chan):
mode = ChanServ.get_mode(chan, ident)
if mode:
self.mode(chan, '+{0} {1}'.format(mode, nick))
def event_kick(self, chan, kicked):
if kicked == self.nickname:
if chan in Database.channels():
self.join(chan)
def event_nick_in_use(self):
debug.error_exit('IRCS is already running.')
def event_notice(self, nick, data):
if '.' in nick or nick == self.server:
args = data.split()
if 'Client connecting' in data:
nick = args[6]
ident = args[7][1:][:-1]
self.event_connection(nick, ident)
def event_private(self, nick, ident, msg):
try:
args = msg.split()
cmd = args[0][1:]
host = ident.split('@')[1]
if cmd == 'husers' and host == self.admin_host:
if len(args) == 1:
self.husers = list()
self.last = {'nick':nick,'cmd':'husers'}
self.who('I', '*')
elif len(args) == 2:
if args[1] == 'kill':
if self.husers:
self.action(nick, 'Killing all hidden users...')
for item in self.husers:
self.kill(item['nick'], 'Killed by IRCS anti-bot protection.')
else:
self.error(nick, 'Hidden users list is empty.', 'Make sure you run !husers first')
elif args[1] == 'gzline':
if self.husers:
self.action(nick, 'Z:Lining all hidden users...')
for item in self.husers:
self.gzline(item['host'], '1d', 'Banned by IRCS anti-bot protection.')
else:
self.error(nick, 'Hidden users list is empty.', 'Make sure you run !husers first')
elif len(args) == 3:
if args [1] == 'join':
channel = args[2]
if channel.startswith('#') and len(channel) <= 20:
if self.husers:
self.action(nick, 'Joining all hidden users to {0}...'.format(channel))
for item in self.husers:
self.sajoin(item['nick'], channel)
else:
self.error(nick, 'Hidden users list is empty.', 'Make sure you run !husers first')
else:
self.error(nick, 'Invalid arguments.')
else:
self.error(nick, 'Invalid arguments.')
else:
self.error(nick, 'Invalid arguments.')
elif cmd == 'mode':
if len(args) > 1:
channel = args[1]
if channel[:1] == '#' and len(channel) <= 20 and debug.check_data(channel):
if ChanServ.get_mode(channel, ident) == 'q' or host == self.admin_host:
if len(args) == 2:
if channel in ChanServ.channels():
data = ChanServ.read(channel)
self.sendmsg(nick, '[{0}]'.format(self.color(channel, purple)))
for row in data:
self.sendmsg(nick, '{0} | {1}'.format(self.color('+' + row[1], grey), self.color(row[0], yellow)))
self.sendmsg(nick, '{0} {1}'.format(self.color('Total:', light_blue), self.color(len(data), grey)))
else:
self.error(nick, self.color(channel, purple) + ' does not exist.')
elif len(args) == 3:
if args[2] in ('a','h','o','v','q'):
if channel in ChanServ.channels():
mode = args[2]
data = ChanServ.read(channel, mode)
if data:
self.sendmsg(nick, '[{0}] {1}'.format(self.color(channel, purple) , self.color('(+{0})'.format(mode), grey)))
for row in data:
self.sendmsg(nick, self.color(row[0], yellow))
self.sendmsg(nick, '{0} {1}'.format(self.color('Total:', light_blue), self.color(len(data), grey)))
else:
self.error(nick, self.color('+{0}'.format(mode), grey) + ' is empty.')
else:
self.error(nick, self.color(channel, purple) + ' does not exist.')
else:
self.error(nick, 'Invalid arguments.')
elif len(args) == 4:
if args[2] in ('a','h','o','v','q') and args[3][:1] in '+-' and len(args[3]) <= 63 and debug.check_data(args[3]):
mode = args[2]
if mode == 'q' and host != self.admin_host:
self.error(nick, 'You do not have permission to change this mode.')
else:
action = args[3][:1]
ident = args[3][1:]
if action == '+':
if not ChanServ.get_mode(channel, ident):
ChanServ.add_mode(channel, ident, mode)
self.sendmsg(nick, '{0} {1} has been {2} to the {3} database.'.format(self.color(ident, light_blue), self.color('(+{0})'.format(mode), grey), self.color('added', green), self.color(channel, purple)))
else:
self.error(nick, '{0} already exists in the {1} database.'.format(self.color(ident, light_blue), self.color(channel, purple)))
elif action == '-':
if ChanServ.get_mode(channel, ident):
ChanServ.del_mode(channel, ident)
self.sendmsg(nick, '{0} {1} has been {2} from the {3} database.'.format(self.color(ident, light_blue), self.color('(+{0})'.format(mode), grey), self.color('removed', red), self.color(channel, purple)))
else:
self.error(nick, '{0} does not exist in the {1} database.'.format(self.color(ident, light_blue), self.color(channel, purple)))
else:
self.error(nick, 'Invalid arguments.')
else:
self.error(nick, 'Invalid arguments.')
else:
self.error(nick, 'You do not have permission to use this command.')
else:
self.error(nick, 'Invalid arguments.')
else:
self.error(nick, 'Invalid arguments.')
elif cmd == 'sync':
if len(args) == 2:
channel = args[1]
if channel[:1] == '#' and len(channel) <= 20 and debug.check_data(channel):
if channel in ChanServ.channels():
if ChanServ.get_mode(channel, ident) == 'q' or host == self.admin_host:
self.action(nick, 'Syncing all modes in {0}...'.format(color(channel, purple)))
self.last['cmd'] = 'sync ' + channel
self.who('h', '*')
else:
self.error(nick, 'You do not have permission to use this command.')
else:
self.error(nick, '{0} does not exist.'.format(color(channel, purple)))
else:
self.error(nick, 'Invalid arguments.')
else:
self.error(nick, 'Invalid arguments.')
elif cmd == 'vhost':
if len(args) == 2:
if args[1] == 'list':
if host == self.admin_host:
vhosts = HostServ.read()
if vhosts:
self.sendmsg(nick, '[{0}]'.format(self.color('Registered Vhosts', purple)))
for vhost in vhosts:
self.sendmsg(nick, '{0} {1}'.format(self.color(vhost[0], yellow), self.color('({0})'.format(vhost[1]), grey)))
self.sendmsg(nick, '{0} {1}'.format(self.color('Total:', light_blue), self.color(len(vhosts), grey)))
else:
self.error(nick, 'Vhost list is empty.')
else:
self.error(nick, 'You do not have permission to use this command.')
elif args[1] == 'off':
status = HostServ.get_status(ident)
if status == 'off':
self.error(nick, 'VHOST is already turned off.')
elif status == 'on':
HostServ.set_status(ident, 'off')
self.sendmsg(nick, 'VHOST has been turned ' + color('off', red))
else:
self.error(nick, 'You do not have a registered VHOST.')
elif args[1] == 'on':
status = HostServ.get_status(ident)
if status == 'off':
HostServ.set_status(ident, 'on')
self.sendmsg(nick, 'VHOST has been turned ' + color('on', green))
elif status == 'on':
self.error(nick, 'Your VHOST is already turned on.')
else:
self.error(nick, 'You do not have a registered VHOST.')
elif args[1] == 'sync':
vhost = HostServ.get_vhost(ident)
if host == vhost:
self.error(nick, 'Your VHOST is already synced and working.')
elif vhost:
self.action(nick, 'Syncing VHOST...')
self.chghost(nick, vhost)
else:
self.error(nick, 'You do not have a registered VHOST.')
elif len(args) == 3:
if args[1] == 'drop':
if host == self.admin_host:
ident = args[2]
if ident in HostServ.idents():
HostServ.delete(ident)
self.sendmsg(nick, '{0} has been {1} from the vhost database.'.format(self.color(ident, light_blue), self.color('removed', red)))
else:
self.error(nick, '{0} does not have a vhost.'.format(self.color(ident, light_blue)))
else:
self.error(nick, 'You do not have permission to use this command.')
elif len(args) == 4:
if args[1] == 'add':
if host == self.admin_host:
ident = args[2]
vhost = args[3]
if ident not in HostServ.idents():
HostServ.add(ident, vhost)
self.sendmsg(nick, '{0} has been {1} from the database.'.format(self.color(ident, light_blue), self.color('added', green)))
else:
self.error(nick, '{0} is already registered.'.format(color(ident, light_blue)))
else:
self.error(nick, 'You do not have permission to use this command.')
else:
self.error(nick, 'Invalid arguments.')
else:
self.error(nick, 'Invalid arguments.')
except Exception as ex:
self.error(nick, 'Unexpected error has occured.', ex)
def event_who(self, chan, user, host, nick):
if self.last:
if self.last['cmd'] == 'husers':
if chan == '*':
self.husers.append({'user':user,'host':host,'nick':nick})
self.sendmsg(self.last['nick'], '{0} {1}'.format(self.color(nick, yellow), self.color('({0}@{1})'.format(user, host), grey)))
elif self.last['cmd'].startswith('sync'):
channel = self.last['cmd'].split()[1]
if chan == channel:
mode = ChanServ.mode(chan, '{0}@{1]'.format(user, host))
if mode:
self.mode(chan, '+{0} {1}'.format(mode, nick))
def gzline(self, host, duration, msg):
self.raw('gzline *@{1} {2} {3}'.format(user, host, duration, msg))
def handle_events(self, data):
args = data.split()
if args[0] == 'PING':
self.raw('PONG ' + args[1][1:])
elif args[1] == '001':
self.event_connect()
elif args[1] == '315':
self.event_end_of_who()
elif args[1] == '352':
chan = args[3]
user = args[4]
host = args[5]
nick = args[7]
self.event_who(chan, user, host, nick)
elif args[1] == '433':
self.event_nick_in_use()
elif args[1] == 'NOTICE':
nick = args[0][1:]
self.event_notice(nick, data)
elif args[1] in ('JOIN','KICK','PRIVMSG'):
nick = args[0].split('!')[0][1:]
if nick != self.nickname:
chan = args[2]
if args[1] == 'JOIN':
host = args[0].split('!')[1]
self.event_join(nick, host, chan[1:])
elif args[1] == 'KICK':
kicked = args[3]
self.event_kick(chan, kicked)
elif args[1] == 'PRIVMSG':
ident = args[0].split('!')[1]
msg = data.split('{0} PRIVMSG {1} :'.format(args[0], chan))[1]
if msg.startswith('!'):
if chan == self.nickname:
self.event_private(nick, ident, msg)
def join(self, chan):
self.raw('JOIN ' + chan)
self.mode(chan, '+q ' + self.nickname)
def kill(self, nick, reason):
self.raw('KILL {0} {1}'.format(nick, reason))
def listen(self):
while True:
try:
data = self.sock.recv(1024).decode('utf-8')
if data:
for line in (line for line in data.split('\r\n') if line):
debug.irc(line)
if line.startswith('ERROR :Closing Link:'):
raise Exception('Connection has closed.')
elif len(line.split()) >= 2:
self.handle_events(line)
else:
debug.error('No data recieved from server.')
break
except (UnicodeDecodeError,UnicodeEncodeError):
debug.error('Unicode error has occured.')
except Exception as ex:
debug.error('Unexpected error occured.', ex)
break
self.event_disconnect()
def mode(self, target, mode):
self.raw('MODE {0} {1}'.format(target, mode))
def oper(self, nick, password):
self.raw('OPER {0} {1}'.format(nick, password))
def part(self, chan, msg):
self.raw('PART {0} {1}'.format(chan, msg))
def raw(self, msg):
self.sock.send(bytes(msg + '\r\n', 'utf-8'))
def sajoin(self, nick, chan):
self.raw('SAJOIN {0} {1}'.format(nick, chan))
def sendmsg(self, target, msg):
self.raw('PRIVMSG {0} :{1}'.format(target, msg))
def who(self, flag, args):
self.raw('who +{0} {1}'.format(flag, args))
IRCS = IRC()

1
irc/bots/ircs/ircs/data/.gitignore vendored Normal file
View File

@ -0,0 +1 @@
!.gitignore

View File

@ -0,0 +1,20 @@
#!/usr/bin/env python
# IRC Services (IRCS) - Developed by acidvegas in Python (https://acid.vegas/ircs)
# ircs.py
import os
import sys
sys.dont_write_bytecode = True
os.chdir(sys.path[0] or '.')
sys.path += ('core',)
import debug
import irc
debug.info()
if not debug.check_version(3):
debug.error_exit('IRCS requires Python 3!')
if debug.check_privileges():
debug.error_exit('Do not run IRCS as admin/root!')
irc.IRCS.connect()