from contextlib import contextmanager from django.db import connections, transaction, connection from django.utils.encoding import force_str from common.utils import get_logger, signer, crypto logger = get_logger(__file__) def get_object_if_need(model, pk): if not isinstance(pk, model): try: return model.objects.get(id=pk) except model.DoesNotExist as e: logger.error(f"DoesNotExist: <{model.__name__}:{pk}> not exist") raise e return pk def get_objects_if_need(model, pks): if not pks: return pks if not isinstance(pks[0], model): objs = list(model.objects.filter(id__in=pks)) if len(objs) != len(pks): pks = set(pks) exists_pks = {o.id for o in objs} not_found_pks = ",".join(pks - exists_pks) logger.error(f"DoesNotExist: <{model.__name__}: {not_found_pks}>") return objs return pks def get_objects(model, pks): if not pks: return pks objs = list(model.objects.filter(id__in=pks)) if len(objs) != len(pks): pks = set(pks) exists_pks = {o.id for o in objs} not_found_pks = pks - exists_pks logger.error(f"DoesNotExist: <{model.__name__}: {not_found_pks}>") return objs # 复制 django.db.close_old_connections, 因为它没有导出,ide 提示有问题 def close_old_connections(): for conn in connections.all(): conn.close_if_unusable_or_obsolete() @contextmanager def safe_db_connection(): in_atomic_block = connection.in_atomic_block # 当前是否处于事务中 autocommit = transaction.get_autocommit() # 是否启用了自动提交 created = False try: if not connection.is_usable(): connection.close() connection.connect() created = True yield finally: # 如果不是事务中(API 请求中可能需要提交事务),则关闭连接 if created and not in_atomic_block and autocommit: print("close connection in safe_db_connection") close_old_connections() @contextmanager def open_db_connection(alias="default"): connection = transaction.get_connection(alias) try: connection.connect() with transaction.atomic(): yield connection finally: connection.close() class Encryptor: def __init__(self, value): self.value = force_str(value) def decrypt(self): plain_value = crypto.decrypt(self.value) # 如果没有解开,使用原来的signer解密 if not plain_value: plain_value = signer.unsign(self.value) or "" return plain_value def encrypt(self): return crypto.encrypt(self.value)