8dcc01b2be
* 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>
138 lines
3.8 KiB
Dart
138 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;
|
|
}
|