auth_store.dart 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. import 'package:chicken_farm/apis/index.dart';
  2. import 'package:chicken_farm/core/config/app_config.dart';
  3. import 'package:chicken_farm/core/services/navigation_service.dart';
  4. import 'package:chicken_farm/core/utils/jwt_token.dart';
  5. import 'package:chicken_farm/core/utils/logger.dart';
  6. import 'package:chicken_farm/modes/auth/login_model.dart';
  7. import 'package:chicken_farm/modes/user/user_model.dart';
  8. import 'package:chicken_farm/routes/app_routes.dart';
  9. import 'package:flutter_riverpod/flutter_riverpod.dart';
  10. import 'package:go_router/go_router.dart';
  11. /// 认证状态枚举
  12. enum AuthState { authenticated, unauthenticated, loading }
  13. /// 用户认证信息状态
  14. class AuthInfo {
  15. final AuthState state;
  16. final String? token;
  17. final UserModel? user;
  18. final List<String>? permissions;
  19. final List<String>? roles;
  20. AuthInfo({
  21. required this.state,
  22. this.token,
  23. this.user,
  24. this.permissions,
  25. this.roles,
  26. });
  27. /// 创建已认证状态
  28. AuthInfo.authenticated({
  29. required String token,
  30. required UserModel user,
  31. List<String>? permissions,
  32. List<String>? roles,
  33. }) : this(
  34. state: AuthState.authenticated,
  35. token: token,
  36. user: user,
  37. permissions: permissions,
  38. roles: roles,
  39. );
  40. /// 创建未认证状态
  41. AuthInfo.unauthenticated()
  42. : this(
  43. state: AuthState.unauthenticated,
  44. token: null,
  45. user: null,
  46. permissions: const [],
  47. roles: const [],
  48. );
  49. /// 创建加载状态
  50. AuthInfo.loading()
  51. : this(
  52. state: AuthState.loading,
  53. token: null,
  54. user: null,
  55. permissions: const [],
  56. roles: const [],
  57. );
  58. /// 复制对象并更新部分字段
  59. AuthInfo copyWith({
  60. AuthState? state,
  61. String? token,
  62. UserModel? user,
  63. List<String>? permissions,
  64. List<String>? roles,
  65. }) {
  66. return AuthInfo(
  67. state: state ?? this.state,
  68. token: token ?? this.token,
  69. user: user ?? this.user,
  70. permissions: permissions ?? this.permissions,
  71. roles: roles ?? this.roles,
  72. );
  73. }
  74. }
  75. class AuthStore extends StateNotifier<AuthInfo> {
  76. AuthStore() : super(AuthInfo.unauthenticated()) {
  77. _init();
  78. }
  79. /// 初始化认证状态
  80. Future<void> _init() async {
  81. state = AuthInfo.loading();
  82. try {
  83. final token = await JwtToken.getToken();
  84. if (token != null) {
  85. // 如果有token,则设置为已认证状态
  86. // 尝试获取用户信息以验证token有效性
  87. try {
  88. final userInfo = await apis.loginApi.getInfo();
  89. state = AuthInfo.authenticated(
  90. token: token,
  91. user: userInfo.user!,
  92. permissions: userInfo.permissions,
  93. roles: userInfo.roles,
  94. );
  95. logger.i('已登录 state: $state');
  96. } catch (e) {
  97. // Token无效,清除本地存储
  98. await JwtToken.clear();
  99. state = AuthInfo.unauthenticated();
  100. }
  101. } else {
  102. state = AuthInfo.unauthenticated();
  103. }
  104. } catch (e) {
  105. state = AuthInfo.unauthenticated();
  106. }
  107. }
  108. /// 登录操作
  109. Future<void> login(LoginModel loginModel) async {
  110. state = AuthInfo.loading();
  111. try {
  112. loginModel.clientId = AppConfig.clientId;
  113. final authResult = await apis.loginApi.login(loginModel);
  114. final token = authResult.accessToken;
  115. await JwtToken.setToken(token, authResult.refreshToken);
  116. // 获取用户信息
  117. final userInfo = await apis.loginApi.getInfo();
  118. state = AuthInfo.authenticated(
  119. token: token,
  120. user: userInfo.user!,
  121. permissions: userInfo.permissions,
  122. roles: userInfo.roles,
  123. );
  124. logger.i('登录成功 state: $state');
  125. } catch (e) {
  126. await JwtToken.clear();
  127. state = AuthInfo.unauthenticated();
  128. }
  129. }
  130. /// 登出操作
  131. Future<void> logout() async {
  132. try {
  133. await apis.loginApi.logout();
  134. } catch (e) {
  135. // 即使API调用失败也要清除本地状态
  136. logger.e('Logout API call failed: $e');
  137. } finally {
  138. await JwtToken.clear();
  139. state = AuthInfo.unauthenticated();
  140. if (NavigationService.navigatorKey.currentState != null &&
  141. NavigationService.navigatorKey.currentContext != null) {
  142. NavigationService.navigatorKey.currentContext!.goNamed(
  143. AppRouteNames.login,
  144. );
  145. }
  146. }
  147. }
  148. /// 刷新token
  149. Future<void> refreshToken() async {
  150. try {
  151. final refreshToken = await JwtToken.getRefreshToken();
  152. if (refreshToken != null) {
  153. final authResult = await apis.loginApi.refreshToken(refreshToken);
  154. final newToken = authResult.accessToken;
  155. await JwtToken.setToken(newToken, authResult.refreshToken);
  156. state = state.copyWith(token: newToken);
  157. }
  158. } catch (e) {
  159. // 刷新失败则登出
  160. await logout();
  161. rethrow;
  162. }
  163. }
  164. /// 重新获取用户信息
  165. Future<void> getUserInfo() async {
  166. try {
  167. final userInfo = await apis.loginApi.getInfo();
  168. state = state.copyWith(
  169. user: userInfo.user,
  170. permissions: userInfo.permissions,
  171. roles: userInfo.roles,
  172. );
  173. } catch (e) {
  174. rethrow;
  175. }
  176. }
  177. /// 检查是否有特定权限
  178. bool hasPermission(String permission) {
  179. return state.permissions?.contains(permission) ?? false;
  180. }
  181. /// 检查是否有特定角色
  182. bool hasRole(String role) {
  183. return state.roles?.contains(role) ?? false;
  184. }
  185. /// 是否是超级管理员
  186. bool isSuperAdmin() {
  187. if (state.user == null) return false;
  188. return state.user!.userName == "admin" ||
  189. (state.roles?.contains("super_admin") ?? false);
  190. }
  191. }
  192. // 添加 Provider 实例
  193. final authStoreProvider = StateNotifierProvider<AuthStore, AuthInfo>(
  194. (ref) => AuthStore(),
  195. );