2016-04-04 17:42:37 +00:00
|
|
|
import base64, os
|
|
|
|
import scrypt
|
|
|
|
|
2018-08-28 17:16:19 +00:00
|
|
|
REQUIRES_IDENTIFY = ("You need to be identified to use that command "
|
|
|
|
"(/msg %s register | /msg %s identify)")
|
|
|
|
|
2016-04-04 17:42:37 +00:00
|
|
|
class Module(object):
|
2018-09-02 18:54:45 +00:00
|
|
|
def __init__(self, bot, events, exports):
|
2016-04-04 17:42:37 +00:00
|
|
|
self.bot = bot
|
2018-09-04 08:59:18 +00:00
|
|
|
events.on("new.user").hook(self.new_user)
|
|
|
|
events.on("preprocess.command").hook(
|
2016-04-06 17:23:02 +00:00
|
|
|
self.preprocess_command)
|
2018-09-04 08:59:18 +00:00
|
|
|
events.on("received.part").hook(self.on_part)
|
|
|
|
events.on("received.nick").hook(self.on_nick)
|
2018-09-03 10:30:54 +00:00
|
|
|
|
2018-08-31 11:55:52 +00:00
|
|
|
events.on("received").on("command").on("identify"
|
2016-04-06 11:02:44 +00:00
|
|
|
).hook(self.identify, private_only=True, min_args=1,
|
|
|
|
usage="<password>", help="Identify yourself")
|
2018-08-31 11:55:52 +00:00
|
|
|
events.on("received").on("command").on("register"
|
2016-04-06 11:02:44 +00:00
|
|
|
).hook(self.register, private_only=True, min_args=1,
|
|
|
|
usage="<password>", help="Register your nickname")
|
2018-08-31 11:55:52 +00:00
|
|
|
events.on("received.command.logout").hook(self.logout,
|
2018-08-28 15:53:21 +00:00
|
|
|
private_only=True, help="Sign out from the bot")
|
2018-09-03 10:30:54 +00:00
|
|
|
events.on("received.command.resetpassword").hook(
|
|
|
|
self.reset_password, private_only=True,
|
|
|
|
help="Reset a user's password", min_args=2,
|
|
|
|
usage="<nickname> <password>", permission="resetpassword")
|
2018-08-28 15:53:21 +00:00
|
|
|
|
2018-08-31 11:55:52 +00:00
|
|
|
events.on("received.command.mypermissions").hook(
|
2018-08-28 15:53:21 +00:00
|
|
|
self.my_permissions, authenticated=True)
|
2018-08-31 11:55:52 +00:00
|
|
|
events.on("received.command.givepermission").hook(
|
2018-08-28 15:53:21 +00:00
|
|
|
self.give_permission, min_args=2, permission="givepermission")
|
2018-08-31 11:55:52 +00:00
|
|
|
events.on("received.command.removepermission").hook(
|
2018-08-28 15:53:21 +00:00
|
|
|
self.remove_permission, min_args=2, permission="removepermission")
|
2016-04-04 17:42:37 +00:00
|
|
|
|
|
|
|
def new_user(self, event):
|
|
|
|
self._logout(event["user"])
|
|
|
|
|
|
|
|
def on_part(self, event):
|
2016-04-04 17:48:39 +00:00
|
|
|
if len(event["user"].channels) == 1 and event["user"].identified:
|
2016-04-04 17:42:37 +00:00
|
|
|
event["user"].send_notice("You no longer share any channels "
|
|
|
|
"with me so you have been signed out")
|
|
|
|
|
2018-09-04 08:59:18 +00:00
|
|
|
def on_nick(self, event):
|
|
|
|
if event["user"].identified:
|
|
|
|
self._logout(event["user"])
|
|
|
|
event["user"].send_notice("You've changed nickname so you "
|
|
|
|
"have been signed out")
|
|
|
|
|
2016-04-04 17:42:37 +00:00
|
|
|
def _get_hash(self, user):
|
|
|
|
hash, salt = user.get_setting("authentication", (None, None))
|
|
|
|
return hash, salt
|
2016-04-06 17:23:02 +00:00
|
|
|
|
2016-04-04 17:42:37 +00:00
|
|
|
def _make_salt(self):
|
|
|
|
return base64.b64encode(os.urandom(64)).decode("utf8")
|
2016-04-06 17:23:02 +00:00
|
|
|
|
2016-04-04 17:42:37 +00:00
|
|
|
def _make_hash(self, password, salt=None):
|
|
|
|
salt = salt or self._make_salt()
|
|
|
|
hash = base64.b64encode(scrypt.hash(password, salt)).decode("utf8")
|
|
|
|
return hash, salt
|
2016-04-06 17:23:02 +00:00
|
|
|
|
2016-04-04 17:42:37 +00:00
|
|
|
def _identified(self, user):
|
|
|
|
user.identified = True
|
2016-04-06 17:23:02 +00:00
|
|
|
|
2016-04-04 17:42:37 +00:00
|
|
|
def _logout(self, user):
|
|
|
|
user.identified = False
|
2016-04-06 17:23:02 +00:00
|
|
|
|
2016-04-04 17:42:37 +00:00
|
|
|
def identify(self, event):
|
2016-04-04 17:48:39 +00:00
|
|
|
if not event["user"].channels:
|
|
|
|
event["stderr"].write("You must share at least one channel "
|
|
|
|
"with me before you can identify")
|
|
|
|
return
|
2016-04-04 17:42:37 +00:00
|
|
|
if not event["user"].identified:
|
|
|
|
password = event["args_split"][0]
|
|
|
|
hash, salt = self._get_hash(event["user"])
|
|
|
|
if hash and salt:
|
|
|
|
attempt, _ = self._make_hash(password, salt)
|
|
|
|
if attempt == hash:
|
|
|
|
self._identified(event["user"])
|
|
|
|
event["stdout"].write("Correct password, you have "
|
|
|
|
"been identified.")
|
|
|
|
else:
|
|
|
|
event["stderr"].write("Incorrect password")
|
|
|
|
else:
|
|
|
|
event["stderr"].write("This nickname is not registered")
|
|
|
|
else:
|
|
|
|
event["stderr"].write("You are already identified")
|
|
|
|
|
|
|
|
def register(self, event):
|
|
|
|
hash, salt = self._get_hash(event["user"])
|
|
|
|
if not hash and not salt:
|
|
|
|
password = event["args_split"][0]
|
|
|
|
hash, salt = self._make_hash(password)
|
|
|
|
event["user"].set_setting("authentication", [hash, salt])
|
|
|
|
self._identified(event["user"])
|
|
|
|
event["stdout"].write("Nickname registered successfully")
|
|
|
|
else:
|
|
|
|
event["stderr"].write("This nickname is already registered")
|
|
|
|
|
|
|
|
def logout(self, event):
|
|
|
|
if event["user"].identified:
|
|
|
|
self._logout(event["user"])
|
|
|
|
event["stdout"].write("You have been logged out")
|
|
|
|
else:
|
|
|
|
event["stderr"].write("You are not logged in")
|
2016-04-06 17:23:02 +00:00
|
|
|
|
2018-09-03 10:30:54 +00:00
|
|
|
def reset_password(self, event):
|
|
|
|
target = event["server"].get_user(event["args_split"][0])
|
|
|
|
password = " ".join(event["args_split"][1:])
|
|
|
|
registered = target.get_setting("authentication", None)
|
|
|
|
|
|
|
|
if registered == None:
|
|
|
|
event["stderr"].write("'%s' isn't registered" % target.nickname)
|
|
|
|
else:
|
|
|
|
hash, salt = self._make_hash(password)
|
|
|
|
target.set_setting("authentication", [hash, salt])
|
|
|
|
event["stdout"].write("Reset password for '%s'" %
|
|
|
|
target.nickname)
|
|
|
|
|
2016-04-06 17:23:02 +00:00
|
|
|
def preprocess_command(self, event):
|
2018-08-28 17:16:19 +00:00
|
|
|
authentication = event["user"].get_setting("authentication", None)
|
2016-04-06 17:23:02 +00:00
|
|
|
permission = event["hook"].kwargs.get("permission", None)
|
2018-08-18 17:28:41 +00:00
|
|
|
authenticated = event["hook"].kwargs.get("authenticated", False)
|
2018-08-28 17:16:19 +00:00
|
|
|
protect_registered = event["hook"].kwargs.get("protect_registered",
|
|
|
|
False)
|
|
|
|
|
2018-08-02 22:00:42 +00:00
|
|
|
if permission:
|
|
|
|
identified = event["user"].identified
|
|
|
|
user_permissions = event["user"].get_setting("permissions", [])
|
|
|
|
has_permission = permission and (
|
|
|
|
permission in user_permissions or "*" in user_permissions)
|
|
|
|
if not identified or not has_permission:
|
|
|
|
return "You do not have permission to do that"
|
2018-08-18 17:28:41 +00:00
|
|
|
elif authenticated:
|
|
|
|
if not event["user"].identified:
|
2018-08-28 17:16:19 +00:00
|
|
|
return REQUIRES_IDENTIFY % (event["server"].nickname,
|
|
|
|
event["server"].nickname)
|
|
|
|
elif protect_registered:
|
|
|
|
if authentication and not event["user"].identified:
|
|
|
|
return REQUIRES_IDENTIFY % (event["server"].nickname,
|
|
|
|
event["server"].nickname)
|
2016-05-17 13:50:48 +00:00
|
|
|
|
|
|
|
def my_permissions(self, event):
|
2018-08-02 22:00:42 +00:00
|
|
|
permissions = event["user"].get_setting("permissions", [])
|
2016-05-17 13:50:48 +00:00
|
|
|
event["stdout"].write("Your permissions: %s" % ", ".join(permissions))
|
2018-08-28 15:53:21 +00:00
|
|
|
|
|
|
|
def _get_user_details(self, server, nickname):
|
|
|
|
target = server.get_user(nickname)
|
|
|
|
registered = bool(target.get_setting("authentication", None))
|
|
|
|
permissions = target.get_setting("permissions", [])
|
|
|
|
return [target, registered, permissions]
|
|
|
|
|
|
|
|
def give_permission(self, event):
|
|
|
|
permission = event["args_split"][1].lower()
|
|
|
|
target, registered, permissions = self._get_user_details(
|
|
|
|
event["server"], event["args_split"][0])
|
|
|
|
|
|
|
|
if not registered:
|
|
|
|
event["stderr"].write("%s isn't registered" % target.nickname)
|
|
|
|
return
|
|
|
|
|
|
|
|
if permission in permissions:
|
|
|
|
event["stderr"].write("%s already has permission '%s'" % (
|
|
|
|
target.nickname, permission))
|
|
|
|
else:
|
|
|
|
permissions.append(permission)
|
|
|
|
target.set_setting("permissions", permissions)
|
|
|
|
event["stdout"].write("Gave permission '%s' to %s" % (
|
|
|
|
permission, target.nickname))
|
|
|
|
def remove_permission(self, event):
|
|
|
|
permission = event["args_split"][1].lower()
|
|
|
|
target, registered, permissions = self._get_user_details(
|
|
|
|
event["server"], event["args_split"][0])
|
|
|
|
|
|
|
|
if not registered:
|
|
|
|
event["stderr"].write("%s isn't registered" % target.nickname)
|
|
|
|
return
|
|
|
|
|
|
|
|
if not permission in permissions:
|
|
|
|
event["stderr"].write("%s already has permission '%s'" % (
|
|
|
|
target.nickname, permission))
|
|
|
|
else:
|
|
|
|
permissions.remove(permission)
|
2018-08-29 13:34:52 +00:00
|
|
|
if not permissions:
|
|
|
|
target.del_setting("permissions")
|
|
|
|
else:
|
|
|
|
target.set_setting("permissions", permissions)
|
2018-08-28 15:53:21 +00:00
|
|
|
event["stdout"].write("Removed permission '%s' from %s" % (
|
|
|
|
permission, target.nickname))
|