account_repository.dart 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. import 'dart:async';
  2. import 'package:get/get.dart';
  3. import 'package:injectable/injectable.dart';
  4. import 'package:location/base/app_base_request.dart';
  5. import 'package:location/data/api/atmob_api.dart';
  6. import 'package:location/data/api/request/login_request.dart';
  7. import 'package:location/data/api/request/send_code_request.dart';
  8. import 'package:location/data/bean/location_info.dart';
  9. import 'package:location/data/bean/member_status_info.dart';
  10. import 'package:location/data/bean/user_info.dart';
  11. import 'package:location/data/consts/constants.dart';
  12. import 'package:location/data/consts/error_code.dart';
  13. import 'package:location/data/repositories/friends_repository.dart';
  14. import 'package:location/data/repositories/message_repository.dart';
  15. import 'package:location/data/repositories/phone_event_repository.dart';
  16. import 'package:location/data/repositories/urgent_contact_repository.dart';
  17. import 'package:location/di/get_it.dart';
  18. import 'package:location/push_notification/ios_push_notification_service.dart';
  19. import 'package:location/resource/string.gen.dart';
  20. import 'package:location/socket/atmob_location_client.dart';
  21. import 'package:location/utils/async_util.dart';
  22. import 'package:location/utils/atmob_log.dart';
  23. import 'package:location/utils/http_handler.dart';
  24. import 'package:location/utils/mmkv_util.dart';
  25. import '../../sdk/map/map_helper.dart';
  26. import '../api/request/notification_report_request.dart';
  27. import '../api/request/user_avatar_update_request.dart';
  28. import '../api/response/login_response.dart';
  29. import '../api/response/member_status_response.dart';
  30. @lazySingleton
  31. class AccountRepository {
  32. final AtmobApi atmobApi;
  33. final String tag = "AccountRepository";
  34. static final String keyAccountLoginPhoneNum = 'key_account_login_phone_num';
  35. static final String keyAccountLoginToken = 'key_account_login_token';
  36. static final String keyAccountLoginUserId = 'key_account_login_user_id';
  37. RxnString loginPhoneNum = RxnString();
  38. RxBool isLogin = RxBool(false);
  39. Rxn<MemberStatusInfo> memberStatusInfo = Rxn<MemberStatusInfo>();
  40. int? _lastRequestCodeTime;
  41. int _errorCodeTimes = 0;
  42. Timer? refreshMemberHandler;
  43. CancelableFuture? memberStatusFuture;
  44. static String? token = KVUtil.getString(keyAccountLoginToken, null);
  45. late final FriendsRepository friendsRepository;
  46. late final MessageRepository messageRepository;
  47. late final UrgentContactRepository urgentContactRepository;
  48. late final PhoneEventRepository phoneEventRepository;
  49. final Rx<UserInfo> mineUserInfo = Rx<UserInfo>(UserInfo(
  50. id: Constants.mineLocationId,
  51. phoneNumber: StringName.locationMine,
  52. isMine: true));
  53. AccountRepository(this.atmobApi) {
  54. AtmobLog.d(tag, '$tag....init');
  55. friendsRepository = FriendsRepository.getInstance();
  56. messageRepository = MessageRepository.getInstance();
  57. urgentContactRepository = UrgentContactRepository.getInstance();
  58. phoneEventRepository = PhoneEventRepository.getInstance();
  59. isLogin.bindStream(
  60. loginPhoneNum.map((value) {
  61. return value?.isNotEmpty == true;
  62. }),
  63. );
  64. loginPhoneNum.value = KVUtil.getString(keyAccountLoginPhoneNum, null);
  65. refreshMemberStatus();
  66. MapHelper.addLocationListener((location) {
  67. mineUserInfo.value.lastLocation.value =
  68. LocationInfo.fromMapLocation(location);
  69. });
  70. }
  71. static AccountRepository getInstance() {
  72. return getIt.get<AccountRepository>();
  73. }
  74. Future<void> loginSendCode(String phoneNum) {
  75. final currentTime = DateTime.now().millisecondsSinceEpoch;
  76. // 检查是否在 60 秒内重复请求
  77. if (currentTime - (_lastRequestCodeTime ?? 0) < 60 * 1000) {
  78. throw RequestCodeTooOftenException();
  79. }
  80. return atmobApi
  81. .loginSendCode(SendCodeRequest(phoneNum))
  82. .then(HttpHandler.handle(true))
  83. .then((value) {
  84. _lastRequestCodeTime = currentTime;
  85. _errorCodeTimes = 0;
  86. });
  87. }
  88. Future<LoginResponse> loginUserLogin(
  89. String phoneNum, String verificationCode) {
  90. if (_errorCodeTimes >= 5) {
  91. return Future.error(LoginTooOftenException());
  92. }
  93. return atmobApi
  94. .loginUserLogin(LoginRequest(phoneNum, verificationCode))
  95. .then(HttpHandler.handle(true))
  96. .then((response) {
  97. _errorCodeTimes = 0;
  98. onLoginSuccess(phoneNum, response.authToken);
  99. return response;
  100. }).catchError((error) {
  101. if (error is ServerErrorException &&
  102. error.code == ErrorCode.verificationCodeError) {
  103. _errorCodeTimes++;
  104. }
  105. throw error;
  106. });
  107. }
  108. void onLoginSuccess(String phoneNum, String authToken) {
  109. AccountRepository.token = authToken;
  110. loginPhoneNum.value = phoneNum;
  111. KVUtil.putString(keyAccountLoginPhoneNum, phoneNum);
  112. KVUtil.putString(keyAccountLoginToken, authToken);
  113. AtmobLocationClient.connectWebSocket();
  114. refreshMemberStatus();
  115. friendsRepository.refreshFriends();
  116. messageRepository.refreshFriendWaitingCount();
  117. messageRepository.refreshUnreadMessage();
  118. urgentContactRepository.requestUrgentContactList();
  119. //上传推送token
  120. onRequestNotificationReport();
  121. //上报事件
  122. phoneEventRepository.startReportPhoneEvent();
  123. }
  124. void logout() {
  125. token = null;
  126. refreshMemberHandler?.cancel();
  127. AtmobLocationClient.disConnectWebSocket();
  128. KVUtil.putString(keyAccountLoginPhoneNum, null);
  129. KVUtil.putString(keyAccountLoginToken, null);
  130. KVUtil.putString(keyAccountLoginUserId, null);
  131. KVUtil.putString(Constants.keyLastSelectFriendId, null);
  132. loginPhoneNum.value = null;
  133. memberStatusInfo.value = null;
  134. friendsRepository.clearFriends();
  135. messageRepository.clearMessage();
  136. urgentContactRepository.clearContactList();
  137. phoneEventRepository.stopReportPhoneEvent();
  138. }
  139. void refreshMemberStatus() {
  140. memberStatusFuture?.cancel();
  141. memberStatusFuture = AsyncUtil.retryWithExponentialBackoff(
  142. () => getMemberStatus(), 10, predicate: (error) {
  143. if (error is ServerErrorException) {
  144. return error.code != ErrorCode.noLoginError;
  145. }
  146. return true;
  147. });
  148. memberStatusFuture?.then((data) {
  149. AtmobLog.d(tag, "getMemberStatus success: ${memberStatusInfo.value}");
  150. }).catchError((error) {
  151. AtmobLog.e(tag, "getMemberStatus error: $error");
  152. });
  153. }
  154. Future<MemberStatusResponse> getMemberStatus() {
  155. return atmobApi
  156. .getMemberStatus(AppBaseRequest())
  157. .then(HttpHandler.handle(false))
  158. .then((response) {
  159. refreshMemberHandler?.cancel();
  160. updateAvatar(response.avatar);
  161. KVUtil.putString(keyAccountLoginUserId, response.deviceId);
  162. if (!response.permanent && !response.expired) {
  163. refreshMemberHandler = Timer(
  164. Duration(
  165. milliseconds: response.endTimestamp - response.serverTimestamp),
  166. () => refreshMemberStatus());
  167. }
  168. return response;
  169. }).then((response) {
  170. MemberStatusInfo memberStatusInfo = MemberStatusInfo(
  171. level: response.level,
  172. endTimestamp: response.endTimestamp,
  173. expired: response.expired,
  174. permanent: response.permanent,
  175. trialed: response.trialed,
  176. avatar: response.avatar,
  177. trialEndTimestamp: response.trialEndTimestamp);
  178. this.memberStatusInfo.value = memberStatusInfo;
  179. return response;
  180. });
  181. }
  182. void updateAvatar(String? avatar) {
  183. mineUserInfo.value.avatar = avatar;
  184. mineUserInfo.refresh();
  185. }
  186. Future<bool> userAvatarUpdate(String avatar) {
  187. if (avatar == mineUserInfo.value.avatar) {
  188. return Future.value(true);
  189. }
  190. return atmobApi
  191. .userAvatarUpdate(UserAvatarUpdateRequest(avatar))
  192. .then(HttpHandler.handle(true))
  193. .then((_) {
  194. updateAvatar(avatar);
  195. AtmobLog.d(tag, "userAvatarUpdate success: $avatar");
  196. return true;
  197. });
  198. }
  199. bool memberIsExpired() {
  200. return memberStatusInfo.value == null ||
  201. memberStatusInfo.value?.expired == true;
  202. }
  203. Future<void> userClear() {
  204. return atmobApi.userClear(AppBaseRequest()).then(HttpHandler.handle(true));
  205. }
  206. ///请求推送的数据
  207. Future<void> onRequestNotificationReport() async {
  208. // 初始化推送服务
  209. var tokenStr = await IosPushNotificationService.getDeviceToken();
  210. print("tokenStrsfsdf---${tokenStr}");
  211. return atmobApi
  212. .notificationReport(
  213. NotificationReportRequest(deviceToken: tokenStr as String))
  214. .then(HttpHandler.handle(false))
  215. .then((response) {})
  216. .catchError((_) {});
  217. }
  218. }
  219. class RequestCodeTooOftenException implements Exception {
  220. final String message;
  221. /// 可选的构造函数,支持自定义错误信息
  222. RequestCodeTooOftenException([this.message = '请求验证码过于频繁']);
  223. @override
  224. String toString() => message;
  225. }
  226. class LoginTooOftenException implements Exception {
  227. final String message;
  228. /// 可选的构造函数,支持自定义错误信息
  229. LoginTooOftenException([this.message = '登录过于频繁']);
  230. @override
  231. String toString() => message;
  232. }