mirror of
				https://github.com/immich-app/immich.git
				synced 2025-11-04 03:27:09 -05:00 
			
		
		
		
	mobile: fallback authentication client model/type to unknown Add fallback for client model/type if device is not ios or android Signed-off-by: Luis Garcia <git@luigi311.com>
		
			
				
	
	
		
			219 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			219 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
import 'dart:async';
 | 
						|
import 'dart:convert';
 | 
						|
import 'dart:io';
 | 
						|
 | 
						|
import 'package:device_info_plus/device_info_plus.dart';
 | 
						|
import 'package:flutter/material.dart';
 | 
						|
import 'package:http/http.dart';
 | 
						|
import 'package:immich_mobile/domain/models/store.model.dart';
 | 
						|
import 'package:immich_mobile/entities/store.entity.dart';
 | 
						|
import 'package:immich_mobile/utils/url_helper.dart';
 | 
						|
import 'package:logging/logging.dart';
 | 
						|
import 'package:openapi/api.dart';
 | 
						|
 | 
						|
class ApiService implements Authentication {
 | 
						|
  late ApiClient _apiClient;
 | 
						|
 | 
						|
  late UsersApi usersApi;
 | 
						|
  late AuthenticationApi authenticationApi;
 | 
						|
  late OAuthApi oAuthApi;
 | 
						|
  late AlbumsApi albumsApi;
 | 
						|
  late AssetsApi assetsApi;
 | 
						|
  late SearchApi searchApi;
 | 
						|
  late ServerApi serverInfoApi;
 | 
						|
  late MapApi mapApi;
 | 
						|
  late PartnersApi partnersApi;
 | 
						|
  late PeopleApi peopleApi;
 | 
						|
  late SharedLinksApi sharedLinksApi;
 | 
						|
  late SyncApi syncApi;
 | 
						|
  late SystemConfigApi systemConfigApi;
 | 
						|
  late ActivitiesApi activitiesApi;
 | 
						|
  late DownloadApi downloadApi;
 | 
						|
  late TrashApi trashApi;
 | 
						|
  late StacksApi stacksApi;
 | 
						|
  late ViewApi viewApi;
 | 
						|
  late MemoriesApi memoriesApi;
 | 
						|
 | 
						|
  ApiService() {
 | 
						|
    // The below line ensures that the api clients are initialized when the service is instantiated
 | 
						|
    // This is required to avoid late initialization errors when the clients are access before the endpoint is resolved
 | 
						|
    setEndpoint('');
 | 
						|
    final endpoint = Store.tryGet(StoreKey.serverEndpoint);
 | 
						|
    if (endpoint != null && endpoint.isNotEmpty) {
 | 
						|
      setEndpoint(endpoint);
 | 
						|
    }
 | 
						|
  }
 | 
						|
  String? _accessToken;
 | 
						|
  final _log = Logger("ApiService");
 | 
						|
 | 
						|
  setEndpoint(String endpoint) {
 | 
						|
    _apiClient = ApiClient(basePath: endpoint, authentication: this);
 | 
						|
    if (_accessToken != null) {
 | 
						|
      setAccessToken(_accessToken!);
 | 
						|
    }
 | 
						|
    usersApi = UsersApi(_apiClient);
 | 
						|
    authenticationApi = AuthenticationApi(_apiClient);
 | 
						|
    oAuthApi = OAuthApi(_apiClient);
 | 
						|
    albumsApi = AlbumsApi(_apiClient);
 | 
						|
    assetsApi = AssetsApi(_apiClient);
 | 
						|
    serverInfoApi = ServerApi(_apiClient);
 | 
						|
    searchApi = SearchApi(_apiClient);
 | 
						|
    mapApi = MapApi(_apiClient);
 | 
						|
    partnersApi = PartnersApi(_apiClient);
 | 
						|
    peopleApi = PeopleApi(_apiClient);
 | 
						|
    sharedLinksApi = SharedLinksApi(_apiClient);
 | 
						|
    syncApi = SyncApi(_apiClient);
 | 
						|
    systemConfigApi = SystemConfigApi(_apiClient);
 | 
						|
    activitiesApi = ActivitiesApi(_apiClient);
 | 
						|
    downloadApi = DownloadApi(_apiClient);
 | 
						|
    trashApi = TrashApi(_apiClient);
 | 
						|
    stacksApi = StacksApi(_apiClient);
 | 
						|
    viewApi = ViewApi(_apiClient);
 | 
						|
    memoriesApi = MemoriesApi(_apiClient);
 | 
						|
  }
 | 
						|
 | 
						|
  Future<String> resolveAndSetEndpoint(String serverUrl) async {
 | 
						|
    final endpoint = await resolveEndpoint(serverUrl);
 | 
						|
    setEndpoint(endpoint);
 | 
						|
 | 
						|
    // Save in local database for next startup
 | 
						|
    Store.put(StoreKey.serverEndpoint, endpoint);
 | 
						|
    return endpoint;
 | 
						|
  }
 | 
						|
 | 
						|
  /// Takes a server URL and attempts to resolve the API endpoint.
 | 
						|
  ///
 | 
						|
  /// Input: [schema://]host[:port][/path]
 | 
						|
  ///  schema - optional (default: https)
 | 
						|
  ///  host   - required
 | 
						|
  ///  port   - optional (default: based on schema)
 | 
						|
  ///  path   - optional
 | 
						|
  Future<String> resolveEndpoint(String serverUrl) async {
 | 
						|
    String url = sanitizeUrl(serverUrl);
 | 
						|
 | 
						|
    // Check for /.well-known/immich
 | 
						|
    final wellKnownEndpoint = await _getWellKnownEndpoint(url);
 | 
						|
    if (wellKnownEndpoint.isNotEmpty) {
 | 
						|
      url = sanitizeUrl(wellKnownEndpoint);
 | 
						|
    }
 | 
						|
 | 
						|
    if (!await _isEndpointAvailable(url)) {
 | 
						|
      throw ApiException(503, "Server is not reachable");
 | 
						|
    }
 | 
						|
 | 
						|
    // Otherwise, assume the URL provided is the api endpoint
 | 
						|
    return url;
 | 
						|
  }
 | 
						|
 | 
						|
  Future<bool> _isEndpointAvailable(String serverUrl) async {
 | 
						|
    if (!serverUrl.endsWith('/api')) {
 | 
						|
      serverUrl += '/api';
 | 
						|
    }
 | 
						|
 | 
						|
    try {
 | 
						|
      await setEndpoint(serverUrl);
 | 
						|
      await serverInfoApi.pingServer().timeout(const Duration(seconds: 5));
 | 
						|
    } on TimeoutException catch (_) {
 | 
						|
      return false;
 | 
						|
    } on SocketException catch (_) {
 | 
						|
      return false;
 | 
						|
    } catch (error, stackTrace) {
 | 
						|
      _log.severe(
 | 
						|
        "Error while checking server availability",
 | 
						|
        error,
 | 
						|
        stackTrace,
 | 
						|
      );
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  Future<String> _getWellKnownEndpoint(String baseUrl) async {
 | 
						|
    final Client client = Client();
 | 
						|
 | 
						|
    try {
 | 
						|
      var headers = {"Accept": "application/json"};
 | 
						|
      headers.addAll(getRequestHeaders());
 | 
						|
 | 
						|
      final res = await client
 | 
						|
          .get(
 | 
						|
            Uri.parse("$baseUrl/.well-known/immich"),
 | 
						|
            headers: headers,
 | 
						|
          )
 | 
						|
          .timeout(const Duration(seconds: 5));
 | 
						|
 | 
						|
      if (res.statusCode == 200) {
 | 
						|
        final data = jsonDecode(res.body);
 | 
						|
        final endpoint = data['api']['endpoint'].toString();
 | 
						|
 | 
						|
        if (endpoint.startsWith('/')) {
 | 
						|
          // Full URL is relative to base
 | 
						|
          return "$baseUrl$endpoint";
 | 
						|
        }
 | 
						|
        return endpoint;
 | 
						|
      }
 | 
						|
    } catch (e) {
 | 
						|
      debugPrint("Could not locate /.well-known/immich at $baseUrl");
 | 
						|
    }
 | 
						|
 | 
						|
    return "";
 | 
						|
  }
 | 
						|
 | 
						|
  Future<void> setAccessToken(String accessToken) async {
 | 
						|
    _accessToken = accessToken;
 | 
						|
    await Store.put(StoreKey.accessToken, accessToken);
 | 
						|
  }
 | 
						|
 | 
						|
  Future<void> setDeviceInfoHeader() async {
 | 
						|
    DeviceInfoPlugin deviceInfoPlugin = DeviceInfoPlugin();
 | 
						|
 | 
						|
    if (Platform.isIOS) {
 | 
						|
      final iosInfo = await deviceInfoPlugin.iosInfo;
 | 
						|
      authenticationApi.apiClient
 | 
						|
          .addDefaultHeader('deviceModel', iosInfo.utsname.machine);
 | 
						|
      authenticationApi.apiClient.addDefaultHeader('deviceType', 'iOS');
 | 
						|
    } else if (Platform.isAndroid) {
 | 
						|
      final androidInfo = await deviceInfoPlugin.androidInfo;
 | 
						|
      authenticationApi.apiClient
 | 
						|
          .addDefaultHeader('deviceModel', androidInfo.model);
 | 
						|
      authenticationApi.apiClient.addDefaultHeader('deviceType', 'Android');
 | 
						|
    } else {
 | 
						|
      authenticationApi.apiClient.addDefaultHeader('deviceModel', 'Unknown');
 | 
						|
      authenticationApi.apiClient.addDefaultHeader('deviceType', 'Unknown');
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  static Map<String, String> getRequestHeaders() {
 | 
						|
    var accessToken = Store.get(StoreKey.accessToken, "");
 | 
						|
    var customHeadersStr = Store.get(StoreKey.customHeaders, "");
 | 
						|
    var header = <String, String>{};
 | 
						|
    if (accessToken.isNotEmpty) {
 | 
						|
      header['x-immich-user-token'] = accessToken;
 | 
						|
    }
 | 
						|
 | 
						|
    if (customHeadersStr.isEmpty) {
 | 
						|
      return header;
 | 
						|
    }
 | 
						|
 | 
						|
    var customHeaders = jsonDecode(customHeadersStr) as Map;
 | 
						|
    customHeaders.forEach((key, value) {
 | 
						|
      header[key] = value;
 | 
						|
    });
 | 
						|
 | 
						|
    return header;
 | 
						|
  }
 | 
						|
 | 
						|
  @override
 | 
						|
  Future<void> applyToParams(
 | 
						|
    List<QueryParam> queryParams,
 | 
						|
    Map<String, String> headerParams,
 | 
						|
  ) {
 | 
						|
    return Future<void>(() {
 | 
						|
      var headers = ApiService.getRequestHeaders();
 | 
						|
      headerParams.addAll(headers);
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  ApiClient get apiClient => _apiClient;
 | 
						|
}
 |