2016-07-07 09:25:50 +02:00
|
|
|
"""
|
|
|
|
exttargets.py - Implements extended targets like $account:xyz, $oper, etc.
|
|
|
|
"""
|
|
|
|
|
|
|
|
from pylinkirc import world
|
|
|
|
from pylinkirc.log import log
|
|
|
|
|
|
|
|
def bind(func):
|
|
|
|
"""
|
|
|
|
Binds an exttarget with the given name.
|
|
|
|
"""
|
|
|
|
world.exttarget_handlers[func.__name__] = func
|
|
|
|
return func
|
|
|
|
|
|
|
|
@bind
|
|
|
|
def account(irc, host, uid):
|
|
|
|
"""
|
|
|
|
$account exttarget handler. The following forms are supported, with groups separated by a
|
2016-07-07 20:11:13 +02:00
|
|
|
literal colon. Account matching is case insensitive, while network name matching IS case
|
|
|
|
sensitive.
|
2016-07-07 09:25:50 +02:00
|
|
|
|
|
|
|
$account -> Returns True (a match) if the target is registered.
|
|
|
|
$account:accountname -> Returns True if the target's account name matches the one given, and the
|
|
|
|
target is connected to the local network..
|
|
|
|
$account:accountname:netname -> Returns True if both the target's account name and origin
|
|
|
|
network name match the ones given.
|
|
|
|
$account:*:netname -> Matches all logged in users on the given network.
|
|
|
|
"""
|
|
|
|
userobj = irc.users[uid]
|
|
|
|
homenet = irc.name
|
|
|
|
if hasattr(userobj, 'remote'):
|
|
|
|
# User is a PyLink Relay pseudoclient. Use their real services account on their
|
|
|
|
# origin network.
|
|
|
|
homenet, realuid = userobj.remote
|
|
|
|
log.debug('(%s) exttargets.account: Changing UID of relay client %s to %s/%s', irc.name,
|
|
|
|
uid, homenet, realuid)
|
|
|
|
try:
|
|
|
|
userobj = world.networkobjects[homenet].users[realuid]
|
|
|
|
except KeyError: # User lookup failed. Bail and return False.
|
|
|
|
log.exception('(%s) exttargets.account: KeyError finding %s/%s:', irc.name,
|
|
|
|
homenet, realuid)
|
|
|
|
return False
|
|
|
|
|
2016-07-07 20:11:13 +02:00
|
|
|
slogin = irc.toLower(userobj.services_account)
|
2016-07-07 09:25:50 +02:00
|
|
|
|
|
|
|
# Split the given exttarget host into parts, so we know how many to look for.
|
2016-07-07 20:11:13 +02:00
|
|
|
groups = list(map(irc.toLower, host.split(':')))
|
2016-07-07 09:25:50 +02:00
|
|
|
log.debug('(%s) exttargets.account: groups to match: %s', irc.name, groups)
|
|
|
|
|
|
|
|
if len(groups) == 1:
|
|
|
|
# First scenario. Return True if user is logged in.
|
|
|
|
return bool(slogin)
|
|
|
|
elif len(groups) == 2:
|
|
|
|
# Second scenario. Return True if the user's account matches the one given.
|
|
|
|
return slogin == groups[1] and homenet == irc.name
|
|
|
|
else:
|
|
|
|
# Third or fourth scenario. If there are more than 3 groups, the rest are ignored.
|
2016-07-07 09:29:52 +02:00
|
|
|
# In other words: Return True if the user is logged in, the query matches either '*' or the
|
2016-07-07 09:41:31 +02:00
|
|
|
# user's login, and the user is connected on the network requested.
|
2016-07-07 09:29:52 +02:00
|
|
|
return slogin and (groups[1] in ('*', slogin)) and (homenet == groups[2])
|
2016-07-07 20:11:31 +02:00
|
|
|
|
|
|
|
@bind
|
|
|
|
def ircop(irc, host, uid):
|
|
|
|
"""
|
|
|
|
$ircop exttarget handler. The following forms are supported, with groups separated by a
|
|
|
|
literal colon. Oper types are matched case insensitively.
|
|
|
|
|
|
|
|
$ircop -> Returns True (a match) if the target is opered.
|
|
|
|
$ircop:*admin* -> Returns True if the target's is opered and their opertype matches the glob
|
|
|
|
given.
|
|
|
|
"""
|
|
|
|
groups = host.split(':')
|
|
|
|
log.debug('(%s) exttargets.ircop: groups to match: %s', irc.name, groups)
|
|
|
|
|
|
|
|
if len(groups) == 1:
|
|
|
|
# 1st scenario.
|
|
|
|
return irc.isOper(uid, allowAuthed=False)
|
|
|
|
else:
|
|
|
|
# 2nd scenario. Use matchHost (ircmatch) to match the opertype glob to the opertype.
|
|
|
|
return irc.matchHost(groups[1], irc.users[uid].opertype)
|
2016-07-07 20:26:11 +02:00
|
|
|
|
|
|
|
@bind
|
|
|
|
def server(irc, host, uid):
|
|
|
|
"""
|
|
|
|
$server exttarget handler. The following forms are supported, with groups separated by a
|
|
|
|
literal colon. Server names are matched case insensitively, but SIDs ARE case sensitive.
|
|
|
|
|
|
|
|
$server:server.name -> Returns True (a match) if the target is connected on the given server.
|
|
|
|
$server:server.glob -> Returns True (a match) if the target is connected on a server matching the glob.
|
|
|
|
$server:1XY -> Returns True if the target's is connected on the server with the given SID.
|
|
|
|
"""
|
|
|
|
groups = host.split(':')
|
|
|
|
log.debug('(%s) exttargets.server: groups to match: %s', irc.name, groups)
|
|
|
|
|
|
|
|
if len(groups) >= 2:
|
|
|
|
sid = irc.getServer(uid)
|
|
|
|
query = groups[1]
|
|
|
|
# Return True if the SID matches the query or the server's name glob matches it.
|
|
|
|
return sid == query or irc.matchHost(query, irc.getFriendlyName(sid))
|
|
|
|
# $server alone is invalid. Don't match anything.
|
|
|
|
return False
|
2016-07-07 21:00:23 +02:00
|
|
|
|
|
|
|
@bind
|
|
|
|
def channel(irc, host, uid):
|
|
|
|
"""
|
|
|
|
$channel exttarget handler. The following forms are supported, with groups separated by a
|
|
|
|
literal colon. Channel names are matched case insensitively.
|
|
|
|
|
|
|
|
$channel:#channel -> Returns True if the target is in the given channel.
|
|
|
|
$channel:#channel:op -> Returns True if the target is in the given channel, and is opped.
|
|
|
|
Any other supported prefix (owner, admin, op, halfop, voice) can be given, but only one at a
|
|
|
|
time.
|
|
|
|
"""
|
|
|
|
groups = host.split(':')
|
|
|
|
log.debug('(%s) exttargets.channel: groups to match: %s', irc.name, groups)
|
|
|
|
try:
|
|
|
|
channel = groups[1]
|
|
|
|
except IndexError: # No channel given, abort.
|
|
|
|
return False
|
|
|
|
|
|
|
|
if len(groups) == 2:
|
|
|
|
# Just #channel was given as query
|
|
|
|
return uid in irc.channels[channel].users
|
|
|
|
elif len(groups) >= 3:
|
|
|
|
# For things like #channel:op, check if the query is in the user's prefix modes.
|
|
|
|
return groups[2].lower() in irc.channels[channel].getPrefixModes(uid)
|