2021-10-12 04:22:30 -05:00
|
|
|
##########################################################################
|
|
|
|
#
|
|
|
|
# pgAdmin 4 - PostgreSQL Tools
|
|
|
|
#
|
2023-01-02 00:23:55 -06:00
|
|
|
# Copyright (C) 2013 - 2023, The pgAdmin Development Team
|
2021-10-12 04:22:30 -05:00
|
|
|
# This software is released under the PostgreSQL Licence
|
|
|
|
#
|
|
|
|
##########################################################################
|
|
|
|
|
|
|
|
"""A blueprint module implementing the Webserver authentication."""
|
|
|
|
|
2022-08-12 06:40:26 -05:00
|
|
|
import secrets
|
2021-10-12 04:22:30 -05:00
|
|
|
import string
|
|
|
|
import config
|
|
|
|
from flask import request, current_app, session, Response, render_template, \
|
|
|
|
url_for
|
2021-11-24 05:52:57 -06:00
|
|
|
from flask_babel import gettext
|
2021-10-12 04:22:30 -05:00
|
|
|
from flask_security import login_user
|
|
|
|
from .internal import BaseAuthentication
|
|
|
|
from pgadmin.model import User
|
|
|
|
from pgadmin.tools.user_management import create_user
|
|
|
|
from pgadmin.utils.constants import WEBSERVER
|
|
|
|
from pgadmin.utils import PgAdminModule
|
|
|
|
from pgadmin.utils.csrf import pgCSRFProtect
|
|
|
|
from flask_security.utils import logout_user
|
|
|
|
|
|
|
|
|
|
|
|
class WebserverModule(PgAdminModule):
|
2021-11-24 05:52:57 -06:00
|
|
|
def register(self, app, options):
|
2021-10-12 04:22:30 -05:00
|
|
|
# Do not look for the sub_modules,
|
|
|
|
# instead call blueprint.register(...) directly
|
2022-11-18 22:43:41 -06:00
|
|
|
super().register(app, options)
|
2021-10-12 04:22:30 -05:00
|
|
|
|
|
|
|
def get_exposed_url_endpoints(self):
|
|
|
|
return ['webserver.login',
|
|
|
|
'webserver.logout']
|
|
|
|
|
|
|
|
|
|
|
|
def init_app(app):
|
|
|
|
MODULE_NAME = 'webserver'
|
|
|
|
|
|
|
|
blueprint = WebserverModule(MODULE_NAME, __name__, static_url_path='')
|
|
|
|
|
|
|
|
@blueprint.route("/login",
|
|
|
|
endpoint="login", methods=["GET"])
|
|
|
|
@pgCSRFProtect.exempt
|
|
|
|
def webserver_login():
|
|
|
|
logout_user()
|
|
|
|
return Response(render_template("browser/kerberos_login.html",
|
|
|
|
login_url=url_for('security.login'),
|
|
|
|
))
|
|
|
|
|
|
|
|
@blueprint.route("/logout",
|
|
|
|
endpoint="logout", methods=["GET"])
|
|
|
|
@pgCSRFProtect.exempt
|
|
|
|
def webserver_logout():
|
|
|
|
logout_user()
|
|
|
|
return Response(render_template("browser/kerberos_logout.html",
|
|
|
|
login_url=url_for('security.login'),
|
|
|
|
))
|
|
|
|
|
|
|
|
app.register_blueprint(blueprint)
|
|
|
|
|
|
|
|
|
|
|
|
class WebserverAuthentication(BaseAuthentication):
|
|
|
|
LOGIN_VIEW = 'webserver.login'
|
|
|
|
LOGOUT_VIEW = 'webserver.logout'
|
|
|
|
|
|
|
|
def get_source_name(self):
|
|
|
|
return WEBSERVER
|
|
|
|
|
|
|
|
def get_friendly_name(self):
|
|
|
|
return gettext("webserver")
|
|
|
|
|
|
|
|
def validate(self, form):
|
2022-02-16 02:04:24 -06:00
|
|
|
return True, None
|
2021-10-12 04:22:30 -05:00
|
|
|
|
|
|
|
def get_user(self):
|
2021-11-10 04:08:41 -06:00
|
|
|
username = request.environ.get(config.WEBSERVER_REMOTE_USER)
|
|
|
|
if not username:
|
|
|
|
# One more try to get the Remote User from the hearders
|
|
|
|
username = request.headers.get(config.WEBSERVER_REMOTE_USER)
|
|
|
|
return username
|
2021-10-12 04:22:30 -05:00
|
|
|
|
|
|
|
def authenticate(self, form):
|
|
|
|
username = self.get_user()
|
|
|
|
|
|
|
|
if not username:
|
|
|
|
return False, gettext(
|
|
|
|
"Webserver authenticate failed.")
|
|
|
|
|
|
|
|
session['pass_enc_key'] = ''.join(
|
2022-08-12 06:40:26 -05:00
|
|
|
(secrets.choice(string.ascii_lowercase) for _ in range(10)))
|
2021-10-12 04:22:30 -05:00
|
|
|
useremail = request.environ.get('mail')
|
|
|
|
if not useremail:
|
|
|
|
useremail = ''
|
|
|
|
return self.__auto_create_user(username, '')
|
|
|
|
|
|
|
|
def login(self, form):
|
|
|
|
username = self.get_user()
|
|
|
|
if username:
|
|
|
|
user = User.query.filter_by(username=username).first()
|
|
|
|
status = login_user(user)
|
|
|
|
if not status:
|
|
|
|
current_app.logger.exception(self.messages('LOGIN_FAILED'))
|
|
|
|
return False, self.messages('LOGIN_FAILED')
|
2023-02-12 23:41:05 -06:00
|
|
|
current_app.logger.info(
|
|
|
|
"Webserver user {0} logged in.".format(username))
|
2021-10-12 04:22:30 -05:00
|
|
|
return True, None
|
|
|
|
return False, self.messages('LOGIN_FAILED')
|
|
|
|
|
|
|
|
def __auto_create_user(self, username, useremail):
|
|
|
|
"""Add the webserver user to the internal SQLite database."""
|
|
|
|
if config.WEBSERVER_AUTO_CREATE_USER:
|
|
|
|
user = User.query.filter_by(username=username).first()
|
|
|
|
if not user:
|
2023-02-12 23:41:05 -06:00
|
|
|
create_msg = ("Creating user {0} with email {1} "
|
|
|
|
"from auth source Webserver.")
|
|
|
|
current_app.logger.info(create_msg.format(username,
|
|
|
|
useremail))
|
2021-10-12 04:22:30 -05:00
|
|
|
return create_user({
|
|
|
|
'username': username,
|
|
|
|
'email': useremail,
|
|
|
|
'role': 2,
|
|
|
|
'active': True,
|
|
|
|
'auth_source': WEBSERVER
|
|
|
|
})
|
|
|
|
return True, None
|