2011-01-21 05:18:22 -05:00
|
|
|
# Copyright (c) 2008 Duncan Fordyce
|
|
|
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
# of this software and associated documentation files (the "Software"), to deal
|
|
|
|
# in the Software without restriction, including without limitation the rights
|
|
|
|
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
# copies of the Software, and to permit persons to whom the Software is
|
|
|
|
# furnished to do so, subject to the following conditions:
|
|
|
|
# The above copyright notice and this permission notice shall be included in
|
|
|
|
# all copies or substantial portions of the Software.
|
|
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
# THE SOFTWARE.
|
|
|
|
|
2022-03-19 19:48:19 -04:00
|
|
|
import logging
|
|
|
|
import logging.config
|
2021-12-01 12:29:17 -05:00
|
|
|
import ostools
|
|
|
|
_datadir = ostools.getDataDir()
|
|
|
|
logging.config.fileConfig(_datadir + "logging.ini")
|
2021-08-24 09:49:50 -04:00
|
|
|
PchumLog = logging.getLogger('pchumLogger')
|
|
|
|
|
2011-01-21 05:18:22 -05:00
|
|
|
import logging
|
|
|
|
import socket
|
|
|
|
import time
|
|
|
|
import traceback
|
2021-04-22 11:42:24 -04:00
|
|
|
import ssl
|
2022-03-16 23:14:46 -04:00
|
|
|
import json
|
2022-03-18 05:55:01 -04:00
|
|
|
import select
|
2011-01-21 05:18:22 -05:00
|
|
|
|
2022-04-10 23:57:13 -04:00
|
|
|
from oyoyo.parse import parse_raw_irc_command
|
2011-01-21 05:18:22 -05:00
|
|
|
from oyoyo import helpers
|
|
|
|
from oyoyo.cmdhandler import CommandError
|
|
|
|
|
|
|
|
class IRCClientError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class IRCClient:
|
|
|
|
""" IRC Client class. This handles one connection to a server.
|
|
|
|
This can be used either with or without IRCApp ( see connect() docs )
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, cmd_handler, **kwargs):
|
|
|
|
""" the first argument should be an object with attributes/methods named
|
|
|
|
as the irc commands. You may subclass from one of the classes in
|
|
|
|
oyoyo.cmdhandler for convenience but it is not required. The
|
|
|
|
methods should have arguments (prefix, args). prefix is
|
|
|
|
normally the sender of the command. args is a list of arguments.
|
|
|
|
Its recommened you subclass oyoyo.cmdhandler.DefaultCommandHandler,
|
|
|
|
this class provides defaults for callbacks that are required for
|
|
|
|
normal IRC operation.
|
|
|
|
|
|
|
|
all other arguments should be keyword arguments. The most commonly
|
|
|
|
used will be nick, host and port. You can also specify an "on connect"
|
|
|
|
callback. ( check the source for others )
|
|
|
|
|
|
|
|
Warning: By default this class will not block on socket operations, this
|
|
|
|
means if you use a plain while loop your app will consume 100% cpu.
|
|
|
|
To enable blocking pass blocking=True.
|
|
|
|
|
|
|
|
>>> class My_Handler(DefaultCommandHandler):
|
|
|
|
... def privmsg(self, prefix, command, args):
|
|
|
|
... print "%s said %s" % (prefix, args[1])
|
|
|
|
...
|
|
|
|
>>> def connect_callback(c):
|
|
|
|
... helpers.join(c, '#myroom')
|
|
|
|
...
|
|
|
|
>>> cli = IRCClient(My_Handler,
|
|
|
|
... host="irc.freenode.net",
|
|
|
|
... port=6667,
|
|
|
|
... nick="myname",
|
|
|
|
... connect_cb=connect_callback)
|
|
|
|
...
|
|
|
|
>>> cli_con = cli.connect()
|
|
|
|
>>> while 1:
|
|
|
|
... cli_con.next()
|
|
|
|
...
|
|
|
|
"""
|
2021-04-06 13:06:51 -04:00
|
|
|
|
|
|
|
# This should be moved to profiles
|
|
|
|
|
2022-03-16 23:14:46 -04:00
|
|
|
with open(_datadir + "server.json", "r") as server_file:
|
|
|
|
read_file = server_file.read()
|
|
|
|
server_file.close()
|
|
|
|
server_obj = json.loads(read_file)
|
|
|
|
TLS = server_obj['TLS']
|
|
|
|
#print("TLS-status is: " + str(TLS))
|
|
|
|
if TLS == False:
|
|
|
|
#print("false")
|
|
|
|
self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
2022-03-18 05:55:01 -04:00
|
|
|
#self.socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
|
2022-03-16 23:14:46 -04:00
|
|
|
else:
|
2021-04-06 13:06:51 -04:00
|
|
|
self.context = ssl.create_default_context()
|
|
|
|
self.context.check_hostname = False
|
|
|
|
self.context.verify_mode = ssl.CERT_NONE
|
|
|
|
self.bare_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
|
|
|
self.socket = self.context.wrap_socket(self.bare_socket)
|
2022-03-18 05:55:01 -04:00
|
|
|
#self.socket.setsockopt(socket.IPPROTO_TCP, socket.SO_KEEPALIVE, 1)
|
|
|
|
self.blocking = True
|
|
|
|
self.socket.setblocking(self.blocking)
|
2021-04-06 13:06:51 -04:00
|
|
|
|
2011-01-21 05:18:22 -05:00
|
|
|
self.nick = None
|
|
|
|
self.real_name = None
|
|
|
|
self.host = None
|
|
|
|
self.port = None
|
|
|
|
self.connect_cb = None
|
2011-02-14 12:39:26 -05:00
|
|
|
self.timeout = None
|
2011-01-21 05:18:22 -05:00
|
|
|
|
|
|
|
self.__dict__.update(kwargs)
|
|
|
|
self.command_handler = cmd_handler(self)
|
|
|
|
|
2011-02-21 14:07:59 -05:00
|
|
|
self._end = False
|
2011-01-21 05:18:22 -05:00
|
|
|
|
|
|
|
def send(self, *args, **kwargs):
|
|
|
|
""" send a message to the connected server. all arguments are joined
|
|
|
|
with a space for convenience, for example the following are identical
|
|
|
|
|
|
|
|
>>> cli.send("JOIN %s" % some_room)
|
|
|
|
>>> cli.send("JOIN", some_room)
|
|
|
|
|
|
|
|
In python 2, all args must be of type str or unicode, *BUT* if they are
|
|
|
|
unicode they will be converted to str with the encoding specified by
|
|
|
|
the 'encoding' keyword argument (default 'utf8').
|
|
|
|
In python 3, all args must be of type str or bytes, *BUT* if they are
|
|
|
|
str they will be converted to bytes with the encoding specified by the
|
|
|
|
'encoding' keyword argument (default 'utf8').
|
|
|
|
"""
|
|
|
|
# Convert all args to bytes if not already
|
|
|
|
encoding = kwargs.get('encoding') or 'utf8'
|
|
|
|
bargs = []
|
|
|
|
for arg in args:
|
|
|
|
if isinstance(arg, str):
|
|
|
|
bargs.append(bytes(arg, encoding))
|
|
|
|
elif isinstance(arg, bytes):
|
|
|
|
bargs.append(arg)
|
|
|
|
elif type(arg).__name__ == 'unicode':
|
|
|
|
bargs.append(arg.encode(encoding))
|
|
|
|
else:
|
2022-03-18 05:55:01 -04:00
|
|
|
PchumLog.warning('Refusing to send one of the args from provided: %s'% repr([(type(arg), arg) for arg in args]))
|
2011-01-21 05:18:22 -05:00
|
|
|
raise IRCClientError('Refusing to send one of the args from provided: %s'
|
|
|
|
% repr([(type(arg), arg) for arg in args]))
|
|
|
|
|
2021-03-24 15:51:33 -04:00
|
|
|
msg = bytes(" ", "UTF-8").join(bargs)
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info('---> send "%s"' % msg)
|
2011-02-15 12:10:57 -05:00
|
|
|
try:
|
2022-03-18 05:55:01 -04:00
|
|
|
# Extra block to give a failed write another try, causes a reconnect otherwise
|
|
|
|
retry = 0
|
|
|
|
while retry < 5:
|
|
|
|
try:
|
|
|
|
ready_to_read, ready_to_write, in_error = select.select([], [self.socket], [])
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.debug("ready_to_write (len %s): " % str(len(ready_to_write)) + str(ready_to_write))
|
2022-03-18 05:55:01 -04:00
|
|
|
#ready_to_write[0].sendall(msg + bytes("\r\n", "UTF-8"))
|
|
|
|
for x in ready_to_write:
|
|
|
|
x.sendall(msg + bytes("\r\n", "UTF-8"))
|
|
|
|
break
|
2022-03-19 19:48:19 -04:00
|
|
|
except (socket.error, ValueError) as e:# "file descriptor cannot be a negative integer"
|
2022-03-18 05:55:01 -04:00
|
|
|
retry += 1
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.warning("socket.error (retry %s) %s" % (str(retry), str(e)))
|
2021-03-23 17:36:43 -04:00
|
|
|
except socket.error as se:
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.warning("socket.error %s" % str(se))
|
2011-02-15 12:10:57 -05:00
|
|
|
try: # a little dance of compatibility to get the errno
|
2011-02-18 03:17:13 -05:00
|
|
|
errno = se.errno
|
2011-02-15 12:10:57 -05:00
|
|
|
except AttributeError:
|
2011-02-18 03:17:13 -05:00
|
|
|
errno = se[0]
|
2011-02-15 12:10:57 -05:00
|
|
|
if not self.blocking and errno == 11:
|
|
|
|
pass
|
|
|
|
else:
|
2011-02-18 03:17:13 -05:00
|
|
|
raise se
|
2011-01-21 05:18:22 -05:00
|
|
|
|
|
|
|
def connect(self):
|
|
|
|
""" initiates the connection to the server set in self.host:self.port
|
|
|
|
"""
|
2011-02-14 12:39:26 -05:00
|
|
|
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info('connecting to %s:%s' % (self.host, self.port))
|
2011-02-21 14:07:59 -05:00
|
|
|
self.socket.connect(("%s" % self.host, self.port))
|
|
|
|
if not self.blocking:
|
|
|
|
self.socket.setblocking(0)
|
|
|
|
if self.timeout:
|
|
|
|
self.socket.settimeout(self.timeout)
|
|
|
|
helpers.nick(self, self.nick)
|
|
|
|
helpers.user(self, self.nick, self.real_name)
|
|
|
|
|
|
|
|
if self.connect_cb:
|
|
|
|
self.connect_cb(self)
|
2011-01-21 05:18:22 -05:00
|
|
|
|
2011-02-21 14:07:59 -05:00
|
|
|
def conn(self):
|
|
|
|
"""returns a generator object. """
|
|
|
|
try:
|
2011-01-21 05:18:22 -05:00
|
|
|
buffer = bytes()
|
|
|
|
while not self._end:
|
|
|
|
try:
|
2022-03-18 05:55:01 -04:00
|
|
|
ready_to_read, ready_to_write, in_error = select.select([self.socket], [], []) # Don't wanna cause an unnecessary timeout
|
|
|
|
# Though this could probably be 90
|
|
|
|
PchumLog.debug("ready_to_read (len %s): " % len(ready_to_read) + str(ready_to_read))
|
|
|
|
for x in ready_to_read:
|
|
|
|
buffer += x.recv(1024)
|
|
|
|
#print("pre-recv")
|
|
|
|
#buffer += self.socket.recv(1024)
|
|
|
|
#print("post-recv")
|
|
|
|
#print(buffer)
|
2022-03-16 23:14:46 -04:00
|
|
|
#raise socket.timeout
|
2021-03-23 17:36:43 -04:00
|
|
|
except socket.timeout as e:
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.warning("timeout in client.py, " + str(e))
|
2011-02-19 21:38:06 -05:00
|
|
|
if self._end:
|
|
|
|
break
|
2011-02-18 21:02:54 -05:00
|
|
|
raise e
|
2022-03-19 19:48:19 -04:00
|
|
|
except (socket.error, ValueError) as e:
|
2022-03-18 05:55:01 -04:00
|
|
|
PchumLog.warning("conn socket.error %s in %s" % (e, self))
|
2011-02-19 21:38:06 -05:00
|
|
|
if self._end:
|
|
|
|
break
|
2011-01-21 05:18:22 -05:00
|
|
|
try: # a little dance of compatibility to get the errno
|
|
|
|
errno = e.errno
|
|
|
|
except AttributeError:
|
|
|
|
errno = e[0]
|
|
|
|
if not self.blocking and errno == 11:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
raise e
|
|
|
|
else:
|
2011-02-19 21:38:06 -05:00
|
|
|
if self._end:
|
|
|
|
break
|
2022-03-18 05:55:01 -04:00
|
|
|
PchumLog.debug("pre buffer check, len(buffer) = " + str(len(buffer)))
|
2011-02-16 06:11:09 -05:00
|
|
|
if len(buffer) == 0 and self.blocking:
|
2022-03-18 05:55:01 -04:00
|
|
|
PchumLog.debug("len(buffer) = 0")
|
2011-02-16 06:11:09 -05:00
|
|
|
raise socket.error("Connection closed")
|
|
|
|
|
2021-03-24 15:51:33 -04:00
|
|
|
data = buffer.split(bytes("\n", "UTF-8"))
|
2011-01-21 05:18:22 -05:00
|
|
|
buffer = data.pop()
|
|
|
|
|
2021-08-24 09:49:50 -04:00
|
|
|
PchumLog.debug("data = " + str(data))
|
|
|
|
|
2011-01-21 05:18:22 -05:00
|
|
|
for el in data:
|
2022-03-18 05:55:01 -04:00
|
|
|
PchumLog.debug("el=%s, data=%s" % (el,data))
|
2011-01-21 05:18:22 -05:00
|
|
|
prefix, command, args = parse_raw_irc_command(el)
|
|
|
|
try:
|
|
|
|
self.command_handler.run(command, prefix, *args)
|
2022-03-16 23:14:46 -04:00
|
|
|
except CommandError as e:
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.debug("CommandError %s" % str(e))
|
2011-01-21 05:18:22 -05:00
|
|
|
|
|
|
|
yield True
|
2021-03-23 17:36:43 -04:00
|
|
|
except socket.timeout as se:
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.debug("passing timeout")
|
2011-02-18 21:02:54 -05:00
|
|
|
raise se
|
2021-03-23 17:36:43 -04:00
|
|
|
except socket.error as se:
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.debug("problem: %s" % (str(se)))
|
2011-02-19 21:38:06 -05:00
|
|
|
if self.socket:
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info('error: closing socket')
|
2011-02-18 03:17:13 -05:00
|
|
|
self.socket.close()
|
|
|
|
raise se
|
2016-11-29 15:20:41 -05:00
|
|
|
except Exception as e:
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.debug("other exception: %s" % str(e))
|
2011-02-21 14:07:59 -05:00
|
|
|
raise e
|
2011-02-18 03:17:13 -05:00
|
|
|
else:
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.debug("ending while, end is %s" % self._end)
|
2011-01-21 05:18:22 -05:00
|
|
|
if self.socket:
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info('finished: closing socket')
|
2011-01-21 05:18:22 -05:00
|
|
|
self.socket.close()
|
2011-02-19 21:38:06 -05:00
|
|
|
yield False
|
2011-02-14 16:15:32 -05:00
|
|
|
def close(self):
|
2011-02-19 18:06:54 -05:00
|
|
|
# with extreme prejudice
|
2011-02-18 21:02:54 -05:00
|
|
|
if self.socket:
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info('shutdown socket')
|
2021-03-25 18:08:08 -04:00
|
|
|
#print("shutdown socket")
|
2011-02-19 21:38:06 -05:00
|
|
|
self._end = True
|
2022-03-18 05:55:01 -04:00
|
|
|
try:
|
2022-03-18 09:50:35 -04:00
|
|
|
self.socket.shutdown(socket.SHUT_RDWR)
|
2022-03-18 05:55:01 -04:00
|
|
|
except OSError as e:
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.warning("Error while shutting down socket, already broken? %s" % str(e))
|
2022-03-18 05:55:01 -04:00
|
|
|
try:
|
|
|
|
self.socket.close()
|
|
|
|
except OSError as e:
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.warning("Error while closing socket, already broken? %s" % str(e))
|
2021-03-25 18:08:08 -04:00
|
|
|
|
|
|
|
def quit(self, msg):
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info("QUIT")
|
|
|
|
self.socket.sendall(bytes(msg + "\n", "UTF-8"))
|
2021-03-25 18:08:08 -04:00
|
|
|
|
2011-01-21 05:18:22 -05:00
|
|
|
class IRCApp:
|
|
|
|
""" This class manages several IRCClient instances without the use of threads.
|
|
|
|
(Non-threaded) Timer functionality is also included.
|
|
|
|
"""
|
|
|
|
|
|
|
|
class _ClientDesc:
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
self.con = None
|
|
|
|
self.autoreconnect = False
|
|
|
|
self.__dict__.update(kwargs)
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self._clients = {}
|
|
|
|
self._timers = []
|
|
|
|
self.running = False
|
|
|
|
self.sleep_time = 0.5
|
|
|
|
|
|
|
|
def addClient(self, client, autoreconnect=False):
|
|
|
|
""" add a client object to the application. setting autoreconnect
|
|
|
|
to true will mean the application will attempt to reconnect the client
|
|
|
|
after every disconnect. you can also set autoreconnect to a number
|
|
|
|
to specify how many reconnects should happen.
|
|
|
|
|
|
|
|
warning: if you add a client that has blocking set to true,
|
|
|
|
timers will no longer function properly """
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info('added client %s (ar=%s)' % (client, autoreconnect))
|
2011-01-21 05:18:22 -05:00
|
|
|
self._clients[client] = self._ClientDesc(autoreconnect=autoreconnect)
|
|
|
|
|
|
|
|
def addTimer(self, seconds, cb):
|
|
|
|
""" add a timed callback. accuracy is not specified, you can only
|
|
|
|
garuntee the callback will be called after seconds has passed.
|
|
|
|
( the only advantage to these timers is they dont use threads )
|
|
|
|
"""
|
2016-11-13 01:12:58 -05:00
|
|
|
assert callable(cb)
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info('added timer to call %s in %ss' % (cb, seconds))
|
2011-01-21 05:18:22 -05:00
|
|
|
self._timers.append((time.time() + seconds, cb))
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
""" run the application. this will block until stop() is called """
|
|
|
|
# TODO: convert this to use generators too?
|
|
|
|
self.running = True
|
|
|
|
while self.running:
|
|
|
|
found_one_alive = False
|
|
|
|
|
2021-03-23 17:36:43 -04:00
|
|
|
for client, clientdesc in self._clients.items():
|
2011-01-21 05:18:22 -05:00
|
|
|
if clientdesc.con is None:
|
|
|
|
clientdesc.con = client.connect()
|
|
|
|
|
|
|
|
try:
|
2021-03-23 17:36:43 -04:00
|
|
|
next(clientdesc.con)
|
2016-11-29 15:20:41 -05:00
|
|
|
except Exception as e:
|
2022-03-18 09:50:35 -04:00
|
|
|
PchumLog.error('client error %s' % str(e))
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.error(traceback.format_exc())
|
2011-01-21 05:18:22 -05:00
|
|
|
if clientdesc.autoreconnect:
|
|
|
|
clientdesc.con = None
|
|
|
|
if isinstance(clientdesc.autoreconnect, (int, float)):
|
|
|
|
clientdesc.autoreconnect -= 1
|
|
|
|
found_one_alive = True
|
|
|
|
else:
|
|
|
|
clientdesc.con = False
|
|
|
|
else:
|
|
|
|
found_one_alive = True
|
|
|
|
|
|
|
|
if not found_one_alive:
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info('nothing left alive... quiting')
|
2011-01-21 05:18:22 -05:00
|
|
|
self.stop()
|
|
|
|
|
|
|
|
now = time.time()
|
|
|
|
timers = self._timers[:]
|
|
|
|
self._timers = []
|
|
|
|
for target_time, cb in timers:
|
|
|
|
if now > target_time:
|
2022-03-16 23:14:46 -04:00
|
|
|
PchumLog.info('calling timer cb %s' % cb)
|
2011-01-21 05:18:22 -05:00
|
|
|
cb()
|
|
|
|
else:
|
|
|
|
self._timers.append((target_time, cb))
|
|
|
|
|
|
|
|
time.sleep(self.sleep_time)
|
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
""" stop the application """
|
|
|
|
self.running = False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|