|
|
|
# Copyright (c) 2016-2017, Neil Booth
|
|
|
|
#
|
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# See the file "LICENCE" for information about the copyright
|
|
|
|
# and warranty status of this software.
|
|
|
|
|
|
|
|
'''IRC connectivity to discover peers.
|
|
|
|
|
|
|
|
Only calling start() requires the IRC Python module.
|
|
|
|
'''
|
|
|
|
|
|
|
|
import asyncio
|
|
|
|
import re
|
|
|
|
import socket
|
|
|
|
|
|
|
|
from collections import namedtuple
|
|
|
|
|
|
|
|
from lib.hash import double_sha256
|
|
|
|
from lib.util import LoggedClass
|
|
|
|
|
|
|
|
|
|
|
|
class IRC(LoggedClass):
|
|
|
|
|
|
|
|
Peer = namedtuple('Peer', 'ip_addr host ports')
|
|
|
|
|
|
|
|
class DisconnectedError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def __init__(self, env):
|
|
|
|
super().__init__()
|
|
|
|
self.env = env
|
|
|
|
|
|
|
|
# If this isn't something a peer or client expects
|
|
|
|
# then you won't appear in the client's network dialog box
|
|
|
|
irc_address = (env.coin.IRC_SERVER, env.coin.IRC_PORT)
|
|
|
|
self.channel = env.coin.IRC_CHANNEL
|
|
|
|
self.prefix = env.coin.IRC_PREFIX
|
|
|
|
|
|
|
|
self.clients = []
|
|
|
|
self.nick = '{}{}'.format(self.prefix,
|
|
|
|
env.irc_nick if env.irc_nick else
|
|
|
|
double_sha256(env.report_host.encode())
|
|
|
|
[:5].hex())
|
|
|
|
self.clients.append(IrcClient(irc_address, self.nick,
|
|
|
|
env.report_host,
|
|
|
|
env.report_tcp_port,
|
|
|
|
env.report_ssl_port))
|
|
|
|
if env.report_host_tor:
|
|
|
|
self.clients.append(IrcClient(irc_address, self.nick + '_tor',
|
|
|
|
env.report_host_tor,
|
|
|
|
env.report_tcp_port_tor,
|
|
|
|
env.report_ssl_port_tor))
|
|
|
|
|
|
|
|
self.peer_regexp = re.compile('({}[^!]*)!'.format(self.prefix))
|
|
|
|
self.peers = {}
|
|
|
|
|
|
|
|
async def start(self):
|
|
|
|
'''Start IRC connections if enabled in environment.'''
|
|
|
|
try:
|
|
|
|
if self.env.irc:
|
|
|
|
await self.join()
|
|
|
|
else:
|
|
|
|
self.logger.info('IRC is disabled')
|
|
|
|
except asyncio.CancelledError:
|
|
|
|
pass
|
|
|
|
except Exception as e:
|
|
|
|
self.logger.error(str(e))
|
|
|
|
|
|
|
|
async def join(self):
|
|
|
|
import irc.client as irc_client
|
|
|
|
from jaraco.stream import buffer
|
|
|
|
|
|
|
|
# see https://pypi.python.org/pypi/irc under DecodingInput
|
|
|
|
irc_client.ServerConnection.buffer_class = \
|
|
|
|
buffer.LenientDecodingLineBuffer
|
|
|
|
|
|
|
|
# Register handlers for events we're interested in
|
|
|
|
reactor = irc_client.Reactor()
|
|
|
|
for event in 'welcome join quit kick whoreply disconnect'.split():
|
|
|
|
reactor.add_global_handler(event, getattr(self, 'on_' + event))
|
|
|
|
|
|
|
|
# Note: Multiple nicks in same channel will trigger duplicate events
|
|
|
|
for client in self.clients:
|
|
|
|
client.connection = reactor.server()
|
|
|
|
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
for client in self.clients:
|
|
|
|
self.logger.info('Joining IRC in {} as "{}" with '
|
|
|
|
'real name "{}"'
|
|
|
|
.format(self.channel, client.nick,
|
|
|
|
client.realname))
|
|
|
|
client.connect()
|
|
|
|
while True:
|
|
|
|
reactor.process_once()
|
|
|
|
await asyncio.sleep(2)
|
|
|
|
except irc_client.ServerConnectionError as e:
|
|
|
|
self.logger.error('connection error: {}'.format(e))
|
|
|
|
except self.DisconnectedError:
|
|
|
|
self.logger.error('disconnected')
|
|
|
|
await asyncio.sleep(10)
|
|
|
|
|
|
|
|
def log_event(self, event):
|
|
|
|
self.logger.info('IRC event type {} source {} args {}'
|
|
|
|
.format(event.type, event.source, event.arguments))
|
|
|
|
|
|
|
|
def on_welcome(self, connection, event):
|
|
|
|
'''Called when we connect to irc server.'''
|
|
|
|
connection.join(self.channel)
|
|
|
|
|
|
|
|
def on_disconnect(self, connection, event):
|
|
|
|
'''Called if we are disconnected.'''
|
|
|
|
self.log_event(event)
|
|
|
|
raise self.DisconnectedError
|
|
|
|
|
|
|
|
def on_join(self, connection, event):
|
|
|
|
'''Called when someone new connects to our channel, including us.'''
|
|
|
|
# /who the channel when we join. We used to /who on each
|
|
|
|
# namreply event, but the IRC server would frequently kick us
|
|
|
|
# for flooding. This requests only once including the tor case.
|
|
|
|
if event.source.startswith(self.nick + '!'):
|
|
|
|
connection.who(self.channel)
|
|
|
|
else:
|
|
|
|
match = self.peer_regexp.match(event.source)
|
|
|
|
if match:
|
|
|
|
connection.who(match.group(1))
|
|
|
|
|
|
|
|
def on_quit(self, connection, event):
|
|
|
|
'''Called when someone leaves our channel.'''
|
|
|
|
match = self.peer_regexp.match(event.source)
|
|
|
|
if match:
|
|
|
|
self.peers.pop(match.group(1), None)
|
|
|
|
|
|
|
|
def on_kick(self, connection, event):
|
|
|
|
'''Called when someone is kicked from our channel.'''
|
|
|
|
self.log_event(event)
|
|
|
|
match = self.peer_regexp.match(event.arguments[0])
|
|
|
|
if match:
|
|
|
|
self.peers.pop(match.group(1), None)
|
|
|
|
|
|
|
|
def on_whoreply(self, connection, event):
|
|
|
|
'''Called when a response to our who requests arrives.
|
|
|
|
|
|
|
|
The nick is the 4th argument, and real name is in the 6th
|
|
|
|
argument preceeded by '0 ' for some reason.
|
|
|
|
'''
|
|
|
|
try:
|
|
|
|
nick = event.arguments[4]
|
|
|
|
if nick.startswith(self.prefix):
|
|
|
|
line = event.arguments[6].split()
|
|
|
|
try:
|
|
|
|
ip_addr = socket.gethostbyname(line[1])
|
|
|
|
except socket.error:
|
|
|
|
# Could be .onion or IPv6.
|
|
|
|
ip_addr = line[1]
|
|
|
|
peer = self.Peer(ip_addr, line[1], line[2:])
|
|
|
|
self.peers[nick] = peer
|
|
|
|
except (IndexError, UnicodeError):
|
|
|
|
# UnicodeError comes from invalid domains (issue #68)
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class IrcClient(LoggedClass):
|
|
|
|
|
|
|
|
VERSION = '1.0'
|
|
|
|
DEFAULT_PORTS = {'t': 50001, 's': 50002}
|
|
|
|
|
|
|
|
def __init__(self, irc_address, nick, host, tcp_port, ssl_port):
|
|
|
|
super().__init__()
|
|
|
|
self.irc_host, self.irc_port = irc_address
|
|
|
|
self.nick = nick
|
|
|
|
self.realname = self.create_realname(host, tcp_port, ssl_port)
|
|
|
|
self.connection = None
|
|
|
|
|
|
|
|
def connect(self, keepalive=60):
|
|
|
|
'''Connect this client to its IRC server'''
|
|
|
|
self.connection.connect(self.irc_host, self.irc_port, self.nick,
|
|
|
|
ircname=self.realname)
|
|
|
|
self.connection.set_keepalive(keepalive)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def create_realname(cls, host, tcp_port, ssl_port):
|
|
|
|
def port_text(letter, port):
|
|
|
|
if not port:
|
|
|
|
return ''
|
|
|
|
if port == cls.DEFAULT_PORTS.get(letter):
|
|
|
|
return ' ' + letter
|
|
|
|
else:
|
|
|
|
return ' ' + letter + str(port)
|
|
|
|
|
|
|
|
tcp = port_text('t', tcp_port)
|
|
|
|
ssl = port_text('s', ssl_port)
|
|
|
|
return '{} v{}{}{}'.format(host, cls.VERSION, tcp, ssl)
|