mirror of
				https://github.com/immich-app/immich.git
				synced 2025-11-03 19:17:11 -05:00 
			
		
		
		
	* fix: ensure oauth state param matches before finishing oauth flow Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * chore: upgrade openid-client to v6 Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * feat: use PKCE for oauth2 on supported clients Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * feat: use state and PKCE in mobile app Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * fix: remove obsolete oauth repository init Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * fix: rewrite callback url if mobile redirect url is enabled Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * fix: propagate oidc client error cause when oauth callback fails Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * fix: adapt auth service tests to required state and PKCE params Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * fix: update sdk types Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * fix: adapt oauth e2e test to work with PKCE Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> * fix: allow insecure (http) oauth clients Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> --------- Signed-off-by: Tin Pecirep <tin.pecirep@gmail.com> Co-authored-by: Jason Rasmussen <jason@rasm.me>
		
			
				
	
	
		
			68 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			68 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
import 'package:flutter_web_auth_2/flutter_web_auth_2.dart';
 | 
						|
import 'package:immich_mobile/services/api.service.dart';
 | 
						|
import 'package:logging/logging.dart';
 | 
						|
import 'package:openapi/api.dart';
 | 
						|
 | 
						|
// Redirect URL = app.immich:///oauth-callback
 | 
						|
 | 
						|
class OAuthService {
 | 
						|
  final ApiService _apiService;
 | 
						|
  final callbackUrlScheme = 'app.immich';
 | 
						|
  final log = Logger('OAuthService');
 | 
						|
  OAuthService(this._apiService);
 | 
						|
 | 
						|
  Future<String?> getOAuthServerUrl(
 | 
						|
    String serverUrl,
 | 
						|
    String state,
 | 
						|
    String codeChallenge,
 | 
						|
  ) async {
 | 
						|
    // Resolve API server endpoint from user provided serverUrl
 | 
						|
    await _apiService.resolveAndSetEndpoint(serverUrl);
 | 
						|
    final redirectUri = '$callbackUrlScheme:///oauth-callback';
 | 
						|
    log.info(
 | 
						|
      "Starting OAuth flow with redirect URI: $redirectUri",
 | 
						|
    );
 | 
						|
 | 
						|
    final dto = await _apiService.oAuthApi.startOAuth(
 | 
						|
      OAuthConfigDto(
 | 
						|
        redirectUri: redirectUri,
 | 
						|
        state: state,
 | 
						|
        codeChallenge: codeChallenge,
 | 
						|
      ),
 | 
						|
    );
 | 
						|
 | 
						|
    final authUrl = dto?.url;
 | 
						|
    log.info('Received Authorization URL: $authUrl');
 | 
						|
 | 
						|
    return authUrl;
 | 
						|
  }
 | 
						|
 | 
						|
  Future<LoginResponseDto?> oAuthLogin(
 | 
						|
    String oauthUrl,
 | 
						|
    String state,
 | 
						|
    String codeVerifier,
 | 
						|
  ) async {
 | 
						|
    String result = await FlutterWebAuth2.authenticate(
 | 
						|
      url: oauthUrl,
 | 
						|
      callbackUrlScheme: callbackUrlScheme,
 | 
						|
    );
 | 
						|
 | 
						|
    log.info('Received OAuth callback: $result');
 | 
						|
 | 
						|
    if (result.startsWith('app.immich:/oauth-callback')) {
 | 
						|
      result = result.replaceAll(
 | 
						|
        'app.immich:/oauth-callback',
 | 
						|
        'app.immich:///oauth-callback',
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    return await _apiService.oAuthApi.finishOAuth(
 | 
						|
      OAuthCallbackDto(
 | 
						|
        url: result,
 | 
						|
        state: state,
 | 
						|
        codeVerifier: codeVerifier,
 | 
						|
      ),
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 |