From ce1b3b483d5af845e3d5a78601788f15bcabfb63 Mon Sep 17 00:00:00 2001 From: Sujan Adhikari <109404840+Sujanadh@users.noreply.github.com> Date: Tue, 13 Aug 2024 14:01:27 +0545 Subject: [PATCH] refactor(backend): correct username of contributors (#1751) * refactor: correct username of contributors using sql in single db transaction * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> --- src/backend/app/projects/project_crud.py | 34 +++++++++------------- src/backend/app/projects/project_routes.py | 6 ++-- 2 files changed, 16 insertions(+), 24 deletions(-) diff --git a/src/backend/app/projects/project_crud.py b/src/backend/app/projects/project_crud.py index 58d7357d03..c9c0f7eff1 100644 --- a/src/backend/app/projects/project_crud.py +++ b/src/backend/app/projects/project_crud.py @@ -1525,37 +1525,31 @@ async def get_dashboard_detail( return project -async def get_project_users(db: Session, project_id: int, db_user: db_models.DbUser): +async def get_project_users(db: Session, project_id: int): """Get the users and their contributions for a project. Args: db (Session): The database session. project_id (int): The ID of the project. - db_user (DbUser): User that called the endpoint. Returns: List[Dict[str, Union[str, int]]]: A list of dictionaries containing the username and the number of contributions made by each user for the specified project. """ - # TODO refactor this - # TODO it could probably just be a single raw SQL statement - contributors = ( - db.query(db_models.DbTaskHistory) - .filter(db_models.DbTaskHistory.project_id == project_id) - .all() - ) - unique_user_ids = { - user.user_id for user in contributors if user.user_id is not None - } - response = [] - - for user_id in unique_user_ids: - contributions = count_user_contributions(db, user_id, project_id) - response.append({"user": db_user.username, "contributions": contributions}) - - response = sorted(response, key=lambda x: x["contributions"], reverse=True) - return response + query = text(""" + SELECT u.username, COUNT(th.user_id) as contributions + FROM users u + JOIN task_history th ON u.id = th.user_id + WHERE th.project_id = :project_id + GROUP BY u.username + ORDER BY contributions DESC + """) + result = db.execute(query, {"project_id": project_id}).fetchall() + + return [ + {"user": row.username, "contributions": row.contributions} for row in result + ] def count_user_contributions(db: Session, user_id: int, project_id: int) -> int: diff --git a/src/backend/app/projects/project_routes.py b/src/backend/app/projects/project_routes.py index f43464d164..58f3ade518 100644 --- a/src/backend/app/projects/project_routes.py +++ b/src/backend/app/projects/project_routes.py @@ -1187,7 +1187,6 @@ async def project_dashboard( @router.get("/contributors/{project_id}") async def get_contributors( - project_id: int, db: Session = Depends(database.get_db), project_user: ProjectUserDict = Depends(mapper), ): @@ -1195,9 +1194,8 @@ async def get_contributors( TODO use a pydantic model for return type """ - db_user = project_user.get("user") - project_users = await project_crud.get_project_users(db, project_id, db_user) - return project_users + project = project_user.get("project") + return await project_crud.get_project_users(db, project.id) @router.post("/add-manager/")