Hand over regex-based-command responsibilities to commands
module
This commit is contained in:
parent
d64d12f8fd
commit
4e0e63e7f8
5 changed files with 182 additions and 191 deletions
|
@ -90,10 +90,9 @@ class Module(ModuleManager.BaseModule):
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def message(self, event, command, args_index=1):
|
def _find_command_hook(self, server, command, is_channel):
|
||||||
args_split = event["message_split"][args_index:]
|
|
||||||
if not self.has_command(command):
|
if not self.has_command(command):
|
||||||
aliases = self._get_aliases(event["server"])
|
aliases = self._get_aliases(server)
|
||||||
if command.lower() in aliases:
|
if command.lower() in aliases:
|
||||||
command, _, new_args = aliases[command.lower()].partition(" ")
|
command, _, new_args = aliases[command.lower()].partition(" ")
|
||||||
|
|
||||||
|
@ -102,12 +101,8 @@ class Module(ModuleManager.BaseModule):
|
||||||
except IndexError:
|
except IndexError:
|
||||||
return
|
return
|
||||||
|
|
||||||
if self.has_command(command):
|
|
||||||
if self._is_ignored(event["server"], event["user"], command):
|
|
||||||
return
|
|
||||||
|
|
||||||
hook = None
|
hook = None
|
||||||
target = None
|
if self.has_command(command):
|
||||||
for potential_hook in self.get_hooks(command):
|
for potential_hook in self.get_hooks(command):
|
||||||
alias_of = self._get_alias_of(potential_hook)
|
alias_of = self._get_alias_of(potential_hook)
|
||||||
if alias_of:
|
if alias_of:
|
||||||
|
@ -118,40 +113,40 @@ class Module(ModuleManager.BaseModule):
|
||||||
"'%s' is an alias of unknown command '%s'"
|
"'%s' is an alias of unknown command '%s'"
|
||||||
% (command.lower(), alias_of.lower()))
|
% (command.lower(), alias_of.lower()))
|
||||||
|
|
||||||
is_channel = "channel" in event
|
|
||||||
if not is_channel and potential_hook.kwargs.get("channel_only"):
|
if not is_channel and potential_hook.kwargs.get("channel_only"):
|
||||||
continue
|
continue
|
||||||
if is_channel and potential_hook.kwargs.get("private_only"):
|
if is_channel and potential_hook.kwargs.get("private_only"):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
hook = potential_hook
|
hook = potential_hook
|
||||||
target = event["user"] if not is_channel else event["channel"]
|
|
||||||
break
|
break
|
||||||
|
|
||||||
if not hook:
|
return hook
|
||||||
return
|
|
||||||
|
def command(self, server, target, is_channel, user, command, args_split,
|
||||||
|
tags, statusmsg, hook, **kwargs):
|
||||||
|
if self._is_ignored(server, user, command):
|
||||||
|
return False
|
||||||
|
|
||||||
module_name = self._get_prefix(hook) or ""
|
module_name = self._get_prefix(hook) or ""
|
||||||
if not module_name and hasattr(hook.function, "__self__"):
|
if not module_name and hasattr(hook.function, "__self__"):
|
||||||
module_name = hook.function.__self__._name
|
module_name = hook.function.__self__._name
|
||||||
|
|
||||||
msgid = MSGID_TAG.get_value(event["tags"])
|
msgid = MSGID_TAG.get_value(tags)
|
||||||
statusmsg = "".join(event.get("statusmsg", []))
|
stdout = outs.StdOut(server, module_name, target, msgid, statusmsg)
|
||||||
stdout = outs.StdOut(event["server"], module_name, target, msgid,
|
stderr = outs.StdErr(server, module_name, target, msgid, statusmsg)
|
||||||
statusmsg)
|
command_method = self._command_method(target, server)
|
||||||
stderr = outs.StdErr(event["server"], module_name, target, msgid,
|
|
||||||
statusmsg)
|
|
||||||
command_method = self._command_method(target, event["server"])
|
|
||||||
|
|
||||||
if hook.kwargs.get("remove_empty", True):
|
if hook.kwargs.get("remove_empty", True):
|
||||||
args_split = list(filter(None, args_split))
|
args_split = list(filter(None, args_split))
|
||||||
|
|
||||||
|
target.buffer.skip_next()
|
||||||
|
|
||||||
min_args = hook.kwargs.get("min_args")
|
min_args = hook.kwargs.get("min_args")
|
||||||
if min_args and len(args_split) < min_args:
|
if min_args and len(args_split) < min_args:
|
||||||
command_prefix = ""
|
command_prefix = ""
|
||||||
if is_channel:
|
if is_channel:
|
||||||
command_prefix = self._command_prefix(event["server"],
|
command_prefix = self._command_prefix(server, target)
|
||||||
target)
|
|
||||||
usage = self._get_usage(hook, command, command_prefix)
|
usage = self._get_usage(hook, command, command_prefix)
|
||||||
if usage:
|
if usage:
|
||||||
stderr.write("Not enough arguments, usage: %s" %
|
stderr.write("Not enough arguments, usage: %s" %
|
||||||
|
@ -161,9 +156,9 @@ class Module(ModuleManager.BaseModule):
|
||||||
min_args).send(command_method)
|
min_args).send(command_method)
|
||||||
else:
|
else:
|
||||||
returns = self.events.on("preprocess.command").call_unsafe(
|
returns = self.events.on("preprocess.command").call_unsafe(
|
||||||
hook=hook, user=event["user"], server=event["server"],
|
hook=hook, user=user, server=server, target=target,
|
||||||
target=target, is_channel=is_channel, tags=event["tags"],
|
is_channel=is_channel, tags=tags, args_split=args_split,
|
||||||
args_split=args_split)
|
command=command)
|
||||||
|
|
||||||
hard_fail = False
|
hard_fail = False
|
||||||
force_success = False
|
force_success = False
|
||||||
|
@ -181,17 +176,14 @@ class Module(ModuleManager.BaseModule):
|
||||||
if error:
|
if error:
|
||||||
stderr.write(error).send(command_method)
|
stderr.write(error).send(command_method)
|
||||||
target.buffer.skip_next()
|
target.buffer.skip_next()
|
||||||
return
|
return True
|
||||||
|
|
||||||
args = " ".join(args_split)
|
args = " ".join(args_split)
|
||||||
server = event["server"]
|
|
||||||
user = event["user"]
|
|
||||||
|
|
||||||
new_event = self.events.on("received.command").on(command
|
new_event = self.events.on(hook.event_name).make_event(user=user,
|
||||||
).make_event(user=user, server=server, target=target,
|
server=server, target=target, args=args, tags=tags,
|
||||||
args=args, tags=event["tags"], args_split=args_split,
|
args_split=args_split, stdout=stdout, stderr=stderr,
|
||||||
stdout=stdout, stderr=stderr, command=command.lower(),
|
is_channel=is_channel, command=command, **kwargs)
|
||||||
is_channel=is_channel)
|
|
||||||
|
|
||||||
self.log.trace("calling command '%s': %s",
|
self.log.trace("calling command '%s': %s",
|
||||||
[command, new_event.kwargs])
|
[command, new_event.kwargs])
|
||||||
|
@ -201,14 +193,12 @@ class Module(ModuleManager.BaseModule):
|
||||||
stderr.write(str(e))
|
stderr.write(str(e))
|
||||||
|
|
||||||
if not hook.kwargs.get("skip_out", False):
|
if not hook.kwargs.get("skip_out", False):
|
||||||
command_method = self._command_method(
|
command_method = self._command_method(target, server)
|
||||||
target, event["server"])
|
|
||||||
stdout.send(command_method)
|
stdout.send(command_method)
|
||||||
stderr.send(command_method)
|
stderr.send(command_method)
|
||||||
target.last_stdout = stdout
|
target.last_stdout = stdout
|
||||||
target.last_stderr = stderr
|
target.last_stderr = stderr
|
||||||
target.buffer.skip_next()
|
return new_event.eaten
|
||||||
event.eat()
|
|
||||||
|
|
||||||
def _command_prefix(self, server, channel):
|
def _command_prefix(self, server, channel):
|
||||||
return channel.get_setting("command-prefix",
|
return channel.get_setting("command-prefix",
|
||||||
|
@ -225,22 +215,53 @@ class Module(ModuleManager.BaseModule):
|
||||||
prefixed_commands = event["channel"].get_setting("prefixed-commands", True)
|
prefixed_commands = event["channel"].get_setting("prefixed-commands", True)
|
||||||
|
|
||||||
command_prefix = self._command_prefix(event["server"], event["channel"])
|
command_prefix = self._command_prefix(event["server"], event["channel"])
|
||||||
|
command = None
|
||||||
|
args_split = None
|
||||||
if event["message_split"][0].startswith(command_prefix):
|
if event["message_split"][0].startswith(command_prefix):
|
||||||
if not prefixed_commands:
|
if not prefixed_commands:
|
||||||
return
|
return
|
||||||
command = event["message_split"][0].replace(
|
command = event["message_split"][0].replace(
|
||||||
command_prefix, "", 1).lower()
|
command_prefix, "", 1).lower()
|
||||||
self.message(event, command)
|
args_split = event["message_split"][1:]
|
||||||
elif len(event["message_split"]) > 1 and self.is_highlight(
|
elif len(event["message_split"]) > 1 and self.is_highlight(
|
||||||
event["server"], event["message_split"][0]):
|
event["server"], event["message_split"][0]):
|
||||||
command = event["message_split"][1].lower()
|
command = event["message_split"][1].lower()
|
||||||
self.message(event, command, 2)
|
args_split = event["message_split"][2:]
|
||||||
|
|
||||||
|
if command:
|
||||||
|
hook = self._find_command_hook(event["server"], command, True)
|
||||||
|
if hook:
|
||||||
|
self.command(event["server"], event["channel"], True,
|
||||||
|
event["user"], command, args_split, event["tags"],
|
||||||
|
"".join(event["statusmsg"]), hook)
|
||||||
|
else:
|
||||||
|
regex_hook = self.events.on("command.regex").get_hooks()
|
||||||
|
for hook in regex_hook:
|
||||||
|
pattern = hook.get_kwarg("pattern", None)
|
||||||
|
if not pattern and hook.get_kwarg("pattern-url", None) == "1":
|
||||||
|
pattern = utils.http.REGEX_URL
|
||||||
|
|
||||||
|
if pattern:
|
||||||
|
match = re.match(pattern, event["message"])
|
||||||
|
if match:
|
||||||
|
command = hook.get_kwarg("command", "")
|
||||||
|
res = self.command(event["server"], event["channel"],
|
||||||
|
True, event["user"], command, "", event["tags"],
|
||||||
|
"".join(event["statusmsg"]), hook, match=match,
|
||||||
|
message=event["message"])
|
||||||
|
|
||||||
|
if res:
|
||||||
|
break
|
||||||
|
|
||||||
@utils.hook("received.message.private", priority=EventManager.PRIORITY_LOW)
|
@utils.hook("received.message.private", priority=EventManager.PRIORITY_LOW)
|
||||||
def private_message(self, event):
|
def private_message(self, event):
|
||||||
if event["message_split"] and not event["action"]:
|
if event["message_split"] and not event["action"]:
|
||||||
command = event["message_split"][0].lower()
|
command = event["message_split"][0].lower()
|
||||||
self.message(event, command)
|
hook = self._find_command_hook(event["server"], command, False)
|
||||||
|
if hook:
|
||||||
|
self.command(event["server"], event["user"], False,
|
||||||
|
event["user"], command, event["message_split"][1:],
|
||||||
|
event["tags"], "", hook)
|
||||||
|
|
||||||
def _get_help(self, hook):
|
def _get_help(self, hook):
|
||||||
return hook.get_kwarg("help", None) or hook.docstring.description
|
return hook.get_kwarg("help", None) or hook.docstring.description
|
||||||
|
|
|
@ -20,37 +20,26 @@ class Module(ModuleManager.BaseModule):
|
||||||
return utils.irc.color(str(karma), utils.consts.LIGHTGREEN)
|
return utils.irc.color(str(karma), utils.consts.LIGHTGREEN)
|
||||||
return str(karma)
|
return str(karma)
|
||||||
|
|
||||||
|
|
||||||
@utils.hook("new.user")
|
@utils.hook("new.user")
|
||||||
def new_user(self, event):
|
def new_user(self, event):
|
||||||
event["user"].last_karma = None
|
event["user"].last_karma = None
|
||||||
|
|
||||||
@utils.hook("received.message.channel")
|
@utils.hook("command.regex")
|
||||||
def channel_message(self, event):
|
def channel_message(self, event):
|
||||||
match = re.match(REGEX_KARMA, event["message"].strip())
|
"""
|
||||||
if match and not event["action"]:
|
:command: karma
|
||||||
is_ignored_f = self.exports.get_one("is-ignored",
|
:pattern: ^(.*[^-+])[-+]*(\+{2,}|\-{2,})$
|
||||||
lambda _1, _2: False)
|
"""
|
||||||
if is_ignored_f(event["server"], event["user"], "karma"):
|
verbose = event["target"].get_setting("karma-verbose", False)
|
||||||
return
|
|
||||||
|
|
||||||
is_silenced_f = self.exports.get_one("is-silenced", lambda _: False)
|
|
||||||
if is_silenced_f(event["channel"]):
|
|
||||||
return
|
|
||||||
|
|
||||||
verbose = event["channel"].get_setting("karma-verbose", False)
|
|
||||||
nickname_only = event["server"].get_setting("karma-nickname-only",
|
nickname_only = event["server"].get_setting("karma-nickname-only",
|
||||||
False)
|
False)
|
||||||
|
|
||||||
if not event["user"].last_karma or (time.time()-event["user"
|
if not event["user"].last_karma or (time.time()-event["user"
|
||||||
].last_karma) >= KARMA_DELAY_SECONDS:
|
].last_karma) >= KARMA_DELAY_SECONDS:
|
||||||
target = match.group(1).strip().rstrip("".join(WORD_STOP))
|
target = event["match"].group(1).strip().rstrip("".join(WORD_STOP))
|
||||||
if event["server"].irc_lower(target) == event["user"].name:
|
if event["server"].irc_lower(target) == event["user"].name:
|
||||||
if verbose:
|
if verbose:
|
||||||
self.events.on("send.stderr").call(
|
event["stdout"].write("You cannot change your own karma")
|
||||||
module_name="Karma", target=event["channel"],
|
|
||||||
message="You cannot change your own karma",
|
|
||||||
server=event["server"])
|
|
||||||
return
|
return
|
||||||
|
|
||||||
setting = "karma-%s" % target
|
setting = "karma-%s" % target
|
||||||
|
@ -59,10 +48,10 @@ class Module(ModuleManager.BaseModule):
|
||||||
user = event["server"].get_user(target)
|
user = event["server"].get_user(target)
|
||||||
setting = "karma"
|
setting = "karma"
|
||||||
setting_target = user
|
setting_target = user
|
||||||
if not event["channel"].has_user(user):
|
if not event["target"].has_user(user):
|
||||||
return
|
return
|
||||||
|
|
||||||
positive = match.group(2)[0] == "+"
|
positive = event["match"].group(2)[0] == "+"
|
||||||
karma = setting_target.get_setting(setting, 0)
|
karma = setting_target.get_setting(setting, 0)
|
||||||
karma += 1 if positive else -1
|
karma += 1 if positive else -1
|
||||||
|
|
||||||
|
@ -73,15 +62,11 @@ class Module(ModuleManager.BaseModule):
|
||||||
|
|
||||||
karma_str = self._karma_str(karma)
|
karma_str = self._karma_str(karma)
|
||||||
if verbose:
|
if verbose:
|
||||||
self.events.on("send.stdout").call(
|
event["stdout"].write(
|
||||||
module_name="Karma", target=event["channel"],
|
"%s now has %s karma" % (target, karma_str))
|
||||||
message="%s now has %s karma" % (target, karma_str),
|
|
||||||
server=event["server"])
|
|
||||||
event["user"].last_karma = time.time()
|
event["user"].last_karma = time.time()
|
||||||
elif verbose:
|
elif verbose:
|
||||||
self.events.on("send.stderr").call(module_name="Karma",
|
event["stderr"].write("Try again in a couple of seconds")
|
||||||
target=event["channel"], server=event["server"],
|
|
||||||
message="Try again in a couple of seconds")
|
|
||||||
|
|
||||||
@utils.hook("received.command.karma")
|
@utils.hook("received.command.karma")
|
||||||
def karma(self, event):
|
def karma(self, event):
|
||||||
|
|
|
@ -12,19 +12,18 @@ REGEX_SED = re.compile("^s/")
|
||||||
"validate": utils.bool_or_none})
|
"validate": utils.bool_or_none})
|
||||||
class Module(ModuleManager.BaseModule):
|
class Module(ModuleManager.BaseModule):
|
||||||
def _closest_setting(self, event, setting, default):
|
def _closest_setting(self, event, setting, default):
|
||||||
return event["channel"].get_setting(setting,
|
return event["target"].get_setting(setting,
|
||||||
event["server"].get_setting(setting, default))
|
event["server"].get_setting(setting, default))
|
||||||
|
|
||||||
@utils.hook("received.message.channel")
|
@utils.hook("command.regex")
|
||||||
def channel_message(self, event):
|
def channel_message(self, event):
|
||||||
|
"""
|
||||||
|
:command: sed
|
||||||
|
:pattern: ^s/
|
||||||
|
"""
|
||||||
sed_split = re.split(REGEX_SPLIT, event["message"], 3)
|
sed_split = re.split(REGEX_SPLIT, event["message"], 3)
|
||||||
if event["message"].startswith("s/") and len(sed_split) > 2:
|
if event["message"].startswith("s/") and len(sed_split) > 2:
|
||||||
if event["action"] or not self._closest_setting(event, "sed",
|
if not self._closest_setting(event, "sed", False):
|
||||||
False):
|
|
||||||
return
|
|
||||||
is_ignored_f = short_url = self.exports.get_one("is-ignored",
|
|
||||||
lambda _1, _2: False)
|
|
||||||
if is_ignored_f(event["server"], event["user"], "sed"):
|
|
||||||
return
|
return
|
||||||
|
|
||||||
regex_flags = 0
|
regex_flags = 0
|
||||||
|
@ -50,15 +49,13 @@ class Module(ModuleManager.BaseModule):
|
||||||
pattern = re.compile(sed_split[1], regex_flags)
|
pattern = re.compile(sed_split[1], regex_flags)
|
||||||
except:
|
except:
|
||||||
traceback.print_exc()
|
traceback.print_exc()
|
||||||
self.events.on("send.stderr").call(target=event["channel"],
|
event["stderr"].write("Invalid regex in pattern")
|
||||||
module_name="Sed", server=event["server"],
|
|
||||||
message="Invalid regex in pattern")
|
|
||||||
return
|
return
|
||||||
replace = utils.irc.bold(sed_split[2].replace("\\/", "/"))
|
replace = utils.irc.bold(sed_split[2].replace("\\/", "/"))
|
||||||
|
|
||||||
for_user = event["user"].nickname if self._closest_setting(event,
|
for_user = event["user"].nickname if self._closest_setting(event,
|
||||||
"sed-sender-only", False) else None
|
"sed-sender-only", False) else None
|
||||||
line = event["channel"].buffer.find(pattern, from_self=False,
|
line = event["target"].buffer.find(pattern, from_self=False,
|
||||||
for_user=for_user, not_pattern=REGEX_SED)
|
for_user=for_user, not_pattern=REGEX_SED)
|
||||||
if line:
|
if line:
|
||||||
new_message = re.sub(pattern, replace, line.message, count)
|
new_message = re.sub(pattern, replace, line.message, count)
|
||||||
|
@ -66,6 +63,4 @@ class Module(ModuleManager.BaseModule):
|
||||||
prefix = "* %s" % line.sender
|
prefix = "* %s" % line.sender
|
||||||
else:
|
else:
|
||||||
prefix = "<%s>" % line.sender
|
prefix = "<%s>" % line.sender
|
||||||
self.events.on("send.stdout").call(target=event[
|
event["stdout"].write("%s %s" % (prefix, new_message))
|
||||||
"channel"], module_name="Sed", server=event["server"],
|
|
||||||
message="%s %s" % (prefix, new_message))
|
|
||||||
|
|
|
@ -44,24 +44,21 @@ class Module(ModuleManager.BaseModule):
|
||||||
else:
|
else:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@utils.hook("received.message.channel",
|
@utils.hook("command.regex",
|
||||||
priority=EventManager.PRIORITY_MONITOR)
|
priority=EventManager.PRIORITY_MONITOR)
|
||||||
def channel_message(self, event):
|
def channel_message(self, event):
|
||||||
match = re.search(utils.http.REGEX_URL, event["message"])
|
"""
|
||||||
if match and event["channel"].get_setting("auto-title", False):
|
:command: title
|
||||||
is_ignored_f = short_url = self.exports.get_one("is-ignored",
|
:pattern-url: 1
|
||||||
lambda _1, _2: False)
|
"""
|
||||||
if is_ignored_f(event["server"], event["user"], "title"):
|
url = event["match"].group(0)
|
||||||
return
|
title = self._get_title(event["target"], event["match"].group(0))
|
||||||
|
|
||||||
url = match.group(0)
|
|
||||||
title = self._get_title(event["channel"], match.group(0))
|
|
||||||
|
|
||||||
if title:
|
if title:
|
||||||
message = title
|
message = title
|
||||||
if event["channel"].get_setting("auto-title-first", False):
|
if event["target"].get_setting("auto-title-first", False):
|
||||||
setting = "url-last-%s" % self._url_hash(url)
|
setting = "url-last-%s" % self._url_hash(url)
|
||||||
first_details = event["channel"].get_setting(setting, None)
|
first_details = event["target"].get_setting(setting, None)
|
||||||
|
|
||||||
if first_details:
|
if first_details:
|
||||||
first_nickname, first_timestamp, _ = first_details
|
first_nickname, first_timestamp, _ = first_details
|
||||||
|
@ -70,14 +67,10 @@ class Module(ModuleManager.BaseModule):
|
||||||
message = "%s (first posted by %s at %s)" % (title,
|
message = "%s (first posted by %s at %s)" % (title,
|
||||||
first_nickname, timestamp_human)
|
first_nickname, timestamp_human)
|
||||||
else:
|
else:
|
||||||
event["channel"].set_setting(setting,
|
event["target"].set_setting(setting,
|
||||||
[event["user"].nickname, utils.iso8601_format_now(),
|
[event["user"].nickname, utils.iso8601_format_now(),
|
||||||
url])
|
url])
|
||||||
|
event["stdout"].write(message)
|
||||||
|
|
||||||
self.events.on("send.stdout").call(target=event["channel"],
|
|
||||||
message=message, module_name="Title",
|
|
||||||
server=event["server"])
|
|
||||||
|
|
||||||
@utils.hook("received.command.t", alias_of="title")
|
@utils.hook("received.command.t", alias_of="title")
|
||||||
@utils.hook("received.command.title", usage="[URL]")
|
@utils.hook("received.command.title", usage="[URL]")
|
||||||
|
|
|
@ -125,20 +125,17 @@ class Module(ModuleManager.BaseModule):
|
||||||
else:
|
else:
|
||||||
event["stderr"].write("No search phrase provided")
|
event["stderr"].write("No search phrase provided")
|
||||||
|
|
||||||
@utils.hook("received.message.channel",
|
@utils.hook("command.regex",
|
||||||
priority=EventManager.PRIORITY_LOW)
|
priority=EventManager.PRIORITY_LOW)
|
||||||
def channel_message(self, event):
|
def channel_message(self, event):
|
||||||
match = re.search(REGEX_YOUTUBE, event["message"])
|
"""
|
||||||
if match and event["channel"].get_setting("auto-youtube", False):
|
:command: youtube
|
||||||
is_ignored_f = short_url = self.exports.get_one("is-ignored",
|
:-pattern: https?://(?:www.)?
|
||||||
lambda _1, _2: False)
|
(?:youtu.be/|youtube.com/watch\?[\S]*v=)([\w\-]{11})
|
||||||
if is_ignored_f(event["server"], event["user"], "youtube"):
|
"""
|
||||||
return
|
if event["target"].get_setting("auto-youtube", False):
|
||||||
|
youtube_id = event["match"].group(1)
|
||||||
youtube_id = match.group(1)
|
|
||||||
video_details = self.video_details(youtube_id)
|
video_details = self.video_details(youtube_id)
|
||||||
if video_details:
|
if video_details:
|
||||||
self.events.on("send.stdout").call(target=event["channel"],
|
event["stdout"].write(video_details)
|
||||||
message=video_details, module_name="Youtube",
|
|
||||||
server=event["server"])
|
|
||||||
event.eat()
|
event.eat()
|
||||||
|
|
Loading…
Reference in a new issue