## 주요 변경사항 ### 아키텍처 개선 - Clean Architecture 패턴 적용 (Domain, Data, Presentation 레이어 분리) - Use Case 패턴 도입으로 비즈니스 로직 캡슐화 - Repository 패턴으로 데이터 접근 추상화 - 의존성 주입 구조 개선 ### 상태 관리 최적화 - 모든 Controller에서 불필요한 상태 관리 로직 제거 - 페이지네이션 로직 통일 및 간소화 - 에러 처리 로직 개선 (에러 메시지 한글화) - 로딩 상태 관리 최적화 ### Mock 서비스 제거 - MockDataService 완전 제거 - 모든 화면을 실제 API 전용으로 전환 - 불필요한 Mock 관련 코드 정리 ### UI/UX 개선 - Overview 화면 대시보드 기능 강화 - 라이선스 만료 알림 위젯 추가 - 사이드바 네비게이션 개선 - 일관된 UI 컴포넌트 사용 ### 코드 품질 - 중복 코드 제거 및 함수 추출 - 파일별 책임 분리 명확화 - 테스트 코드 업데이트 ## 영향 범위 - 모든 화면의 Controller 리팩토링 - API 통신 레이어 구조 개선 - 에러 처리 및 로깅 시스템 개선 ## 향후 계획 - 단위 테스트 커버리지 확대 - 통합 테스트 시나리오 추가 - 성능 모니터링 도구 통합
248 lines
7.8 KiB
Dart
248 lines
7.8 KiB
Dart
import 'dart:async';
|
|
import 'dart:convert';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:dartz/dartz.dart';
|
|
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
|
|
import 'package:injectable/injectable.dart';
|
|
import 'package:superport/core/errors/failures.dart';
|
|
import 'package:superport/data/datasources/remote/auth_remote_datasource.dart';
|
|
import 'package:superport/data/models/auth/auth_user.dart';
|
|
import 'package:superport/data/models/auth/login_request.dart';
|
|
import 'package:superport/data/models/auth/login_response.dart';
|
|
import 'package:superport/data/models/auth/logout_request.dart';
|
|
import 'package:superport/data/models/auth/refresh_token_request.dart';
|
|
import 'package:superport/data/models/auth/token_response.dart';
|
|
import 'package:superport/core/config/environment.dart' as env;
|
|
|
|
abstract class AuthService {
|
|
Future<Either<Failure, LoginResponse>> login(LoginRequest request);
|
|
Future<Either<Failure, void>> logout();
|
|
Future<Either<Failure, TokenResponse>> refreshToken();
|
|
Future<bool> isLoggedIn();
|
|
Future<AuthUser?> getCurrentUser();
|
|
Future<String?> getAccessToken();
|
|
Future<String?> getRefreshToken();
|
|
Future<void> clearSession();
|
|
Stream<bool> get authStateChanges;
|
|
}
|
|
|
|
@LazySingleton(as: AuthService)
|
|
class AuthServiceImpl implements AuthService {
|
|
final AuthRemoteDataSource _authRemoteDataSource;
|
|
final FlutterSecureStorage _secureStorage;
|
|
|
|
static const String _accessTokenKey = 'access_token';
|
|
static const String _refreshTokenKey = 'refresh_token';
|
|
static const String _userKey = 'user';
|
|
static const String _tokenExpiryKey = 'token_expiry';
|
|
|
|
final _authStateController = StreamController<bool>.broadcast();
|
|
|
|
AuthServiceImpl(
|
|
this._authRemoteDataSource,
|
|
this._secureStorage,
|
|
);
|
|
|
|
@override
|
|
Future<Either<Failure, LoginResponse>> login(LoginRequest request) async {
|
|
try {
|
|
debugPrint('[AuthService] login 시작');
|
|
|
|
// API 모드로 로그인 처리
|
|
debugPrint('[AuthService] API 모드로 로그인 처리');
|
|
final result = await _authRemoteDataSource.login(request);
|
|
|
|
return await result.fold(
|
|
(failure) async => Left(failure),
|
|
(loginResponse) async {
|
|
// 토큰 및 사용자 정보 저장
|
|
await _saveTokens(
|
|
loginResponse.accessToken,
|
|
loginResponse.refreshToken,
|
|
loginResponse.expiresIn,
|
|
);
|
|
await _saveUser(loginResponse.user);
|
|
|
|
// 인증 상태 변경 알림
|
|
_authStateController.add(true);
|
|
|
|
return Right(loginResponse);
|
|
},
|
|
);
|
|
} catch (e, stackTrace) {
|
|
debugPrint('[AuthService] login 예외 발생: $e');
|
|
debugPrint('[AuthService] Stack trace: $stackTrace');
|
|
return Left(ServerFailure(message: '로그인 처리 중 오류가 발생했습니다.'));
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<Either<Failure, void>> logout() async {
|
|
try {
|
|
final refreshToken = await getRefreshToken();
|
|
if (refreshToken != null) {
|
|
final request = LogoutRequest(refreshToken: refreshToken);
|
|
await _authRemoteDataSource.logout(request);
|
|
}
|
|
|
|
await clearSession();
|
|
_authStateController.add(false);
|
|
|
|
return const Right(null);
|
|
} catch (e) {
|
|
// 로그아웃 API 실패 시에도 로컬 세션은 정리
|
|
await clearSession();
|
|
_authStateController.add(false);
|
|
return const Right(null);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<Either<Failure, TokenResponse>> refreshToken() async {
|
|
try {
|
|
final refreshToken = await getRefreshToken();
|
|
if (refreshToken == null) {
|
|
return Left(AuthenticationFailure(
|
|
message: '리프레시 토큰이 없습니다. 다시 로그인해주세요.',
|
|
));
|
|
}
|
|
|
|
final request = RefreshTokenRequest(refreshToken: refreshToken);
|
|
final result = await _authRemoteDataSource.refreshToken(request);
|
|
|
|
return await result.fold(
|
|
(failure) async {
|
|
// 토큰 갱신 실패 시 세션 정리
|
|
await clearSession();
|
|
_authStateController.add(false);
|
|
return Left(failure);
|
|
},
|
|
(tokenResponse) async {
|
|
// 새 토큰 저장
|
|
await _saveTokens(
|
|
tokenResponse.accessToken,
|
|
tokenResponse.refreshToken,
|
|
tokenResponse.expiresIn,
|
|
);
|
|
return Right(tokenResponse);
|
|
},
|
|
);
|
|
} catch (e) {
|
|
return Left(ServerFailure(message: '토큰 갱신 중 오류가 발생했습니다.'));
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<bool> isLoggedIn() async {
|
|
try {
|
|
final accessToken = await getAccessToken();
|
|
if (accessToken == null) return false;
|
|
|
|
// 토큰 만료 확인
|
|
final expiryStr = await _secureStorage.read(key: _tokenExpiryKey);
|
|
if (expiryStr != null) {
|
|
final expiry = DateTime.parse(expiryStr);
|
|
if (DateTime.now().isAfter(expiry)) {
|
|
// 토큰이 만료되었으면 갱신 시도
|
|
final refreshResult = await refreshToken();
|
|
return refreshResult.isRight();
|
|
}
|
|
}
|
|
|
|
return true;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<AuthUser?> getCurrentUser() async {
|
|
try {
|
|
final userStr = await _secureStorage.read(key: _userKey);
|
|
if (userStr == null) return null;
|
|
|
|
final userJson = jsonDecode(userStr);
|
|
return AuthUser.fromJson(userJson);
|
|
} catch (e) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<String?> getAccessToken() async {
|
|
try {
|
|
final token = await _secureStorage.read(key: _accessTokenKey);
|
|
if (token != null && token.length > 20) {
|
|
debugPrint('[AuthService] getAccessToken: Found (${token.substring(0, 20)}...)');
|
|
} else if (token != null) {
|
|
debugPrint('[AuthService] getAccessToken: Found (${token})');
|
|
} else {
|
|
debugPrint('[AuthService] getAccessToken: Not found');
|
|
}
|
|
return token;
|
|
} catch (e) {
|
|
debugPrint('[AuthService] getAccessToken error: $e');
|
|
return null;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<String?> getRefreshToken() async {
|
|
try {
|
|
return await _secureStorage.read(key: _refreshTokenKey);
|
|
} catch (e) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> clearSession() async {
|
|
try {
|
|
await _secureStorage.delete(key: _accessTokenKey);
|
|
await _secureStorage.delete(key: _refreshTokenKey);
|
|
await _secureStorage.delete(key: _userKey);
|
|
await _secureStorage.delete(key: _tokenExpiryKey);
|
|
} catch (e) {
|
|
// 에러가 발생해도 계속 진행
|
|
}
|
|
}
|
|
|
|
@override
|
|
Stream<bool> get authStateChanges => _authStateController.stream;
|
|
|
|
Future<void> _saveTokens(
|
|
String accessToken,
|
|
String refreshToken,
|
|
int expiresIn,
|
|
) async {
|
|
debugPrint('[AuthService] Saving tokens...');
|
|
final accessTokenPreview = accessToken.length > 20 ? '${accessToken.substring(0, 20)}...' : accessToken;
|
|
final refreshTokenPreview = refreshToken.length > 20 ? '${refreshToken.substring(0, 20)}...' : refreshToken;
|
|
debugPrint('[AuthService] Access token: $accessTokenPreview');
|
|
debugPrint('[AuthService] Refresh token: $refreshTokenPreview');
|
|
debugPrint('[AuthService] Expires in: $expiresIn seconds');
|
|
|
|
await _secureStorage.write(key: _accessTokenKey, value: accessToken);
|
|
await _secureStorage.write(key: _refreshTokenKey, value: refreshToken);
|
|
|
|
// 토큰 만료 시간 저장 (현재 시간 + expiresIn 초)
|
|
final expiry = DateTime.now().add(Duration(seconds: expiresIn));
|
|
await _secureStorage.write(
|
|
key: _tokenExpiryKey,
|
|
value: expiry.toIso8601String(),
|
|
);
|
|
|
|
debugPrint('[AuthService] Tokens saved successfully');
|
|
}
|
|
|
|
Future<void> _saveUser(AuthUser user) async {
|
|
final userJson = jsonEncode(user.toJson());
|
|
await _secureStorage.write(key: _userKey, value: userJson);
|
|
}
|
|
|
|
void dispose() {
|
|
_authStateController.close();
|
|
}
|
|
} |