pesterchum/scripts/irc/outgoing.py

140 lines
4.8 KiB
Python
Raw Normal View History

2023-01-31 19:12:43 -05:00
"""Class and functions for sending outgoing IRC commands."""
import logging
log = logging.getLogger("pchumLogger")
2023-02-03 14:39:16 -05:00
class SendIRC:
2023-01-31 19:12:43 -05:00
"""Provides functions for outgoing IRC commands."""
def __init__(self):
self.socket = None # INET socket connected with server.
def send(self, *args: str, text=None):
"""Send a command to the IRC server.
Takes either a string or a list of strings.
The 'text' argument is for the final parameter, which can have spaces."""
# Return if disconnected
if not self.socket or self.socket.fileno() == -1:
2023-02-03 14:39:16 -05:00
log.error(
"Send attempted while disconnected, args: %s, text: %s.", args, text
)
2023-01-31 19:12:43 -05:00
return
command = ""
# Convert command arguments to a single string if passed.
if args:
command += " ".join(args)
# If text is passed, add ':' to imply everything after it is one parameter.
if text:
command += f" :{text}"
# Add characters for end of line in IRC.
command += "\r\n"
# UTF-8 is the prefered encoding in 2023.
outgoing_bytes = command.encode(encoding="utf-8", errors="replace")
try:
2023-02-03 14:39:16 -05:00
log.debug("Sending: %s", command)
2023-01-31 19:12:43 -05:00
self.socket.send(outgoing_bytes)
except OSError:
2023-02-03 14:39:16 -05:00
log.exception("Error while sending: '%s'", command.strip())
2023-01-31 19:12:43 -05:00
self.socket.close()
2023-02-03 14:39:16 -05:00
def ping(self, token):
2023-01-31 19:12:43 -05:00
"""Send PING command to server to check for connectivity."""
self.send("PING", text=token)
2023-02-03 14:39:16 -05:00
def pong(self, token):
2023-01-31 19:12:43 -05:00
"""Send PONG command to reply to server PING."""
self.send("PONG", token)
def nick(self, nick):
"""Send USER command to communicate nick to server."""
self.send("NICK", nick)
def user(self, username, realname):
"""Send USER command to communicate username and realname to server."""
self.send("USER", username, "0", "*", text=realname)
2023-02-03 16:46:48 -05:00
def privmsg(self, target, text):
2023-01-31 19:12:43 -05:00
"""Send PRIVMSG command to send a message."""
2023-02-03 14:39:16 -05:00
for line in text.split("\n"):
self.send("PRIVMSG", target, text=line)
2023-01-31 19:12:43 -05:00
2023-02-03 14:39:16 -05:00
def names(self, channel):
2023-01-31 19:12:43 -05:00
"""Send NAMES command to view channel members."""
self.send("NAMES", channel)
def kick(self, channel, user, reason=""):
"""Send KICK command to force user from channel."""
if reason:
self.send(f"KICK {channel} {user}", text=reason)
else:
self.send(f"KICK {channel} {user}")
2023-02-03 16:46:48 -05:00
def mode(self, target, modestring="", mode_arguments=""):
2023-01-31 19:12:43 -05:00
"""Set or remove modes from target."""
outgoing_mode = " ".join([target, modestring, mode_arguments]).strip()
self.send("MODE", outgoing_mode)
def ctcp(self, target, command, msg=""):
"""Send Client-to-Client Protocol message."""
2023-02-03 14:39:16 -05:00
outgoing_ctcp = " ".join(
[command, msg]
).strip() # Extra spaces break protocol, so strip.
2023-02-03 16:46:48 -05:00
self.privmsg(target, f"\x01{outgoing_ctcp}\x01")
2023-01-31 19:12:43 -05:00
2023-02-03 14:39:16 -05:00
def metadata(self, target, subcommand, *params):
"""Send Metadata command to get or set metadata.
See IRC metadata draft specification:
https://gist.github.com/k4bek4be/92c2937cefd49990fbebd001faf2b237
"""
2023-01-31 19:12:43 -05:00
self.send("METADATA", target, subcommand, *params)
2023-02-03 14:39:16 -05:00
def cap(self, subcommand, *params):
"""Send IRCv3 CAP command for capability negotiation.
See: https://ircv3.net/specs/extensions/capability-negotiation.html"""
2023-01-31 19:12:43 -05:00
self.send("CAP", subcommand, *params)
2023-02-03 14:39:16 -05:00
def join(self, channel, key=""):
2023-01-31 19:12:43 -05:00
"""Send JOIN command to join a channel/memo.
Keys or joining multiple channels is possible in the specification, but unused."""
channel_and_key = " ".join([channel, key]).strip()
self.send("JOIN", channel_and_key)
2023-02-03 14:39:16 -05:00
def part(self, channel):
2023-01-31 19:12:43 -05:00
"""Send PART command to leave a channel/memo.
Providing a reason or leaving multiple channels is possible in the specification."""
self.send("PART", channel)
2023-02-03 14:39:16 -05:00
def notice(self, target, text):
2023-01-31 19:12:43 -05:00
"""Send a NOTICE to a user or channel."""
self.send("NOTICE", target, text=text)
2023-02-03 14:39:16 -05:00
def invite(self, nick, channel):
2023-01-31 19:12:43 -05:00
"""Send INVITE command to invite a user to a channel."""
self.send("INVITE", nick, channel)
def away(self, text=None):
"""AWAY command to mark client as away or no longer away.
No 'text' parameter means the client is no longer away."""
if text:
self.send("AWAY", text=text)
else:
self.send("AWAY")
2023-02-03 16:46:48 -05:00
def list(self):
"""Send LIST command to get list of channels."""
self.send("LIST")
def quit(self, reason=""):
"""Send QUIT to terminate connection."""
self.send("QUIT", text=reason)