2024-11-23 11:54:38 +00:00
|
|
|
import 'dart:async';
|
2024-11-09 10:28:45 +00:00
|
|
|
import 'dart:convert';
|
|
|
|
import 'dart:developer';
|
2024-12-13 17:32:13 +00:00
|
|
|
import 'dart:io';
|
2024-11-08 16:09:46 +00:00
|
|
|
|
|
|
|
import 'package:dio/dio.dart';
|
|
|
|
import 'package:dio_smart_retry/dio_smart_retry.dart';
|
2024-12-13 17:32:13 +00:00
|
|
|
import 'package:flutter/foundation.dart';
|
2024-12-15 14:54:00 +00:00
|
|
|
import 'package:flutter/material.dart';
|
2024-12-13 17:32:13 +00:00
|
|
|
import 'package:package_info_plus/package_info_plus.dart';
|
|
|
|
import 'package:device_info_plus/device_info_plus.dart';
|
2024-12-15 14:54:00 +00:00
|
|
|
import 'package:provider/provider.dart';
|
2024-11-10 14:14:27 +00:00
|
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
2024-12-20 17:58:49 +00:00
|
|
|
import 'package:surface/providers/config.dart';
|
2024-12-15 14:54:00 +00:00
|
|
|
import 'package:surface/providers/widget.dart';
|
2024-11-23 11:54:38 +00:00
|
|
|
import 'package:synchronized/synchronized.dart';
|
2024-11-08 16:09:46 +00:00
|
|
|
|
2024-11-10 14:14:27 +00:00
|
|
|
const kNetworkServerDirectory = [
|
2024-12-07 17:15:17 +00:00
|
|
|
('Solar Network', 'https://api.sn.solsynth.dev'),
|
2024-11-10 14:14:27 +00:00
|
|
|
('Local', 'http://localhost:8001'),
|
|
|
|
];
|
|
|
|
|
2024-12-21 09:23:46 +00:00
|
|
|
Completer<String?>? _refreshCompleter;
|
|
|
|
|
2024-11-08 16:09:46 +00:00
|
|
|
class SnNetworkProvider {
|
2024-11-24 02:54:55 +00:00
|
|
|
late final Dio client;
|
2024-11-08 16:09:46 +00:00
|
|
|
|
2024-11-10 14:14:27 +00:00
|
|
|
late final SharedPreferences _prefs;
|
2024-12-20 17:58:49 +00:00
|
|
|
late final ConfigProvider _config;
|
2024-12-15 14:54:00 +00:00
|
|
|
late final HomeWidgetProvider _home;
|
2024-11-09 10:28:45 +00:00
|
|
|
|
2024-12-13 17:32:13 +00:00
|
|
|
String? _userAgent;
|
|
|
|
|
2024-12-15 14:54:00 +00:00
|
|
|
SnNetworkProvider(BuildContext context) {
|
|
|
|
_home = context.read<HomeWidgetProvider>();
|
|
|
|
|
2024-11-08 16:09:46 +00:00
|
|
|
client = Dio();
|
|
|
|
|
2024-11-18 15:04:36 +00:00
|
|
|
client.interceptors.add(RetryInterceptor(
|
|
|
|
dio: client,
|
|
|
|
retries: 3,
|
|
|
|
retryDelays: const [
|
|
|
|
Duration(milliseconds: 300),
|
|
|
|
Duration(milliseconds: 1000),
|
|
|
|
Duration(milliseconds: 3000),
|
|
|
|
],
|
|
|
|
));
|
2024-11-08 16:09:46 +00:00
|
|
|
|
2024-11-09 10:28:45 +00:00
|
|
|
client.interceptors.add(
|
|
|
|
InterceptorsWrapper(
|
|
|
|
onRequest: (
|
|
|
|
RequestOptions options,
|
|
|
|
RequestInterceptorHandler handler,
|
|
|
|
) async {
|
2024-11-14 16:24:46 +00:00
|
|
|
final atk = await getFreshAtk();
|
|
|
|
if (atk != null) {
|
|
|
|
options.headers['Authorization'] = 'Bearer $atk';
|
2024-11-09 10:28:45 +00:00
|
|
|
}
|
2024-12-13 17:32:13 +00:00
|
|
|
if (_userAgent != null) {
|
|
|
|
options.headers['User-Agent'] = _userAgent!;
|
|
|
|
}
|
2024-11-14 16:24:46 +00:00
|
|
|
return handler.next(options);
|
2024-11-09 10:28:45 +00:00
|
|
|
},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
|
2024-12-20 17:58:49 +00:00
|
|
|
_config = context.read<ConfigProvider>();
|
|
|
|
_config.initialize().then((_) {
|
|
|
|
_prefs = _config.prefs;
|
|
|
|
client.options.baseUrl = _config.serverUrl;
|
2024-11-10 14:14:27 +00:00
|
|
|
});
|
2024-12-23 13:55:07 +00:00
|
|
|
|
2024-11-08 16:09:46 +00:00
|
|
|
}
|
2024-11-09 03:16:14 +00:00
|
|
|
|
2024-12-21 03:56:18 +00:00
|
|
|
static Future<Dio> createOffContextClient() async {
|
|
|
|
final prefs = await SharedPreferences.getInstance();
|
|
|
|
final client = Dio();
|
|
|
|
client.interceptors.add(RetryInterceptor(
|
|
|
|
dio: client,
|
|
|
|
retries: 3,
|
|
|
|
retryDelays: const [
|
|
|
|
Duration(milliseconds: 300),
|
|
|
|
Duration(milliseconds: 1000),
|
|
|
|
Duration(milliseconds: 3000),
|
|
|
|
],
|
|
|
|
));
|
|
|
|
final ua = await _getUserAgent();
|
|
|
|
client.interceptors.add(
|
|
|
|
InterceptorsWrapper(
|
|
|
|
onRequest: (
|
|
|
|
RequestOptions options,
|
|
|
|
RequestInterceptorHandler handler,
|
|
|
|
) async {
|
2024-12-21 13:06:14 +00:00
|
|
|
final atk = await _getFreshAtk(client, prefs.getString(kAtkStoreKey), prefs.getString(kRtkStoreKey), (atk, rtk) {
|
2024-12-21 09:23:46 +00:00
|
|
|
prefs.setString(kAtkStoreKey, atk);
|
|
|
|
prefs.setString(kRtkStoreKey, rtk);
|
|
|
|
});
|
|
|
|
if (atk != null) {
|
|
|
|
options.headers['Authorization'] = 'Bearer $atk';
|
|
|
|
}
|
2024-12-21 03:56:18 +00:00
|
|
|
options.headers['User-Agent'] = ua;
|
|
|
|
return handler.next(options);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
client.options.baseUrl = prefs.getString(kNetworkServerStoreKey) ?? kNetworkServerDefault;
|
|
|
|
|
|
|
|
return client;
|
|
|
|
}
|
|
|
|
|
2024-12-23 13:55:07 +00:00
|
|
|
Future<void> setConfigWithNative() async {
|
|
|
|
_home.saveWidgetData("nex_server_url", client.options.baseUrl);
|
|
|
|
}
|
|
|
|
|
2024-12-21 03:56:18 +00:00
|
|
|
static Future<String> _getUserAgent() async {
|
2024-12-13 17:32:13 +00:00
|
|
|
final String platformInfo;
|
|
|
|
if (kIsWeb) {
|
|
|
|
final deviceInfo = await DeviceInfoPlugin().webBrowserInfo;
|
|
|
|
platformInfo = 'Web; ${deviceInfo.vendor}';
|
|
|
|
} else if (Platform.isAndroid) {
|
|
|
|
final deviceInfo = await DeviceInfoPlugin().androidInfo;
|
|
|
|
platformInfo = 'Android; ${deviceInfo.brand} ${deviceInfo.model}; ${deviceInfo.id}';
|
|
|
|
} else if (Platform.isIOS) {
|
|
|
|
final deviceInfo = await DeviceInfoPlugin().iosInfo;
|
|
|
|
platformInfo = 'iOS; ${deviceInfo.model}; ${deviceInfo.name}';
|
|
|
|
} else if (Platform.isMacOS) {
|
|
|
|
final deviceInfo = await DeviceInfoPlugin().macOsInfo;
|
|
|
|
platformInfo = 'MacOS; ${deviceInfo.model}; ${deviceInfo.hostName}';
|
|
|
|
} else if (Platform.isWindows) {
|
|
|
|
final deviceInfo = await DeviceInfoPlugin().windowsInfo;
|
|
|
|
platformInfo = 'Windows NT; ${deviceInfo.productName}; ${deviceInfo.computerName}';
|
|
|
|
} else if (Platform.isLinux) {
|
|
|
|
final deviceInfo = await DeviceInfoPlugin().linuxInfo;
|
|
|
|
platformInfo = 'Linux; ${deviceInfo.prettyName}';
|
|
|
|
} else {
|
|
|
|
platformInfo = 'Unknown';
|
|
|
|
}
|
|
|
|
|
|
|
|
final packageInfo = await PackageInfo.fromPlatform();
|
|
|
|
|
2024-12-21 03:56:18 +00:00
|
|
|
return 'Solian/${packageInfo.version}+${packageInfo.buildNumber} ($platformInfo)';
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> initializeUserAgent() async {
|
|
|
|
_userAgent = await _getUserAgent();
|
2024-12-13 17:32:13 +00:00
|
|
|
}
|
|
|
|
|
2024-11-23 11:54:38 +00:00
|
|
|
final tkLock = Lock();
|
|
|
|
|
2024-11-14 16:24:46 +00:00
|
|
|
Future<String?> getFreshAtk() async {
|
2024-12-21 13:06:14 +00:00
|
|
|
return await _getFreshAtk(client, _prefs.getString(kAtkStoreKey), _prefs.getString(kRtkStoreKey), (atk, rtk) {
|
2024-12-21 09:23:46 +00:00
|
|
|
setTokenPair(atk, rtk);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-12-21 13:06:14 +00:00
|
|
|
static Future<String?> _getFreshAtk(Dio client, String? atk, String? rtk, Function(String atk, String rtk)? onRefresh) async {
|
2024-11-23 11:54:38 +00:00
|
|
|
if (_refreshCompleter != null) {
|
|
|
|
return await _refreshCompleter!.future;
|
|
|
|
} else {
|
|
|
|
_refreshCompleter = Completer<String?>();
|
|
|
|
}
|
|
|
|
|
2024-11-14 16:24:46 +00:00
|
|
|
try {
|
|
|
|
if (atk != null) {
|
|
|
|
final atkParts = atk.split('.');
|
|
|
|
if (atkParts.length != 3) {
|
|
|
|
throw Exception('invalid format of access token');
|
|
|
|
}
|
|
|
|
|
|
|
|
var rawPayload = atkParts[1].replaceAll('-', '+').replaceAll('_', '/');
|
|
|
|
switch (rawPayload.length % 4) {
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
rawPayload += '==';
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
rawPayload += '=';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw Exception('illegal format of access token payload');
|
|
|
|
}
|
|
|
|
|
|
|
|
final b64 = utf8.fuse(base64Url);
|
|
|
|
final payload = b64.decode(rawPayload);
|
|
|
|
final exp = jsonDecode(payload)['exp'];
|
|
|
|
if (exp <= DateTime.now().millisecondsSinceEpoch ~/ 1000) {
|
|
|
|
log('Access token need refresh, doing it at ${DateTime.now()}');
|
2024-12-21 13:06:14 +00:00
|
|
|
final result = await _refreshToken(client.options.baseUrl, rtk);
|
2024-12-21 09:23:46 +00:00
|
|
|
if (result == null) {
|
|
|
|
atk = null;
|
|
|
|
} else {
|
|
|
|
atk = result.$1;
|
|
|
|
onRefresh?.call(atk, result.$2);
|
|
|
|
}
|
2024-11-14 16:24:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (atk != null) {
|
2024-11-23 11:54:38 +00:00
|
|
|
_refreshCompleter!.complete(atk);
|
2024-11-14 16:24:46 +00:00
|
|
|
return atk;
|
|
|
|
} else {
|
|
|
|
log('Access token refresh failed...');
|
2024-11-23 11:54:38 +00:00
|
|
|
_refreshCompleter!.complete(null);
|
2024-11-14 16:24:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
log('Failed to authenticate user: $err');
|
2024-11-23 11:54:38 +00:00
|
|
|
_refreshCompleter!.completeError(err);
|
|
|
|
} finally {
|
|
|
|
_refreshCompleter = null;
|
2024-11-14 16:24:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-11-09 03:16:14 +00:00
|
|
|
String getAttachmentUrl(String ky) {
|
2024-11-12 16:13:27 +00:00
|
|
|
if (ky.startsWith("http")) return ky;
|
2024-11-09 03:16:14 +00:00
|
|
|
return '${client.options.baseUrl}/cgi/uc/attachments/$ky';
|
|
|
|
}
|
2024-11-09 10:28:45 +00:00
|
|
|
|
2024-11-23 11:54:38 +00:00
|
|
|
void setTokenPair(String atk, String rtk) {
|
|
|
|
_prefs.setString(kAtkStoreKey, atk);
|
|
|
|
_prefs.setString(kRtkStoreKey, rtk);
|
2024-11-09 10:28:45 +00:00
|
|
|
}
|
|
|
|
|
2024-11-23 11:54:38 +00:00
|
|
|
void clearTokenPair() {
|
|
|
|
_prefs.remove(kAtkStoreKey);
|
|
|
|
_prefs.remove(kRtkStoreKey);
|
2024-11-09 10:28:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Future<String?> refreshToken() async {
|
2024-11-23 11:54:38 +00:00
|
|
|
final rtk = _prefs.getString(kRtkStoreKey);
|
2024-12-21 09:23:46 +00:00
|
|
|
final result = await _refreshToken(client.options.baseUrl, rtk);
|
|
|
|
if (result == null) return null;
|
|
|
|
_prefs.setString(kAtkStoreKey, result.$1);
|
|
|
|
_prefs.setString(kRtkStoreKey, result.$2);
|
|
|
|
return result.$1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<(String, String)?> _refreshToken(String baseUrl, String? rtk) async {
|
2024-11-09 10:28:45 +00:00
|
|
|
if (rtk == null) return null;
|
|
|
|
|
2024-11-09 11:32:21 +00:00
|
|
|
final dio = Dio();
|
2024-12-21 09:23:46 +00:00
|
|
|
dio.options.baseUrl = baseUrl;
|
2024-11-09 11:32:21 +00:00
|
|
|
|
|
|
|
final resp = await dio.post('/cgi/id/auth/token', data: {
|
2024-11-09 10:28:45 +00:00
|
|
|
'grant_type': 'refresh_token',
|
|
|
|
'refresh_token': rtk,
|
|
|
|
});
|
|
|
|
|
2024-12-21 09:23:46 +00:00
|
|
|
final String atk = resp.data['access_token'];
|
|
|
|
final String nRtk = resp.data['refresh_token'];
|
2024-11-09 10:28:45 +00:00
|
|
|
|
2024-12-21 09:23:46 +00:00
|
|
|
return (atk, nRtk);
|
2024-11-09 10:28:45 +00:00
|
|
|
}
|
2024-11-10 14:14:27 +00:00
|
|
|
|
|
|
|
void setBaseUrl(String url) {
|
2024-12-20 17:58:49 +00:00
|
|
|
_config.serverUrl = url;
|
2024-11-10 14:14:27 +00:00
|
|
|
client.options.baseUrl = url;
|
|
|
|
}
|
2024-11-08 16:09:46 +00:00
|
|
|
}
|