sys_user.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547
  1. # -*- coding: utf-8 -*-
  2. # @Author : YY
  3. from datetime import datetime
  4. from typing import List, Literal, Optional
  5. from ruoyi_common.constant import UserConstants
  6. from ruoyi_common.exception import ServiceException
  7. from ruoyi_common.sqlalchemy.transaction import Transactional
  8. from ruoyi_common.domain.entity import SysRole, SysUser
  9. from ruoyi_common.utils import security_util
  10. from ruoyi_common.utils.base import LogUtil, StringUtil
  11. from ruoyi_framework.descriptor.datascope import DataScope
  12. from ruoyi_system.domain.entity import SysPost, SysUserPost, SysUserRole
  13. from ruoyi_system.mapper import SysUserMapper
  14. from ruoyi_system.mapper.sys_post import SysPostMapper
  15. from ruoyi_system.mapper.sys_role import SysRoleMapper
  16. from ruoyi_system.mapper.sys_user_post import SysUserPostMapper
  17. from ruoyi_system.mapper.sys_user_role import SysUserRoleMapper
  18. from ruoyi_admin.ext import db
  19. from ruoyi_system.service.sys_config import SysConfigService
  20. class SysUserService:
  21. @classmethod
  22. @DataScope(dept=True, user=True)
  23. def select_user_list(cls, query: SysUser) -> List[SysUser]:
  24. """
  25. 查询用户列表
  26. Args:
  27. query (SysUser): 包含查询条件的传输对象
  28. Returns:
  29. List[SysUser]: 用户列表
  30. """
  31. return SysUserMapper.select_user_list(query)
  32. @classmethod
  33. @DataScope(dept=True, user=True)
  34. def select_allocated_list(cls, query: SysUser) -> List[SysUser]:
  35. """
  36. 查询已分配用户列表
  37. Args:
  38. query (SysUser): 包含查询条件的传输对象
  39. Returns:
  40. List[SysUser]: 已分配用户列表
  41. """
  42. return SysUserMapper.select_allocated_list(query)
  43. @classmethod
  44. @DataScope(dept=True, user=True)
  45. def select_unallocated_list(cls, query: SysUser) -> List[SysUser]:
  46. """
  47. 查询未分配用户列表
  48. Args:
  49. query (SysUser): 包含查询条件的传输对象
  50. Returns:
  51. List[SysUser]: 已分配用户列表
  52. """
  53. return SysUserMapper.select_unallocated_list(query)
  54. @classmethod
  55. def select_user_by_user_name(cls, user_name: str) -> Optional[SysUser]:
  56. """
  57. 根据用户名,查询用户
  58. Args:
  59. user_name (str): 用户名
  60. Returns:
  61. Optional[SysUser]: 用户信息
  62. """
  63. return SysUserMapper.select_user_by_user_name(user_name)
  64. @classmethod
  65. def select_user_by_id(cls, user_id: int) -> Optional[SysUser]:
  66. """
  67. 根据用户ID,查询用户
  68. Args:
  69. user_id (int): 用户ID
  70. Returns:
  71. Optional[SysUser]: 用户信息
  72. """
  73. return SysUserMapper.select_user_by_id(user_id)
  74. @classmethod
  75. def select_user_role_group(cls, user_name: str) -> str:
  76. """
  77. 查询用户角色组
  78. Args:
  79. user_name (str): 用户名
  80. Returns:
  81. str: 角色组
  82. """
  83. eos: List[SysRole] = SysRoleMapper.select_role_list_by_user_name(user_name)
  84. if not eos:
  85. return StringUtil.EMPTY
  86. return ",".join([eo.role_name for eo in eos])
  87. @classmethod
  88. def select_user_post_group(cls, user_name: str) -> str:
  89. """
  90. 查询用户岗位组
  91. Args:
  92. user_name (str): 用户名
  93. Returns:
  94. str: 岗位组
  95. """
  96. eos: List[SysPost] = SysPostMapper.select_posts_by_user_name(user_name)
  97. if not eos:
  98. return StringUtil.EMPTY
  99. return ",".join([eo.post_name for eo in eos])
  100. @classmethod
  101. def check_user_name_unique(cls, user: SysUser) -> Literal["0", "1"]:
  102. """
  103. 校验用户名是否唯一
  104. Args:
  105. user (SysUser): 用户信息
  106. Returns:
  107. str: 唯一标识符, 0-唯一, 1-不唯一
  108. """
  109. user_name = -1 if user.user_name is None else user.user_name
  110. num = SysUserMapper.check_user_name_unique(user_name)
  111. if num > 0:
  112. return UserConstants.NOT_UNIQUE
  113. return UserConstants.UNIQUE
  114. @classmethod
  115. def check_phone_unique(cls, user: SysUser) -> Literal["0", "1"]:
  116. """
  117. 校验手机号是否唯一
  118. Args:
  119. user (SysUser): 用户信息
  120. Returns:
  121. str: 唯一标识符, 0-唯一, 1-不唯一
  122. """
  123. user_id = -1 if user.user_id is None else user.user_id
  124. eo: SysUser = SysUserMapper.check_phone_unique(user.phonenumber)
  125. if eo and eo.user_id != user_id:
  126. return UserConstants.NOT_UNIQUE
  127. return UserConstants.UNIQUE
  128. @classmethod
  129. def check_email_unique(cls, user: SysUser) -> str:
  130. """
  131. 校验邮箱是否唯一
  132. Args:
  133. user (SysUser): 用户信息
  134. Returns:
  135. str: 唯一标识符, 0-唯一, 1-不唯一
  136. """
  137. user_email = -1 if user.email is None else user.email
  138. eo: SysUser = SysUserMapper.check_email_unique(user.email)
  139. if eo and eo.email != user_email:
  140. return UserConstants.NOT_UNIQUE
  141. return UserConstants.UNIQUE
  142. @classmethod
  143. def check_user_allowed(cls, user: SysUser):
  144. """
  145. 检查用户是否允许操作
  146. Args:
  147. user (SysUser): 用户信息
  148. Raises:
  149. ServiceException: 超级管理员用户不允许操作
  150. """
  151. if user.is_admin():
  152. raise ServiceException("不允许操作超级管理员用户")
  153. @classmethod
  154. def check_user_data_scope(cls, user_id: Optional[int]):
  155. """
  156. 检查用户数据权限
  157. Args:
  158. user_id (Optional[int]): 用户ID
  159. Raises:
  160. ServiceException: 无权限访问用户数据
  161. """
  162. if not security_util.login_user_is_admin():
  163. user = SysUser(user_id=user_id) if user_id else SysUser()
  164. users: List[SysUser] = cls.select_user_list(user)
  165. if not users:
  166. raise ServiceException("没有权限访问用户数据")
  167. @classmethod
  168. @Transactional(db.session)
  169. def insert_user(cls, user: SysUser) -> bool:
  170. """
  171. 新增用户
  172. Args:
  173. user (SysUser): 用户信息
  174. Returns:
  175. bool: 操作结果
  176. """
  177. user.password = cls._build_password(user.password)
  178. if user.create_time is None:
  179. now = datetime.now()
  180. user.create_time = now
  181. user.update_time = now
  182. last_pid = SysUserMapper.insert_user(user)
  183. user.user_id = last_pid
  184. cls.insert_user_post_by_user(user)
  185. cls.insert_user_role_by_user(user)
  186. return last_pid > 0
  187. @classmethod
  188. def register_user(cls, user: SysUser) -> bool:
  189. """
  190. 注册用户
  191. Args:
  192. user (SysUser): 用户信息
  193. Returns:
  194. bool: 操作结果
  195. """
  196. user.password = cls._build_password(user.password)
  197. if user.create_time is None:
  198. user.create_time = datetime.now()
  199. flag = SysUserMapper.insert_user(user)
  200. return flag > 0
  201. @classmethod
  202. @Transactional(db.session)
  203. def update_user(cls, user: SysUser) -> bool:
  204. """
  205. 更新用户
  206. Args:
  207. user (SysUser): 用户信息
  208. Returns:
  209. bool: 操作结果
  210. """
  211. user.update_time = datetime.now()
  212. # 删除用户角色关联
  213. SysRoleMapper.delete_user_role_by_user_id(user.user_id)
  214. # 新增用户和角色的关联
  215. cls.insert_user_role_by_user(user)
  216. # 删除用户岗位关联
  217. SysUserPostMapper.delete_user_post_by_user_id(user.user_id)
  218. # 新增用户岗位关联
  219. cls.insert_user_post_by_user(user)
  220. return SysUserMapper.update_user(user)
  221. @classmethod
  222. def update_user_login_info(cls, user: SysUser) -> bool:
  223. """
  224. 更新用户登录信息(登录IP、时间)
  225. Args:
  226. user (SysUser): 用户信息
  227. Returns:
  228. bool: 操作结果
  229. """
  230. user.update_time = datetime.now()
  231. return SysUserMapper.update_user_login_info(user) > 0
  232. @classmethod
  233. @Transactional(db.session)
  234. def insert_user_auth(cls, user_id: int, role_ids: List[int]):
  235. """
  236. 新增用户角色
  237. Args:
  238. user_id: 用户id
  239. role_ids: 角色id列表
  240. """
  241. cls.insert_user_role(user_id, role_ids)
  242. @classmethod
  243. @Transactional(db.session)
  244. def delete_users_by_id(cls, id: int) -> bool:
  245. """
  246. 根据用户ID,删除用户
  247. Args:
  248. id (int): 用户ID
  249. Returns:
  250. bool: 操作结果
  251. """
  252. SysUserRoleMapper.delete_user_role_by_user_id(id)
  253. SysUserPostMapper.delete_user_post_by_user_id(id)
  254. return SysUserMapper.delete_user_by_id(id) > 0
  255. @classmethod
  256. @Transactional(db.session)
  257. def delete_users_by_ids(cls, ids: List[int]) -> bool:
  258. """
  259. 根据用户ID列表,批量删除用户
  260. Args:
  261. ids (List[int]): 用户ID列表
  262. Returns:
  263. bool: 操作结果
  264. """
  265. if not ids:
  266. return False
  267. SysUserRoleMapper.delete_user_role_by_user_ids(ids)
  268. SysUserPostMapper.delete_user_post(ids)
  269. return SysUserMapper.delete_user_by_ids(ids) > 0
  270. @classmethod
  271. def update_user_status(cls, user: SysUser) -> bool:
  272. """
  273. 更新用户状态
  274. Args:
  275. user (SysUser): 用户信息
  276. Returns:
  277. bool: 操作结果
  278. """
  279. user.update_time = datetime.now()
  280. return SysUserMapper.update_user(user) > 0
  281. @classmethod
  282. def update_user_profile(cls, user: SysUser) -> bool:
  283. """
  284. 更新用户个人信息
  285. Args:
  286. user (SysUser): 用户信息
  287. Returns:
  288. bool: 操作结果
  289. """
  290. user.update_time = datetime.now()
  291. return SysUserMapper.update_user(user) > 0
  292. @classmethod
  293. def update_user_avatar(cls, user_name: str, avatar: str) -> bool:
  294. """
  295. 更新用户头像
  296. Args:
  297. user_name (str): 用户名
  298. avatar (str): 头像
  299. Returns:
  300. bool: 操作结果
  301. """
  302. return SysUserMapper.update_user_avatar(user_name, avatar) > 0
  303. @classmethod
  304. def reset_pwd(cls, user: SysUser) -> bool:
  305. """
  306. 重置用户密码
  307. Args:
  308. user (SysUser): 用户信息
  309. Returns:
  310. bool: 操作结果
  311. """
  312. user.password = cls._build_password(user.password)
  313. user.update_time = datetime.now()
  314. return SysUserMapper.update_user(user) > 0
  315. @classmethod
  316. def reset_user_pwd(cls, username: str, password: str) -> bool:
  317. """
  318. 重置用户密码
  319. Args:
  320. username (str): 用户名
  321. password (str): 密码
  322. Returns:
  323. bool: 操作结果
  324. """
  325. return SysUserMapper.reset_user_pwd(username, password) > 0
  326. @classmethod
  327. def insert_user_role_by_user(cls, user: SysUser):
  328. """
  329. 新增用户角色
  330. Args:
  331. user: 用户信息
  332. """
  333. cls.insert_user_role(user.user_id, user.role_ids)
  334. @classmethod
  335. @Transactional(db.session)
  336. def insert_user_role(cls, user_id: int, role_ids: List[int]):
  337. """
  338. 新增用户角色
  339. Args:
  340. user_id: 用户id
  341. role_ids: 角色id列表
  342. """
  343. if role_ids:
  344. lists = [
  345. SysUserRole(user_id=user_id, role_id=role_id)
  346. for role_id in role_ids
  347. ]
  348. SysUserRoleMapper.batch_user_role(lists)
  349. @classmethod
  350. def insert_user_post_by_user(cls, user: SysUser):
  351. """
  352. 新增用户岗位
  353. Args:
  354. user: 用户信息
  355. """
  356. cls.insert_user_post(user.user_id, user.post_ids)
  357. @classmethod
  358. @Transactional(db.session)
  359. def insert_user_post(cls, user_id: int, post_ids: List[int]):
  360. """
  361. 新增用户岗位
  362. Args:
  363. user_id: 用户id
  364. post_ids: 岗位id列表
  365. """
  366. if post_ids:
  367. lists = [
  368. SysUserPost(user_id=user_id, post_id=post_id)
  369. for post_id in post_ids
  370. ]
  371. SysUserPostMapper.batch_user_post(lists)
  372. @classmethod
  373. @Transactional(db.session)
  374. def update_user_roles(cls, user_id: int, role_ids: List[int]) -> bool:
  375. """
  376. 更新用户角色
  377. Args:
  378. user_id: 用户id
  379. role_ids: 角色id列表
  380. Returns:
  381. bool: 操作结果
  382. """
  383. SysUserRoleMapper.delete_user_role_by_user_id(user_id)
  384. cls.insert_user_role(user_id, role_ids)
  385. return True
  386. @classmethod
  387. def import_user(cls, users: List[SysUser], is_update: bool = False) -> str:
  388. """
  389. 导入用户
  390. Args:
  391. users (List[SysUser]): 用户列表
  392. is_update (bool): 是否更新
  393. Returns:
  394. str: 导入消息结果
  395. """
  396. if not users:
  397. raise ServiceException("导入用户不能为空")
  398. success_count = 0
  399. fail_count = 0
  400. success_msg = ""
  401. fail_msg = ""
  402. default_password = SysConfigService.select_config_by_key("sys.user.initPassword")
  403. for user in users:
  404. try:
  405. dto = SysUserMapper.select_user_by_user_name(user.user_name)
  406. if not dto:
  407. user.password = cls._build_password(default_password)
  408. user.create_by_user(security_util.get_user_id())
  409. cls.insert_user(user)
  410. success_count += 1
  411. success_msg += f"<br/> 第{success_count}个账号,导入成功:{user.user_name}"
  412. elif is_update:
  413. user.update_by_user(security_util.get_user_id())
  414. cls.update_user(user)
  415. success_count += 1
  416. success_msg += f"<br/> 第{success_count}个账号,更新成功:{user.user_name}"
  417. else:
  418. fail_count += 1
  419. fail_msg += f"<br/> 第{fail_count}个账号,已存在:{user.user_name}"
  420. except Exception as e:
  421. fail_count += 1
  422. fail_msg += f"<br/> 第{fail_count}个账号,导入失败:{user.user_name},\
  423. 原因:{e.__class__.__name__}"
  424. LogUtil.logger.error(f"导入用户失败,原因:{e}")
  425. if fail_count > 0:
  426. if success_msg:
  427. fail_msg = f"导入成功{success_count}个,失败{fail_count}个。{success_msg} \
  428. <br/>" + fail_msg
  429. else:
  430. fail_msg = f"导入成功{success_count}个,失败{fail_count}个。{fail_msg}"
  431. raise ServiceException(fail_msg)
  432. else:
  433. success_msg = f"恭喜您,数据已全部导入成功!共 {success_count} 条,数据如下:" \
  434. + success_msg
  435. return success_msg
  436. @classmethod
  437. def _build_password(cls, password: Optional[str]) -> str:
  438. """
  439. 构建持久化密码,若为空使用初始化密码,若未加密则加密
  440. """
  441. if not password:
  442. password = SysConfigService.select_config_by_key(
  443. "sys.user.initPassword"
  444. )
  445. if not password:
  446. raise ServiceException("初始化密码未配置")
  447. if cls._is_encrypted(password):
  448. return password
  449. return security_util.encrypt_password(password)
  450. @staticmethod
  451. def _is_encrypted(password: Optional[str]) -> bool:
  452. """
  453. 判断密码是否为bcrypt加密串
  454. """
  455. return bool(password and password.startswith("$2"))