Compare commits

..

4 Commits

Author SHA1 Message Date
Guanciottaman
ec66cd8b8b
Merge ff252d6b14 into 95e09ffc94 2025-03-15 14:48:59 +06:00
Kingkor Roy Tirtho
95e09ffc94 chore: remove certificate check for dio 2025-03-15 14:23:38 +06:00
Kingkor Roy Tirtho
968fd09eb3 chore: add random user agent 2025-03-15 10:48:34 +06:00
Kingkor Roy Tirtho
1a32264bc7 fix: spotify authentication 429 errors 2025-03-15 10:37:29 +06:00
4 changed files with 60 additions and 10 deletions

View File

@ -5,7 +5,7 @@ import 'dart:io';
import 'package:collection/collection.dart'; import 'package:collection/collection.dart';
import 'package:desktop_webview_window/desktop_webview_window.dart'; import 'package:desktop_webview_window/desktop_webview_window.dart';
import 'package:dio/dio.dart'; import 'package:dio/dio.dart';
import 'package:dio/io.dart'; import 'package:dio_http2_adapter/dio_http2_adapter.dart';
import 'package:drift/drift.dart'; import 'package:drift/drift.dart';
import 'package:flutter_inappwebview/flutter_inappwebview.dart' import 'package:flutter_inappwebview/flutter_inappwebview.dart'
hide X509Certificate; hide X509Certificate;
@ -19,6 +19,7 @@ import 'package:spotube/utils/platform.dart';
import 'package:otp_util/otp_util.dart'; import 'package:otp_util/otp_util.dart';
// ignore: implementation_imports // ignore: implementation_imports
import 'package:otp_util/src/utils/generic_util.dart'; import 'package:otp_util/src/utils/generic_util.dart';
import 'package:spotube/utils/service_utils.dart';
extension ExpirationAuthenticationTableData on AuthenticationTableData { extension ExpirationAuthenticationTableData on AuthenticationTableData {
bool get isExpired => DateTime.now().isAfter(expiration); bool get isExpired => DateTime.now().isAfter(expiration);
@ -34,13 +35,17 @@ extension ExpirationAuthenticationTableData on AuthenticationTableData {
class AuthenticationNotifier extends AsyncNotifier<AuthenticationTableData?> { class AuthenticationNotifier extends AsyncNotifier<AuthenticationTableData?> {
static final Dio dio = () { static final Dio dio = () {
final dio = Dio(); final dio = Dio()
..httpClientAdapter = Http2Adapter(
(dio.httpClientAdapter as IOHttpClientAdapter) ConnectionManager(
.createHttpClient = () => HttpClient() idleTimeout: const Duration(seconds: 10),
..badCertificateCallback = (X509Certificate cert, String host, int port) { onClientCreate: (uri, clientSettings) {
return host.endsWith("spotify.com") && port == 443; clientSettings.onBadCertificate = (X509Certificate cert) {
}; return uri.host.endsWith("spotify.com");
};
},
),
);
return dio; return dio;
}(); }();
@ -163,7 +168,18 @@ class AuthenticationNotifier extends AsyncNotifier<AuthenticationTableData?> {
final secret = base32FromBytes(secretBytes, secretSauce); final secret = base32FromBytes(secretBytes, secretSauce);
final res = await dio.get("https://open.spotify.com/server-time"); final res = await dio.get(
"https://open.spotify.com/server-time",
options: Options(
headers: {
"Host": "open.spotify.com",
"User-Agent": ServiceUtils.randomUserAgent(
kIsDesktop ? UserAgentDevice.desktop : UserAgentDevice.mobile,
),
"accept": "*/*",
},
),
);
final serverTimeSeconds = res.data["serverTime"] as int; final serverTimeSeconds = res.data["serverTime"] as int;
final totp = TOTP( final totp = TOTP(

View File

@ -1,6 +1,6 @@
import 'dart:math';
import 'dart:typed_data'; import 'dart:typed_data';
import 'package:auto_route/auto_route.dart';
import 'package:dio/dio.dart'; import 'package:dio/dio.dart';
import 'package:flutter_cache_manager/flutter_cache_manager.dart'; import 'package:flutter_cache_manager/flutter_cache_manager.dart';
@ -28,6 +28,11 @@ import 'package:spotube/collections/env.dart';
import 'package:version/version.dart'; import 'package:version/version.dart';
enum UserAgentDevice {
desktop,
mobile,
}
abstract class ServiceUtils { abstract class ServiceUtils {
static final _englishMatcherRegex = RegExp( static final _englishMatcherRegex = RegExp(
"^[a-zA-Z0-9\\s!\"#\$%&\\'()*+,-.\\/:;<=>?@\\[\\]^_`{|}~]*\$", "^[a-zA-Z0-9\\s!\"#\$%&\\'()*+,-.\\/:;<=>?@\\[\\]^_`{|}~]*\$",
@ -417,4 +422,16 @@ abstract class ServiceUtils {
return null; return null;
} }
} }
static int randomNumber(int min, int max) {
return min + Random().nextInt(max - min);
}
static String randomUserAgent(UserAgentDevice type) {
if (type == UserAgentDevice.desktop) {
return "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_${randomNumber(11, 15)}_${randomNumber(4, 9)}) AppleWebKit/${randomNumber(530, 537)}.${randomNumber(30, 37)} (KHTML, like Gecko) Chrome/${randomNumber(80, 105)}.0.${randomNumber(3000, 4500)}.${randomNumber(60, 125)} Safari/${randomNumber(530, 537)}.${randomNumber(30, 36)}";
} else {
return "Mozilla/5.0 (Linux; Android ${randomNumber(8, 13)}) AppleWebKit/${randomNumber(530, 537)}.${randomNumber(30, 36)} (KHTML, like Gecko) Chrome/${randomNumber(101, 116)}.0.${randomNumber(3000, 6000)}.${randomNumber(60, 125)} Mobile Safari/${randomNumber(530, 537)}.${randomNumber(30, 36)}";
}
}
} }

View File

@ -544,6 +544,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "5.7.0" version: "5.7.0"
dio_http2_adapter:
dependency: "direct main"
description:
name: dio_http2_adapter
sha256: b8bd5d587fd228a461711f8b82f378ccd4bf1fbf7802e7663ca60d7b5ce0e3aa
url: "https://pub.dev"
source: hosted
version: "2.6.0"
dio_web_adapter: dio_web_adapter:
dependency: transitive dependency: transitive
description: description:
@ -1186,6 +1194,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.2.2" version: "1.2.2"
http2:
dependency: transitive
description:
name: http2
sha256: "382d3aefc5bd6dc68c6b892d7664f29b5beb3251611ae946a98d35158a82bbfa"
url: "https://pub.dev"
source: hosted
version: "2.3.1"
http_methods: http_methods:
dependency: transitive dependency: transitive
description: description:

View File

@ -141,6 +141,7 @@ dependencies:
http_parser: ^4.1.2 http_parser: ^4.1.2
collection: any collection: any
otp_util: ^1.0.2 otp_util: ^1.0.2
dio_http2_adapter: ^2.6.0
dev_dependencies: dev_dependencies:
build_runner: ^2.4.13 build_runner: ^2.4.13