mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-30 18:35:00 -04:00 
			
		
		
		
	* add shared links page * feat(mobile): shared link items * feat(mobile): create / edit shared links page * server: add changeExpiryTime to SharedLinkEditDto * fix(mobile): edit expiry to never * mobile: add icon when shares list is empty * mobile: create new share from album / timeline * mobile: add translation texts * mobile: minor ui fixes * fix: handle serverURL with /api path * mobile: show share link on successful creation * mobile: shared links list - 2 column layout * mobile: use sharedlink pod class instead of dto * mobile: show error on link creation * mobile: show share icon only when remote assets are in selection * mobile: use server endpoint instead of server url * styling * styling --------- Co-authored-by: shalong-tanwen <139912620+shalong-tanwen@users.noreply.github.com> Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
		
			
				
	
	
		
			139 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			139 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:async';
 | |
| import 'dart:convert';
 | |
| import 'dart:io';
 | |
| 
 | |
| import 'package:flutter/material.dart';
 | |
| import 'package:immich_mobile/shared/models/store.dart';
 | |
| import 'package:immich_mobile/utils/url_helper.dart';
 | |
| import 'package:openapi/api.dart';
 | |
| import 'package:http/http.dart';
 | |
| 
 | |
| class ApiService {
 | |
|   late ApiClient _apiClient;
 | |
| 
 | |
|   late UserApi userApi;
 | |
|   late AuthenticationApi authenticationApi;
 | |
|   late OAuthApi oAuthApi;
 | |
|   late AlbumApi albumApi;
 | |
|   late AssetApi assetApi;
 | |
|   late SearchApi searchApi;
 | |
|   late ServerInfoApi serverInfoApi;
 | |
|   late PartnerApi partnerApi;
 | |
|   late PersonApi personApi;
 | |
|   late AuditApi auditApi;
 | |
|   late SharedLinkApi sharedLinkApi;
 | |
| 
 | |
|   ApiService() {
 | |
|     final endpoint = Store.tryGet(StoreKey.serverEndpoint);
 | |
|     if (endpoint != null && endpoint.isNotEmpty) {
 | |
|       setEndpoint(endpoint);
 | |
|     }
 | |
|   }
 | |
|   String? _authToken;
 | |
| 
 | |
|   setEndpoint(String endpoint) {
 | |
|     _apiClient = ApiClient(basePath: endpoint);
 | |
|     if (_authToken != null) {
 | |
|       setAccessToken(_authToken!);
 | |
|     }
 | |
|     userApi = UserApi(_apiClient);
 | |
|     authenticationApi = AuthenticationApi(_apiClient);
 | |
|     oAuthApi = OAuthApi(_apiClient);
 | |
|     albumApi = AlbumApi(_apiClient);
 | |
|     assetApi = AssetApi(_apiClient);
 | |
|     serverInfoApi = ServerInfoApi(_apiClient);
 | |
|     searchApi = SearchApi(_apiClient);
 | |
|     partnerApi = PartnerApi(_apiClient);
 | |
|     personApi = PersonApi(_apiClient);
 | |
|     auditApi = AuditApi(_apiClient);
 | |
|     sharedLinkApi = SharedLinkApi(_apiClient);
 | |
|   }
 | |
| 
 | |
|   Future<String> resolveAndSetEndpoint(String serverUrl) async {
 | |
|     final endpoint = await _resolveEndpoint(serverUrl);
 | |
|     setEndpoint(endpoint);
 | |
| 
 | |
|     // Save in hivebox 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 {
 | |
|     final url = sanitizeUrl(serverUrl);
 | |
| 
 | |
|     if (!await _isEndpointAvailable(serverUrl)) {
 | |
|       throw ApiException(503, "Server is not reachable");
 | |
|     }
 | |
| 
 | |
|     // Check for /.well-known/immich
 | |
|     final wellKnownEndpoint = await _getWellKnownEndpoint(url);
 | |
|     if (wellKnownEndpoint.isNotEmpty) return wellKnownEndpoint;
 | |
| 
 | |
|     // Otherwise, assume the URL provided is the api endpoint
 | |
|     return url;
 | |
|   }
 | |
| 
 | |
|   Future<bool> _isEndpointAvailable(String serverUrl) async {
 | |
|     final Client client = Client();
 | |
| 
 | |
|     if (!serverUrl.endsWith('/api')) {
 | |
|       serverUrl += '/api';
 | |
|     }
 | |
| 
 | |
|     // Throw Socket or Timeout exceptions,
 | |
|     // we do not care if the endpoints hits an HTTP error
 | |
|     try {
 | |
|       await client
 | |
|           .get(
 | |
|             Uri.parse(serverUrl),
 | |
|           )
 | |
|           .timeout(const Duration(seconds: 5));
 | |
|     } on TimeoutException catch (_) {
 | |
|       return false;
 | |
|     } on SocketException catch (_) {
 | |
|       return false;
 | |
|     }
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   Future<String> _getWellKnownEndpoint(String baseUrl) async {
 | |
|     final Client client = Client();
 | |
| 
 | |
|     try {
 | |
|       final res = await client.get(
 | |
|         Uri.parse("$baseUrl/.well-known/immich"),
 | |
|         headers: {"Accept": "application/json"},
 | |
|       );
 | |
| 
 | |
|       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 "";
 | |
|   }
 | |
| 
 | |
|   setAccessToken(String accessToken) {
 | |
|     _authToken = accessToken;
 | |
|     _apiClient.addDefaultHeader('Authorization', 'Bearer $accessToken');
 | |
|   }
 | |
| 
 | |
|   ApiClient get apiClient => _apiClient;
 | |
| }
 |