user_service.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. """
  2. User management service.
  3. """
  4. from sqlalchemy import func, select
  5. from sqlalchemy.ext.asyncio import AsyncSession
  6. from app.core.security import hash_password
  7. from app.models.user import User
  8. from app.schemas.user import UserCreate, UserUpdate
  9. async def create_user(
  10. db: AsyncSession,
  11. data: UserCreate,
  12. ) -> User:
  13. """
  14. Create a new user.
  15. Args:
  16. db: Database session
  17. data: User creation data
  18. Returns:
  19. Created user
  20. """
  21. # Hash password
  22. hashed_password = hash_password(data.password)
  23. user = User(
  24. email=data.email,
  25. hashed_password=hashed_password,
  26. full_name=data.full_name,
  27. phone=data.phone,
  28. role=data.role,
  29. organization_id=data.organization_id,
  30. status="active", # Superadmin creates active users
  31. email_verified=True, # Auto-verify for superadmin-created users
  32. )
  33. db.add(user)
  34. await db.commit()
  35. await db.refresh(user)
  36. return user
  37. async def get_user(db: AsyncSession, user_id: int) -> User | None:
  38. """
  39. Get user by ID.
  40. Args:
  41. db: Database session
  42. user_id: User ID
  43. Returns:
  44. User or None
  45. """
  46. result = await db.execute(select(User).where(User.id == user_id))
  47. return result.scalar_one_or_none()
  48. async def get_user_by_email(db: AsyncSession, email: str) -> User | None:
  49. """
  50. Get user by email.
  51. Args:
  52. db: Database session
  53. email: User email
  54. Returns:
  55. User or None
  56. """
  57. result = await db.execute(select(User).where(User.email == email))
  58. return result.scalar_one_or_none()
  59. async def list_users(
  60. db: AsyncSession,
  61. skip: int = 0,
  62. limit: int = 100,
  63. organization_id: int | None = None,
  64. role: str | None = None,
  65. status: str | None = None,
  66. ) -> tuple[list[User], int]:
  67. """
  68. List users with pagination and filters.
  69. Args:
  70. db: Database session
  71. skip: Number of records to skip
  72. limit: Maximum number of records to return
  73. organization_id: Filter by organization (optional)
  74. role: Filter by role (optional)
  75. status: Filter by status (optional)
  76. Returns:
  77. Tuple of (users list, total count)
  78. """
  79. # Build query
  80. query = select(User)
  81. if organization_id is not None:
  82. query = query.where(User.organization_id == organization_id)
  83. if role:
  84. query = query.where(User.role == role)
  85. if status:
  86. query = query.where(User.status == status)
  87. # Get total count
  88. count_query = select(func.count()).select_from(User)
  89. if organization_id is not None:
  90. count_query = count_query.where(User.organization_id == organization_id)
  91. if role:
  92. count_query = count_query.where(User.role == role)
  93. if status:
  94. count_query = count_query.where(User.status == status)
  95. total_result = await db.execute(count_query)
  96. total = total_result.scalar_one()
  97. # Get paginated results
  98. query = query.offset(skip).limit(limit).order_by(User.created_at.desc())
  99. result = await db.execute(query)
  100. users = list(result.scalars().all())
  101. return users, total
  102. async def update_user(
  103. db: AsyncSession,
  104. user_id: int,
  105. data: UserUpdate,
  106. ) -> User | None:
  107. """
  108. Update user.
  109. Args:
  110. db: Database session
  111. user_id: User ID
  112. data: Update data
  113. Returns:
  114. Updated user or None if not found
  115. """
  116. result = await db.execute(select(User).where(User.id == user_id))
  117. user = result.scalar_one_or_none()
  118. if not user:
  119. return None
  120. # Update fields
  121. update_data = data.model_dump(exclude_unset=True)
  122. for field, value in update_data.items():
  123. setattr(user, field, value)
  124. await db.commit()
  125. await db.refresh(user)
  126. return user
  127. async def delete_user(
  128. db: AsyncSession,
  129. user_id: int,
  130. ) -> bool:
  131. """
  132. Delete user.
  133. Args:
  134. db: Database session
  135. user_id: User ID
  136. Returns:
  137. True if deleted, False if not found
  138. """
  139. result = await db.execute(select(User).where(User.id == user_id))
  140. user = result.scalar_one_or_none()
  141. if not user:
  142. return False
  143. await db.delete(user)
  144. await db.commit()
  145. return True
  146. async def change_user_password(
  147. db: AsyncSession,
  148. user_id: int,
  149. new_password: str,
  150. ) -> User | None:
  151. """
  152. Change user password.
  153. Args:
  154. db: Database session
  155. user_id: User ID
  156. new_password: New password (plain text)
  157. Returns:
  158. Updated user or None if not found
  159. """
  160. result = await db.execute(select(User).where(User.id == user_id))
  161. user = result.scalar_one_or_none()
  162. if not user:
  163. return None
  164. user.hashed_password = hash_password(new_password)
  165. await db.commit()
  166. await db.refresh(user)
  167. return user