From c4cda52e2741e853f4392384cb5f4571f70797a8 Mon Sep 17 00:00:00 2001 From: violeta <50527601+violetaperezandrade@users.noreply.github.com> Date: Sat, 20 Apr 2024 14:42:18 -0300 Subject: [PATCH] Merge HAN-102: Rollback de SQL (#11) --- app/exceptions/UserException.py | 3 +- app/main.py | 12 ++--- app/repository/Users.py | 7 +++ app/repository/sql_exception_handling.py | 59 ++++++++++++++++++++++++ app/service/Users.py | 13 +++++- 5 files changed, 84 insertions(+), 10 deletions(-) create mode 100644 app/repository/sql_exception_handling.py diff --git a/app/exceptions/UserException.py b/app/exceptions/UserException.py index 022828e..4da9858 100644 --- a/app/exceptions/UserException.py +++ b/app/exceptions/UserException.py @@ -18,7 +18,6 @@ def __init__(self): class InvalidURL(HTTPException): - def __init__(self, id: int): + def __init__(self, detail: str): status_code = status.HTTP_400_BAD_REQUEST - detail = "Invalid URL" super().__init__(status_code=status_code, detail=detail) diff --git a/app/main.py b/app/main.py index 763f6bb..203173f 100644 --- a/app/main.py +++ b/app/main.py @@ -13,22 +13,22 @@ @app.get("/") -async def root(): - return {"message": "Hello World"} +def root(): + return {"message": "users service"} @app.get("/users/{user_id}") -async def get_users(user_id: int): +def get_users(user_id: int): return users_controller.handle_get_user(user_id) @app.get("/users") -async def get_all_users(): +def get_all_users(): return users_controller.handle_get_all_users() @app.post("/users") -async def create_user(user_data: CreateUserSchema): +def create_user(user_data: CreateUserSchema): return users_controller.handle_create_user(user_data.dict()) @@ -38,5 +38,5 @@ def login_with_google(request: LoginRequest): @app.patch("/users/{user_id}") -async def update_user(user_id: int, update_data: UpdateUserSchema): +def update_user(user_id: int, update_data: UpdateUserSchema): return users_controller.handle_update_user(user_id, update_data.dict()) diff --git a/app/repository/Users.py b/app/repository/Users.py index 54cfdaa..38d13b7 100644 --- a/app/repository/Users.py +++ b/app/repository/Users.py @@ -5,6 +5,7 @@ from models.database import Base from models.users import User from datetime import date +from .sql_exception_handling import withSQLExceptionsHandle class UsersRepository: @@ -27,22 +28,27 @@ def shutdown(self): def rollback(self): self.session.rollback() + @withSQLExceptionsHandle() def add(self, record: Base): self.session.add(record) self.session.commit() + @withSQLExceptionsHandle() def get_user(self, user_id: int): user = self.session.query(User).filter_by(id=user_id).first() return user.__dict__ if user else None + @withSQLExceptionsHandle() def get_user_by_email(self, email: str): user = self.session.query(User).filter_by(email=email).first() return user.__dict__ if user else None + @withSQLExceptionsHandle() def get_all_users(self): users = self.session.query(User).all() return self.__parse_result(users) + @withSQLExceptionsHandle() def create_user( self, email: str, @@ -76,6 +82,7 @@ def create_user( self.session.commit() return new_user + @withSQLExceptionsHandle() def edit_user(self, user_id: int, data_to_edit: dict): user = self.session.query(User).filter_by(id=user_id).first() for field, value in data_to_edit.items(): diff --git a/app/repository/sql_exception_handling.py b/app/repository/sql_exception_handling.py new file mode 100644 index 0000000..fa81f24 --- /dev/null +++ b/app/repository/sql_exception_handling.py @@ -0,0 +1,59 @@ +from psycopg2.errors import UniqueViolation +from sqlalchemy.exc import PendingRollbackError, IntegrityError, NoResultFound +from fastapi import status, HTTPException +import logging + +logger = logging.getLogger("app") +logger.setLevel("DEBUG") + + +def handle_common_errors(err): + if isinstance(err, IntegrityError): + if isinstance(err.orig, UniqueViolation): + parsed_error = err.orig.pgerror.split("\n") + raise HTTPException( + status_code=status.HTTP_400_BAD_REQUEST, + detail={ + "error": parsed_error[0], + "detail": parsed_error[1] + }) + + raise HTTPException( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, + detail=format(err)) + + if isinstance(err, PendingRollbackError): + logger.warning(format(err)) + raise HTTPException( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, + detail=format(err) + ) + + if isinstance(err, NoResultFound): + raise HTTPException( + status_code=status.HTTP_404_NOT_FOUND, detail=format(err) + ) + + logger.error(format(err)) + raise err + + +def withSQLExceptionsHandle(async_mode: bool = False): + def decorator(func): + async def handleAsyncSQLException(*args, **kwargs): + try: + return await func(*args, **kwargs) + except Exception as err: + return handle_common_errors(err) + + def handleSyncSQLException(*args, **kwargs): + try: + return func(*args, **kwargs) + except Exception as err: + return handle_common_errors(err) + + return ( + handleAsyncSQLException if async_mode else handleSyncSQLException + ) + + return decorator diff --git a/app/service/Users.py b/app/service/Users.py index a78442b..b47debd 100644 --- a/app/service/Users.py +++ b/app/service/Users.py @@ -24,7 +24,12 @@ def get_all_users(self): def create_user(self, user_data: dict): if not self._validate_location(user_data.get("location")): raise InvalidData() - return self.user_repository.create_user(**user_data) + try: + user = self.user_repository.create_user(**user_data) + return user + except Exception as e: + self.user_repository.rollback() + raise e def update_user(self, user_id: int, update_data: dict): # TODO: aca habria que chequear a partir del token, session o algo que @@ -38,7 +43,11 @@ def update_user(self, user_id: int, update_data: dict): if not re.match(r'^https?://(?:[a-zA-Z0-9-]+\.)+[a-zA-Z]{2,6}' r'(?:/[^/#?]+)+(?:\?.*)?$', photo_url): raise InvalidURL("Invalid photo URL") - self.user_repository.edit_user(user_id, filtered_update_data) + try: + self.user_repository.edit_user(user_id, filtered_update_data) + except Exception as e: + self.user_repository.rollback() + raise e def login(self, auth_code: str): access_token = self._get_access_token(auth_code)