crafty-4/app/classes/web/public_handler.py

265 lines
9.8 KiB
Python
Raw Normal View History

2020-08-11 20:36:09 -04:00
import logging
2024-02-19 20:18:47 -05:00
import json
2024-02-19 12:03:02 -05:00
import nh3
2024-02-19 20:18:47 -05:00
from jsonschema import validate
from jsonschema.exceptions import ValidationError
2020-08-11 20:36:09 -04:00
from app.classes.shared.helpers import Helpers
from app.classes.models.users import HelperUsers
from app.classes.web.base_handler import BaseHandler
2020-08-11 20:36:09 -04:00
2022-03-08 04:40:44 +00:00
logger = logging.getLogger(__name__)
2023-11-05 13:26:27 -05:00
auth_log = logging.getLogger("auth")
2020-08-11 20:36:09 -04:00
class PublicHandler(BaseHandler):
2022-01-15 02:23:50 +02:00
def set_current_user(self, user_id: str = None):
expire_days = self.helper.get_setting("cookie_expire")
2020-08-11 20:36:09 -04:00
# if helper comes back with false
if not expire_days:
expire_days = "5"
2022-01-15 02:23:50 +02:00
if user_id is not None:
self.set_cookie(
2022-04-11 11:08:36 +01:00
"token",
self.controller.authentication.generate(user_id),
expires_days=int(expire_days),
)
2020-08-11 20:36:09 -04:00
else:
self.clear_cookie("token")
# self.clear_cookie("user")
# self.clear_cookie("user_data")
2020-08-11 20:36:09 -04:00
def get(self, page=None):
# pylint: disable=no-member
error = nh3.clean(self.get_argument("error", "Invalid Login!"))
error_msg = nh3.clean(self.get_argument("error_msg", ""))
# pylint: enable=no-member
page_data = {
"version": self.helper.get_version_string(),
"error": error,
"lang": self.helper.get_setting("language"),
"lang_page": self.helper.get_lang_page(self.helper.get_setting("language")),
"query": "",
"background": self.controller.cached_login,
"login_opacity": self.controller.management.get_login_opacity(),
}
2022-03-19 01:48:24 +00:00
if self.request.query:
2024-02-19 20:18:47 -05:00
page_data["query"] = self.request.query_arguments.get("next")[0].decode()
2020-08-13 10:38:36 -04:00
# sensible defaults
template = "public/404.html"
2020-08-11 20:36:09 -04:00
if page == "login":
template = "public/login.html"
2020-08-23 18:43:28 -04:00
2023-03-06 23:38:04 +00:00
elif page == "404":
2020-08-23 18:43:28 -04:00
template = "public/404.html"
elif page == "error":
template = "public/error.html"
2020-08-11 20:36:09 -04:00
2023-03-06 23:38:04 +00:00
elif page == "offline":
template = "public/offline.html"
elif page == "logout":
2024-01-28 13:44:44 -05:00
exec_user = self.get_current_user()
self.clear_cookie("token")
2024-01-28 13:44:44 -05:00
# Delete anti-lockout-user on lockout...it's one time use
if exec_user[2]["username"] == "anti-lockout-user":
self.controller.users.stop_anti_lockout()
# self.clear_cookie("user")
# self.clear_cookie("user_data")
self.redirect("/login")
return
# if we have no page, let's go to login
2020-08-11 20:36:09 -04:00
else:
2024-02-19 20:18:47 -05:00
return self.redirect("/login")
2020-08-11 20:36:09 -04:00
self.render(
template,
data=page_data,
translate=self.translator.translate,
error_msg=error_msg,
)
2020-08-11 20:36:09 -04:00
def post(self, page=None):
2024-02-19 20:18:47 -05:00
login_schema = {
"type": "object",
"properties": {
"username": {
"type": "string",
},
"password": {"type": "string"},
},
"required": ["username", "password"],
"additionalProperties": False,
}
try:
data = json.loads(self.request.body)
except json.decoder.JSONDecodeError as e:
logger.error(
"Invalid JSON schema for API"
f" login attempt from {self.get_remote_ip()}"
)
return self.finish_json(
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
)
try:
validate(data, login_schema)
except ValidationError as e:
logger.error(
"Invalid JSON schema for API"
f" login attempt from {self.get_remote_ip()}"
)
return self.finish_json(
400,
{
"status": "error",
2024-02-20 18:50:32 -05:00
"error": "VWggb2ghIFN0aW5reS 🪠",
2024-02-19 20:18:47 -05:00
"error_data": str(e),
},
)
page_data = {
"version": self.helper.get_version_string(),
"lang": self.helper.get_setting("language"),
2022-04-14 19:34:21 -04:00
"lang_page": self.helper.get_lang_page(self.helper.get_setting("language")),
"query": "",
}
2022-03-19 01:48:24 +00:00
if self.request.query:
2024-02-19 20:18:47 -05:00
page_data["query"] = self.request.query_arguments.get("next")[0].decode()
if page == "login":
2024-02-19 20:18:47 -05:00
data = json.loads(self.request.body)
2023-11-05 13:26:27 -05:00
auth_log.info(
f"User attempting to authenticate from {self.get_remote_ip()}"
)
2024-02-19 20:18:47 -05:00
entered_username = nh3.clean(data["username"]) # pylint: disable=no-member
2024-02-20 18:50:32 -05:00
entered_password = data["password"]
2022-04-12 18:33:00 -04:00
try:
user_id = HelperUsers.get_user_id_by_name(entered_username.lower())
user_data = HelperUsers.get_user_model(user_id)
2022-04-12 18:33:00 -04:00
except:
2023-11-05 13:26:27 -05:00
self.controller.log_attempt(self.get_remote_ip(), entered_username)
auth_log.error(
f"User attempted to log into {entered_username}."
f" Authentication failed from remote IP {self.get_remote_ip()}"
" Users does not exist."
)
2024-02-19 20:18:47 -05:00
self.finish_json(
403,
{
"status": "error",
"error": self.helper.translation.translate(
"login", "incorrect", self.helper.get_setting("language")
),
},
)
2022-04-12 18:33:00 -04:00
# self.clear_cookie("user")
# self.clear_cookie("user_data")
2024-02-19 20:18:47 -05:00
return self.clear_cookie("token")
2020-08-13 10:38:36 -04:00
# if we don't have a user
if not user_data:
2023-11-05 13:26:27 -05:00
auth_log.error(
f"User attempted to log into {entered_username}. Authentication"
f" failed from remote IP {self.get_remote_ip()}"
" User does not exist."
)
self.controller.log_attempt(self.get_remote_ip(), entered_username)
2024-02-19 20:18:47 -05:00
self.finish_json(
403,
{
"status": "error",
"error": self.helper.translation.translate(
"login", "incorrect", self.helper.get_setting("language")
),
},
)
# self.clear_cookie("user")
# self.clear_cookie("user_data")
2024-02-19 20:18:47 -05:00
return self.clear_cookie("token")
2020-08-13 10:38:36 -04:00
# if they are disabled
if not user_data.enabled:
2023-11-05 13:26:27 -05:00
auth_log.error(
f"User attempted to log into {entered_username}. "
f"Authentication failed from remote IP {self.get_remote_ip()}."
" User account disabled"
)
self.controller.log_attempt(self.get_remote_ip(), entered_username)
2024-02-19 20:18:47 -05:00
self.finish_json(
403,
{
"status": "error",
"error": self.helper.translation.translate(
"login", "disabled", self.helper.get_setting("language")
),
},
)
# self.clear_cookie("user")
# self.clear_cookie("user_data")
2024-02-19 20:18:47 -05:00
return self.clear_cookie("token")
login_result = self.helper.verify_pass(entered_password, user_data.password)
# Valid Login
if login_result:
2022-01-15 02:23:50 +02:00
self.set_current_user(user_data.user_id)
logger.info(
f"User: {user_data} Logged in from IP: {self.get_remote_ip()}"
)
2022-06-12 17:17:58 -04:00
if not user_data.last_ip and user_data.username == "admin":
2022-06-12 17:03:42 -04:00
self.controller.first_login = True
2020-08-13 10:38:36 -04:00
# record this login
user_data.last_ip = self.get_remote_ip()
user_data.last_login = Helpers.get_time_as_string()
user_data.save()
2023-11-05 13:26:27 -05:00
auth_log.info(
f"{entered_username} successfully"
" authenticated and logged"
f" into panel from remote IP {self.get_remote_ip()}"
)
# log this login
self.controller.management.add_to_audit_log(
user_data.user_id, "Logged in", 0, self.get_remote_ip()
)
2024-02-19 20:18:47 -05:00
return self.finish_json(
200, {"status": "ok", "data": {"message": "login successful!"}}
)
2024-02-21 00:04:07 -05:00
# We'll continue on and handle unsuccessful logins
auth_log.error(
f"User attempted to log into {entered_username}."
f" Authentication failed from remote IP {self.get_remote_ip()}"
)
self.controller.log_attempt(self.get_remote_ip(), entered_username)
# self.clear_cookie("user")
# self.clear_cookie("user_data")
self.clear_cookie("token")
error_msg = self.helper.translation.translate(
"login", "incorrect", self.helper.get_setting("language")
)
if entered_password == "app/config/default-creds.txt":
error_msg += ". "
error_msg += self.helper.translation.translate(
"login", "defaultPath", self.helper.get_setting("language")
2024-02-19 20:18:47 -05:00
)
2024-02-21 00:04:07 -05:00
# log this failed login attempt
self.controller.management.add_to_audit_log(
user_data.user_id, "Tried to log in", 0, self.get_remote_ip()
)
return self.finish_json(
403,
{"status": "error", "error": error_msg},
)
else:
2024-02-19 20:18:47 -05:00
self.redirect("/login?")