2023-01-19 04:27:02 -06:00
|
|
|
##########################################################################
|
|
|
|
#
|
|
|
|
# pgAdmin 4 - PostgreSQL Tools
|
|
|
|
#
|
|
|
|
# Copyright (C) 2013 - 2023, The pgAdmin Development Team
|
|
|
|
# This software is released under the PostgreSQL Licence
|
|
|
|
#
|
|
|
|
#########################################################################
|
|
|
|
|
|
|
|
"""Server heartbeat manager."""
|
|
|
|
|
|
|
|
|
|
|
|
import threading
|
|
|
|
import datetime
|
|
|
|
import config
|
|
|
|
from flask import session, current_app
|
2023-01-30 04:09:34 -06:00
|
|
|
from flask_babel import gettext
|
2023-01-19 04:27:02 -06:00
|
|
|
|
|
|
|
|
|
|
|
def log_server_heartbeat(data):
|
|
|
|
"""Log Server Heartbeat."""
|
|
|
|
from config import PG_DEFAULT_DRIVER
|
|
|
|
from pgadmin.utils.driver import get_driver
|
|
|
|
manager = get_driver(PG_DEFAULT_DRIVER).connection_manager(int(data['sid'])
|
|
|
|
)
|
|
|
|
|
|
|
|
_server_heartbeat = getattr(current_app, '_pgadmin_server_heartbeat', {})
|
|
|
|
|
|
|
|
if session.sid not in _server_heartbeat:
|
|
|
|
_server_heartbeat[session.sid] = {}
|
|
|
|
|
2023-01-30 04:09:34 -06:00
|
|
|
if not manager:
|
|
|
|
stop_server_heartbeat(data)
|
|
|
|
return False, gettext("Manager not found. Stopped Heartbeat logging.")
|
|
|
|
else:
|
|
|
|
_server_heartbeat[session.sid][data['sid']] = {
|
|
|
|
'timestamp': datetime.datetime.now(),
|
|
|
|
'conn': manager.connections
|
|
|
|
}
|
|
|
|
current_app.logger.debug(
|
|
|
|
"Heartbeat logged for the session id##server id: {0}##{1}".format(
|
|
|
|
session.sid, data['sid']))
|
|
|
|
|
|
|
|
setattr(current_app, '_pgadmin_server_heartbeat', _server_heartbeat)
|
|
|
|
return True, gettext("Heartbeat logged successfully.")
|
|
|
|
|
|
|
|
|
|
|
|
def stop_server_heartbeat(data):
|
|
|
|
"""Stop logging server heartbeat."""
|
|
|
|
_server_heartbeat = getattr(current_app, '_pgadmin_server_heartbeat', {})
|
|
|
|
|
|
|
|
if session.sid in _server_heartbeat and \
|
|
|
|
data['sid'] in _server_heartbeat[session.sid]:
|
|
|
|
_server_heartbeat[session.sid].pop(data['sid'])
|
|
|
|
|
|
|
|
current_app.logger.debug(
|
|
|
|
"Heartbeat logging stopped for the session"
|
|
|
|
" id##server id: {0}##{1}".format(session.sid, data['sid']))
|
2023-01-19 04:27:02 -06:00
|
|
|
|
2023-01-30 04:09:34 -06:00
|
|
|
setattr(current_app, '_pgadmin_server_heartbeat', _server_heartbeat)
|
2023-01-19 04:27:02 -06:00
|
|
|
|
2023-01-30 04:09:34 -06:00
|
|
|
return True, gettext("Stopped Heartbeat logging.")
|
2023-01-19 04:27:02 -06:00
|
|
|
|
|
|
|
|
|
|
|
def get_server_heartbeat(server_id):
|
|
|
|
_server_heartbeat = getattr(current_app, '_pgadmin_server_heartbeat', {})
|
|
|
|
|
|
|
|
if session.sid in _server_heartbeat and server_id in _server_heartbeat[
|
|
|
|
session.sid]:
|
|
|
|
return _server_heartbeat[session.sid][server_id]
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
class ServerHeartbeatTimer():
|
|
|
|
def __init__(self, sec, _app):
|
|
|
|
def func_wrapper():
|
|
|
|
self.t = threading.Timer(sec, func_wrapper)
|
|
|
|
self.t.start()
|
|
|
|
self.release_server_heartbeat()
|
|
|
|
self.t = threading.Timer(sec, func_wrapper)
|
2023-01-24 05:52:29 -06:00
|
|
|
self.t.daemon = True
|
2023-01-19 04:27:02 -06:00
|
|
|
self.t.start()
|
|
|
|
self._app = _app
|
|
|
|
|
|
|
|
def release_server_heartbeat(self):
|
|
|
|
with self._app.app_context():
|
|
|
|
_server_heartbeat = getattr(self._app,
|
|
|
|
'_pgadmin_server_heartbeat', {})
|
|
|
|
if len(_server_heartbeat) > 0:
|
|
|
|
for sess_id in list(_server_heartbeat):
|
|
|
|
for sid in list(_server_heartbeat[sess_id]):
|
|
|
|
last_heartbeat_time = _server_heartbeat[sess_id][sid][
|
|
|
|
'timestamp']
|
|
|
|
current_time = datetime.datetime.now()
|
|
|
|
diff = current_time - last_heartbeat_time
|
|
|
|
|
|
|
|
# Wait for 4 times then the timeout
|
|
|
|
if diff.total_seconds() > (
|
|
|
|
config.SERVER_HEARTBEAT_TIMEOUT * 4):
|
|
|
|
self._release_connections(
|
|
|
|
_server_heartbeat[sess_id][sid]['conn'],
|
|
|
|
sess_id, sid)
|
|
|
|
_server_heartbeat[sess_id].pop(sid)
|
|
|
|
if len(_server_heartbeat[sess_id]) == 0:
|
|
|
|
_server_heartbeat.pop(sess_id)
|
|
|
|
setattr(self._app, '_pgadmin_server_heartbeat',
|
|
|
|
_server_heartbeat)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _release_connections(server_conn, sess_id, sid):
|
|
|
|
for d in server_conn:
|
2023-01-24 05:52:29 -06:00
|
|
|
try:
|
|
|
|
# Release the connection
|
|
|
|
server_conn[d]._release()
|
|
|
|
# Reconnect on the reload
|
|
|
|
server_conn[d].wasConnected = True
|
|
|
|
current_app.logger.debug(
|
|
|
|
"Heartbeat not received. Released "
|
|
|
|
"connection for the session "
|
|
|
|
"id##server id: {0}##{1}".format(
|
|
|
|
sess_id, sid))
|
|
|
|
except Exception as e:
|
|
|
|
current_app.logger.exception(e)
|
2023-01-19 04:27:02 -06:00
|
|
|
|
|
|
|
def cancel(self):
|
|
|
|
self.t.cancel()
|
|
|
|
|
|
|
|
|
|
|
|
def init_app(app):
|
|
|
|
setattr(app, '_pgadmin_server_heartbeat', {})
|
|
|
|
ServerHeartbeatTimer(sec=config.SERVER_HEARTBEAT_TIMEOUT,
|
|
|
|
_app=app)
|