2016-03-07 05:48:24 -06:00
|
|
|
##########################################################################
|
|
|
|
#
|
|
|
|
# pgAdmin 4 - PostgreSQL Tools
|
|
|
|
#
|
2024-01-01 02:43:48 -06:00
|
|
|
# Copyright (C) 2013 - 2024, The pgAdmin Development Team
|
2016-03-07 05:48:24 -06:00
|
|
|
# This software is released under the PostgreSQL Licence
|
|
|
|
#
|
|
|
|
##########################################################################
|
|
|
|
|
|
|
|
"""
|
|
|
|
Implements the routes for creating Preferences/Options Dialog on the client
|
|
|
|
side and for getting/setting preferences.
|
|
|
|
"""
|
|
|
|
|
2018-03-19 12:09:19 -05:00
|
|
|
import config
|
2023-02-14 23:40:12 -06:00
|
|
|
import json
|
2017-03-24 09:20:10 -05:00
|
|
|
from flask import render_template, url_for, Response, request, session
|
2021-11-24 05:52:57 -06:00
|
|
|
from flask_babel import gettext
|
2016-07-22 10:25:23 -05:00
|
|
|
from flask_security import login_required
|
2016-03-07 05:48:24 -06:00
|
|
|
from pgadmin.utils import PgAdminModule
|
|
|
|
from pgadmin.utils.ajax import success_return, \
|
|
|
|
make_response as ajax_response, internal_server_error
|
2023-12-19 04:22:57 -06:00
|
|
|
from pgadmin.utils.ajax import make_json_response
|
2016-03-07 05:48:24 -06:00
|
|
|
from pgadmin.utils.preferences import Preferences
|
2020-08-20 09:56:51 -05:00
|
|
|
from pgadmin.utils.constants import MIMETYPE_APP_JS
|
2020-09-03 02:29:28 -05:00
|
|
|
from pgadmin.browser.server_groups import ServerGroupModule as sgm
|
2016-03-07 05:48:24 -06:00
|
|
|
|
|
|
|
MODULE_NAME = 'preferences'
|
|
|
|
|
2016-06-21 08:21:06 -05:00
|
|
|
|
2016-03-07 05:48:24 -06:00
|
|
|
class PreferencesModule(PgAdminModule):
|
|
|
|
"""
|
|
|
|
PreferenceModule represets the preferences of different modules to the
|
|
|
|
user in UI.
|
|
|
|
|
|
|
|
And, allows the user to modify (not add/remove) as per their requirement.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def get_own_menuitems(self):
|
2022-03-21 02:59:26 -05:00
|
|
|
return {}
|
2016-03-07 05:48:24 -06:00
|
|
|
|
2017-06-12 01:31:22 -05:00
|
|
|
def get_exposed_url_endpoints(self):
|
|
|
|
"""
|
|
|
|
Returns:
|
|
|
|
list: a list of url endpoints exposed to the client.
|
|
|
|
"""
|
2018-02-19 05:12:35 -06:00
|
|
|
return [
|
|
|
|
'preferences.index',
|
|
|
|
'preferences.get_by_name',
|
2023-12-19 04:22:57 -06:00
|
|
|
'preferences.get_all',
|
2023-12-21 00:37:26 -06:00
|
|
|
'preferences.get_all_cli',
|
2023-12-19 04:22:57 -06:00
|
|
|
'preferences.update_pref'
|
2018-02-19 05:12:35 -06:00
|
|
|
]
|
2016-03-07 05:48:24 -06:00
|
|
|
|
|
|
|
|
2017-06-12 01:31:22 -05:00
|
|
|
blueprint = PreferencesModule(MODULE_NAME, __name__)
|
2016-03-07 05:48:24 -06:00
|
|
|
|
|
|
|
|
|
|
|
@blueprint.route("/preferences.js")
|
|
|
|
@login_required
|
|
|
|
def script():
|
|
|
|
"""render the required javascript"""
|
2018-02-19 05:12:35 -06:00
|
|
|
return Response(
|
|
|
|
response=render_template("preferences/preferences.js", _=gettext),
|
|
|
|
status=200,
|
2020-08-20 09:56:51 -05:00
|
|
|
mimetype=MIMETYPE_APP_JS
|
2018-02-19 05:12:35 -06:00
|
|
|
)
|
2016-03-07 05:48:24 -06:00
|
|
|
|
|
|
|
|
2017-06-12 01:31:22 -05:00
|
|
|
@blueprint.route("/", methods=["GET"], endpoint='index')
|
|
|
|
@blueprint.route("/<module>/<preference>", endpoint='get_by_name')
|
2016-03-07 05:48:24 -06:00
|
|
|
@login_required
|
2016-08-08 05:59:37 -05:00
|
|
|
def preferences(module=None, preference=None):
|
|
|
|
"""Fetch all/or requested preferences of pgAdmin IV."""
|
|
|
|
|
|
|
|
if module is not None and preference is not None:
|
|
|
|
try:
|
|
|
|
m = Preferences.module(module, create=False)
|
|
|
|
if m is None:
|
|
|
|
return Response(status=404)
|
|
|
|
|
|
|
|
p = m.preference(preference)
|
|
|
|
if p is None:
|
|
|
|
return Response(status=404)
|
|
|
|
|
|
|
|
return ajax_response(
|
|
|
|
response=p.to_json(),
|
|
|
|
status=200
|
|
|
|
)
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
return internal_server_error(errormsg=str(e))
|
2016-03-07 05:48:24 -06:00
|
|
|
|
|
|
|
# Load Preferences
|
2016-08-08 05:59:37 -05:00
|
|
|
pref = Preferences.preferences()
|
2016-03-07 05:48:24 -06:00
|
|
|
res = []
|
|
|
|
|
2016-04-19 13:16:50 -05:00
|
|
|
def label(p):
|
2017-03-24 09:20:10 -05:00
|
|
|
return gettext(p['label'])
|
2016-04-19 13:16:50 -05:00
|
|
|
|
2020-08-10 06:00:07 -05:00
|
|
|
_group_pref_by_categories(pref, res, label)
|
2016-03-07 05:48:24 -06:00
|
|
|
|
|
|
|
return ajax_response(
|
2016-06-21 08:21:06 -05:00
|
|
|
response=sorted(res, key=label),
|
|
|
|
status=200
|
|
|
|
)
|
2016-03-07 05:48:24 -06:00
|
|
|
|
|
|
|
|
2020-08-10 06:00:07 -05:00
|
|
|
def _group_pref_by_categories(pref, res, label):
|
|
|
|
"""
|
|
|
|
Group preference by categories type.
|
|
|
|
:param pref: preference data.
|
|
|
|
:param res: response for request.
|
|
|
|
:param label: get label.
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
for pref_d in pref:
|
|
|
|
if len(pref_d['categories']):
|
|
|
|
_iterate_categories(pref_d, label, res)
|
|
|
|
|
|
|
|
|
|
|
|
def _iterate_categories(pref_d, label, res):
|
|
|
|
"""
|
|
|
|
Iterate preference categories.
|
|
|
|
:param pref_d: preference data
|
|
|
|
:param label: get label.
|
|
|
|
:param res: response.
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
om = {
|
|
|
|
"id": pref_d['id'],
|
|
|
|
"label": gettext(pref_d['label']),
|
|
|
|
"inode": True,
|
|
|
|
"open": True,
|
2022-03-21 02:59:26 -05:00
|
|
|
"children": [],
|
|
|
|
"value": gettext(pref_d['label']),
|
2020-08-10 06:00:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
for c in pref_d['categories']:
|
|
|
|
for p in c['preferences']:
|
|
|
|
if 'label' in p and p['label'] is not None:
|
|
|
|
p['label'] = gettext(p['label'])
|
|
|
|
if 'help_str' in p and p['help_str'] is not None:
|
|
|
|
p['help_str'] = gettext(p['help_str'])
|
|
|
|
oc = {
|
|
|
|
"id": c['id'],
|
|
|
|
"mid": pref_d['id'],
|
2023-03-24 05:14:43 -05:00
|
|
|
"name": c['name'],
|
2020-08-10 06:00:07 -05:00
|
|
|
"label": gettext(c['label']),
|
2022-03-21 02:59:26 -05:00
|
|
|
"value": '{0}{1}'.format(c['id'], gettext(c['label'])),
|
2020-08-10 06:00:07 -05:00
|
|
|
"inode": False,
|
|
|
|
"open": False,
|
2022-03-21 02:59:26 -05:00
|
|
|
"preferences": sorted(c['preferences'], key=label),
|
|
|
|
"showCheckbox": False
|
2020-08-10 06:00:07 -05:00
|
|
|
}
|
|
|
|
|
2022-03-21 02:59:26 -05:00
|
|
|
(om['children']).append(oc)
|
|
|
|
om['children'] = sorted(om['children'], key=label)
|
2020-08-10 06:00:07 -05:00
|
|
|
|
|
|
|
res.append(om)
|
|
|
|
|
|
|
|
|
2017-06-21 11:09:59 -05:00
|
|
|
@blueprint.route("/get_all", methods=["GET"], endpoint='get_all')
|
|
|
|
@login_required
|
|
|
|
def preferences_s():
|
|
|
|
"""Fetch all preferences for caching."""
|
|
|
|
# Load Preferences
|
|
|
|
pref = Preferences.preferences()
|
|
|
|
res = []
|
|
|
|
|
|
|
|
for m in pref:
|
|
|
|
if len(m['categories']):
|
|
|
|
for c in m['categories']:
|
|
|
|
for p in c['preferences']:
|
2017-06-22 06:18:56 -05:00
|
|
|
p['module'] = m['name']
|
2017-06-21 11:09:59 -05:00
|
|
|
res.append(p)
|
|
|
|
|
|
|
|
return ajax_response(
|
|
|
|
response=res,
|
|
|
|
status=200
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-12-21 00:37:26 -06:00
|
|
|
@blueprint.route("/get_all_cli", methods=["GET"], endpoint='get_all_cli')
|
|
|
|
def get_all_cli():
|
|
|
|
"""Fetch all preferences for caching."""
|
|
|
|
# Load Preferences
|
|
|
|
pref = Preferences.preferences()
|
|
|
|
res = {}
|
|
|
|
|
|
|
|
for m in pref:
|
|
|
|
if len(m['categories']):
|
|
|
|
for c in m['categories']:
|
|
|
|
for p in c['preferences']:
|
|
|
|
p['module'] = m['name']
|
|
|
|
res["{0}:{1}:{2}".format(m['label'], p['label'], c['label']
|
|
|
|
)] = "{0}:{1}:{2}".format(
|
|
|
|
p['module'],c['name'],p['name'])
|
|
|
|
|
|
|
|
return ajax_response(
|
|
|
|
response=res,
|
|
|
|
status=200
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-03-21 02:59:26 -05:00
|
|
|
def get_data():
|
2016-03-07 05:48:24 -06:00
|
|
|
"""
|
2022-03-21 02:59:26 -05:00
|
|
|
Get preferences data.
|
|
|
|
:return: Preferences list
|
|
|
|
:rtype: list
|
2016-03-07 05:48:24 -06:00
|
|
|
"""
|
2022-03-21 02:59:26 -05:00
|
|
|
pref_data = request.form if request.form else json.loads(
|
|
|
|
request.data.decode())
|
2016-03-07 05:48:24 -06:00
|
|
|
|
2022-03-21 02:59:26 -05:00
|
|
|
if not pref_data:
|
|
|
|
raise ValueError("Please provide the valid preferences data to save.")
|
2020-10-27 00:33:00 -05:00
|
|
|
|
2022-03-21 02:59:26 -05:00
|
|
|
return pref_data
|
2016-03-07 05:48:24 -06:00
|
|
|
|
|
|
|
|
2022-03-21 02:59:26 -05:00
|
|
|
@blueprint.route("/", methods=["PUT"], endpoint="update")
|
|
|
|
@login_required
|
|
|
|
def save():
|
|
|
|
"""
|
|
|
|
Save a specific preference.
|
|
|
|
"""
|
|
|
|
pref_data = get_data()
|
|
|
|
|
|
|
|
for data in pref_data:
|
|
|
|
if data['name'] in ['vw_edt_tab_title_placeholder',
|
|
|
|
'qt_tab_title_placeholder',
|
|
|
|
'debugger_tab_title_placeholder'] \
|
|
|
|
and data['value'].isspace():
|
|
|
|
data['value'] = ''
|
|
|
|
|
|
|
|
res, msg = Preferences.save(
|
|
|
|
data['mid'], data['category_id'], data['id'], data['value'])
|
|
|
|
sgm.get_nodes(sgm)
|
|
|
|
|
|
|
|
if not res:
|
|
|
|
return internal_server_error(errormsg=msg)
|
|
|
|
|
|
|
|
response = success_return()
|
|
|
|
|
|
|
|
# Set cookie & session for language settings.
|
|
|
|
# This will execute every time as could not find the better way to know
|
|
|
|
# that which preference is getting updated.
|
|
|
|
|
|
|
|
misc_preference = Preferences.module('misc')
|
|
|
|
user_languages = misc_preference.preference(
|
|
|
|
'user_language'
|
|
|
|
)
|
|
|
|
|
|
|
|
language = 'en'
|
|
|
|
if user_languages:
|
|
|
|
language = user_languages.get() or language
|
|
|
|
|
|
|
|
domain = dict()
|
|
|
|
if config.COOKIE_DEFAULT_DOMAIN and \
|
|
|
|
config.COOKIE_DEFAULT_DOMAIN != 'localhost':
|
|
|
|
domain['domain'] = config.COOKIE_DEFAULT_DOMAIN
|
|
|
|
|
|
|
|
setattr(session, 'PGADMIN_LANGUAGE', language)
|
|
|
|
response.set_cookie("PGADMIN_LANGUAGE", value=language,
|
|
|
|
path=config.COOKIE_DEFAULT_PATH,
|
|
|
|
secure=config.SESSION_COOKIE_SECURE,
|
|
|
|
httponly=config.SESSION_COOKIE_HTTPONLY,
|
|
|
|
samesite=config.SESSION_COOKIE_SAMESITE,
|
|
|
|
**domain)
|
2017-03-24 09:20:10 -05:00
|
|
|
|
|
|
|
return response
|
2023-12-19 04:22:57 -06:00
|
|
|
|
|
|
|
|
2023-12-21 00:37:26 -06:00
|
|
|
def save_pref(data):
|
|
|
|
"""
|
|
|
|
Save a specific preference.
|
|
|
|
"""
|
|
|
|
|
|
|
|
if data['name'] in ['vw_edt_tab_title_placeholder',
|
|
|
|
'qt_tab_title_placeholder',
|
|
|
|
'debugger_tab_title_placeholder'] \
|
|
|
|
and data['value'].isspace():
|
|
|
|
data['value'] = ''
|
|
|
|
|
|
|
|
res, msg = Preferences.save_cli(
|
|
|
|
data['mid'], data['category_id'], data['id'], data['user_id'],
|
|
|
|
data['value'])
|
|
|
|
|
|
|
|
if not res:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2023-12-19 04:22:57 -06:00
|
|
|
@blueprint.route("/update", methods=["PUT"], endpoint="update_pref")
|
|
|
|
@login_required
|
|
|
|
def update():
|
|
|
|
"""
|
|
|
|
Update a specific preference.
|
|
|
|
"""
|
|
|
|
pref_data = get_data()
|
|
|
|
pref_data = json.loads(pref_data['pref_data'])
|
|
|
|
|
|
|
|
for data in pref_data:
|
|
|
|
if data['name'] in ['vw_edt_tab_title_placeholder',
|
|
|
|
'qt_tab_title_placeholder',
|
|
|
|
'debugger_tab_title_placeholder'] \
|
|
|
|
and data['value'].isspace():
|
|
|
|
data['value'] = ''
|
|
|
|
|
|
|
|
pref_module = Preferences.module(data['module'])
|
|
|
|
pref = pref_module.preference(data['name'])
|
|
|
|
# set user preferences
|
|
|
|
pref.set(data['value'])
|
|
|
|
|
|
|
|
return make_json_response(
|
|
|
|
data={'data': 'Success'},
|
|
|
|
status=200
|
|
|
|
)
|