import 'dart:convert'; import 'package:get/get.dart'; import 'package:http/http.dart' as http; import 'package:logger/logger.dart'; import 'package:marco/model/user_permission.dart'; import 'package:marco/model/employee_info.dart'; import 'package:marco/model/projects_model.dart'; import 'package:marco/helpers/services/storage/local_storage.dart'; import 'package:marco/helpers/services/auth_service.dart'; final Logger logger = Logger(); class PermissionService { static final Map> _userDataCache = {}; /// Fetches all user-related data (permissions, employee info, projects) static Future> fetchAllUserData( String token, { bool hasRetried = false, }) async { // Return cached data if already available if (_userDataCache.containsKey(token)) { return _userDataCache[token]!; } final uri = Uri.parse('https://stageapi.marcoaiot.com/api/user/profile'); final headers = {'Authorization': 'Bearer $token'}; try { final response = await http.get(uri, headers: headers); if (response.statusCode == 200) { final data = json.decode(response.body)['data']; final result = { 'permissions': _parsePermissions(data['featurePermissions']), 'employeeInfo': _parseEmployeeInfo(data['employeeInfo']), 'projects': _parseProjectsInfo(data['projects']), }; _userDataCache[token] = result; return result; } // Handle 401 by attempting a single retry with refreshed token if (response.statusCode == 401 && !hasRetried) { final refreshed = await AuthService.refreshToken(); if (refreshed) { final newToken = await LocalStorage.getJwtToken(); if (newToken != null && newToken.isNotEmpty) { return fetchAllUserData(newToken, hasRetried: true); } } await _handleUnauthorized(); throw Exception('Unauthorized. Token refresh failed.'); } final error = json.decode(response.body)['message'] ?? 'Unknown error'; throw Exception('Failed to fetch user data: $error'); } catch (e) { logger.e('Error fetching user data: $e'); rethrow; } } /// Clears auth data and redirects to login static Future _handleUnauthorized() async { await LocalStorage.removeToken('jwt_token'); await LocalStorage.removeToken('refresh_token'); await LocalStorage.setLoggedInUser(false); Get.offAllNamed('/auth/login-option'); } /// Converts raw permission data into list of `UserPermission` static List _parsePermissions(List permissions) { return permissions .map((id) => UserPermission.fromJson({'id': id})) .toList(); } /// Converts raw employee JSON into `EmployeeInfo` static EmployeeInfo _parseEmployeeInfo(Map data) { return EmployeeInfo.fromJson(data); } /// Converts raw projects JSON into list of `ProjectInfo` static List _parseProjectsInfo(List projects) { return projects.map((proj) => ProjectInfo.fromJson(proj)).toList(); } }