mirror of
https://github.com/immich-app/immich.git
synced 2025-05-24 01:12:58 -04: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,
|
|
),
|
|
);
|
|
}
|
|
}
|