2021-11-26 03:12:53 +00:00
|
|
|
import json
|
2022-02-07 11:01:13 +00:00
|
|
|
import threading
|
2021-11-26 03:12:53 +00:00
|
|
|
|
2022-03-11 04:56:22 +00:00
|
|
|
from redis import Redis
|
2021-01-26 09:54:12 +00:00
|
|
|
from django.conf import settings
|
|
|
|
|
2022-03-11 04:56:22 +00:00
|
|
|
from jumpserver.const import CONFIG
|
|
|
|
from common.http import is_true
|
2021-11-26 03:12:53 +00:00
|
|
|
from common.db.utils import safe_db_connection
|
|
|
|
from common.utils import get_logger
|
|
|
|
|
|
|
|
logger = get_logger(__name__)
|
|
|
|
|
2021-01-26 09:54:12 +00:00
|
|
|
|
2022-03-11 04:56:22 +00:00
|
|
|
def get_redis_client(db=0):
|
|
|
|
params = {
|
|
|
|
'host': CONFIG.REDIS_HOST,
|
|
|
|
'port': CONFIG.REDIS_PORT,
|
|
|
|
'password': CONFIG.REDIS_PASSWORD,
|
|
|
|
'db': db,
|
|
|
|
"ssl": is_true(CONFIG.REDIS_USE_SSL),
|
2022-03-14 03:35:14 +00:00
|
|
|
'ssl_cert_reqs': CONFIG.REDIS_SSL_REQUIRED,
|
2022-03-11 04:56:22 +00:00
|
|
|
'ssl_keyfile': getattr(settings, 'REDIS_SSL_KEYFILE'),
|
|
|
|
'ssl_certfile': getattr(settings, 'REDIS_SSL_CERTFILE'),
|
|
|
|
'ssl_ca_certs': getattr(settings, 'REDIS_SSL_CA_CERTS'),
|
|
|
|
}
|
|
|
|
return Redis(**params)
|
2021-01-26 09:54:12 +00:00
|
|
|
|
|
|
|
|
2022-02-07 11:01:13 +00:00
|
|
|
class Subscription:
|
|
|
|
def __init__(self, ch, sub, ):
|
2021-01-26 09:54:12 +00:00
|
|
|
self.ch = ch
|
2022-02-07 11:01:13 +00:00
|
|
|
self.sub = sub
|
2021-01-26 09:54:12 +00:00
|
|
|
|
2022-02-07 11:01:13 +00:00
|
|
|
def _handle_msg(self, _next, error, complete):
|
2021-11-26 03:12:53 +00:00
|
|
|
"""
|
|
|
|
handle arg is the pub published
|
|
|
|
|
2022-02-07 11:01:13 +00:00
|
|
|
:param _next: next msg handler
|
|
|
|
:param error: error msg handler
|
|
|
|
:param complete: complete msg handler
|
2021-11-26 03:12:53 +00:00
|
|
|
:return:
|
|
|
|
"""
|
2022-02-07 11:01:13 +00:00
|
|
|
msgs = self.sub.listen()
|
|
|
|
|
|
|
|
if error is None:
|
|
|
|
error = lambda m, i: None
|
|
|
|
|
|
|
|
if complete is None:
|
|
|
|
complete = lambda: None
|
2021-11-26 03:12:53 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
for msg in msgs:
|
|
|
|
if msg["type"] != "message":
|
|
|
|
continue
|
2022-02-07 11:01:13 +00:00
|
|
|
item = None
|
2021-11-26 03:12:53 +00:00
|
|
|
try:
|
|
|
|
item_json = msg['data'].decode()
|
|
|
|
item = json.loads(item_json)
|
|
|
|
|
|
|
|
with safe_db_connection():
|
2022-02-07 11:01:13 +00:00
|
|
|
_next(item)
|
2021-11-26 03:12:53 +00:00
|
|
|
except Exception as e:
|
2022-02-07 11:01:13 +00:00
|
|
|
error(msg, item)
|
2022-03-08 10:47:39 +00:00
|
|
|
logger.error('Subscribe handler handle msg error: {}'.format(e))
|
2021-11-26 03:12:53 +00:00
|
|
|
except Exception as e:
|
2022-03-08 10:47:39 +00:00
|
|
|
logger.error('Consume msg error: {}'.format(e))
|
2021-11-26 03:12:53 +00:00
|
|
|
|
|
|
|
try:
|
2022-02-07 11:01:13 +00:00
|
|
|
complete()
|
2021-11-26 03:12:53 +00:00
|
|
|
except Exception as e:
|
2022-02-07 11:01:13 +00:00
|
|
|
logger.error('Complete subscribe error: {}'.format(e))
|
|
|
|
pass
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.unsubscribe()
|
|
|
|
except Exception as e:
|
|
|
|
logger.error("Redis observer close error: {}".format(e))
|
|
|
|
|
|
|
|
def keep_handle_msg(self, _next, error, complete):
|
|
|
|
t = threading.Thread(target=self._handle_msg, args=(_next, error, complete))
|
|
|
|
t.daemon = True
|
|
|
|
t.start()
|
|
|
|
return t
|
|
|
|
|
|
|
|
def unsubscribe(self):
|
|
|
|
try:
|
|
|
|
self.sub.close()
|
|
|
|
except Exception as e:
|
|
|
|
logger.error('Unsubscribe msg error: {}'.format(e))
|
|
|
|
|
|
|
|
|
|
|
|
class RedisPubSub:
|
|
|
|
def __init__(self, ch, db=10):
|
|
|
|
self.ch = ch
|
|
|
|
self.redis = get_redis_client(db)
|
|
|
|
|
|
|
|
def subscribe(self, _next, error=None, complete=None):
|
|
|
|
ps = self.redis.pubsub()
|
2022-02-08 02:44:42 +00:00
|
|
|
ps.subscribe(self.ch)
|
2022-02-07 11:01:13 +00:00
|
|
|
sub = Subscription(self.ch, ps)
|
|
|
|
sub.keep_handle_msg(_next, error, complete)
|
|
|
|
return sub
|
|
|
|
|
|
|
|
def publish(self, data):
|
|
|
|
data_json = json.dumps(data)
|
|
|
|
self.redis.publish(self.ch, data_json)
|
|
|
|
return True
|