forked from firka/student-legacy
250 lines
6.8 KiB
Dart
250 lines
6.8 KiB
Dart
// ignore_for_file: avoid_print
|
|
|
|
import 'dart:convert';
|
|
import 'dart:io';
|
|
|
|
import 'package:filcnaplo/api/login.dart';
|
|
import 'package:filcnaplo/api/nonce.dart';
|
|
import 'package:filcnaplo/api/providers/user_provider.dart';
|
|
import 'package:filcnaplo/api/providers/status_provider.dart';
|
|
import 'package:filcnaplo/models/settings.dart';
|
|
import 'package:filcnaplo/models/user.dart';
|
|
import 'package:filcnaplo/utils/jwt.dart';
|
|
import 'package:filcnaplo_kreta_api/client/api.dart';
|
|
import 'package:http/http.dart' as http;
|
|
import 'package:http/io_client.dart' as http;
|
|
import 'dart:async';
|
|
|
|
class KretaClient {
|
|
String? accessToken;
|
|
String? refreshToken;
|
|
String? idToken;
|
|
String? userAgent;
|
|
late http.Client client;
|
|
|
|
late final SettingsProvider _settings;
|
|
late final UserProvider _user;
|
|
late final StatusProvider _status;
|
|
|
|
bool _loginRefreshing = false;
|
|
|
|
KretaClient({
|
|
this.accessToken,
|
|
required SettingsProvider settings,
|
|
required UserProvider user,
|
|
required StatusProvider status,
|
|
}) : _settings = settings,
|
|
_user = user,
|
|
_status = status,
|
|
userAgent = settings.config.userAgent {
|
|
var ioclient = HttpClient();
|
|
ioclient.badCertificateCallback = _checkCerts;
|
|
client = http.IOClient(ioclient);
|
|
}
|
|
|
|
bool _checkCerts(X509Certificate cert, String host, int port) {
|
|
return _settings.developerMode;
|
|
}
|
|
|
|
Future<dynamic> getAPI(
|
|
String url, {
|
|
Map<String, String>? headers,
|
|
bool autoHeader = true,
|
|
bool json = true,
|
|
bool rawResponse = false,
|
|
}) async {
|
|
Map<String, String> headerMap;
|
|
|
|
if (rawResponse) json = false;
|
|
|
|
if (headers != null) {
|
|
headerMap = headers;
|
|
} else {
|
|
headerMap = {};
|
|
}
|
|
|
|
try {
|
|
http.Response? res;
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
if (autoHeader) {
|
|
if (!headerMap.containsKey("authorization") && accessToken != null) {
|
|
headerMap["authorization"] = "Bearer $accessToken";
|
|
}
|
|
if (!headerMap.containsKey("user-agent") && userAgent != null) {
|
|
headerMap["user-agent"] = "$userAgent";
|
|
}
|
|
}
|
|
|
|
res = await client.get(Uri.parse(url), headers: headerMap);
|
|
_status.triggerRequest(res);
|
|
|
|
if (res.statusCode == 401) {
|
|
await refreshLogin();
|
|
headerMap.remove("authorization");
|
|
} else {
|
|
break;
|
|
}
|
|
|
|
// Wait before retrying
|
|
await Future.delayed(const Duration(milliseconds: 500));
|
|
}
|
|
|
|
if (res == null) throw "Login error";
|
|
if (res.body == 'invalid_grant' || res.body.replaceAll(' ', '') == '') {
|
|
throw "Auth error";
|
|
}
|
|
|
|
if (json) {
|
|
return jsonDecode(res.body);
|
|
} else if (rawResponse) {
|
|
return res.bodyBytes;
|
|
} else {
|
|
return res.body;
|
|
}
|
|
} on http.ClientException catch (error) {
|
|
print(
|
|
"ERROR: KretaClient.getAPI ($url) ClientException: ${error.message}");
|
|
} catch (error) {
|
|
print("ERROR: KretaClient.getAPI ($url) ${error.runtimeType}: $error");
|
|
}
|
|
}
|
|
|
|
Future<dynamic> postAPI(
|
|
String url, {
|
|
Map<String, String>? headers,
|
|
bool autoHeader = true,
|
|
bool json = true,
|
|
Object? body,
|
|
}) async {
|
|
Map<String, String> headerMap;
|
|
|
|
if (headers != null) {
|
|
headerMap = headers;
|
|
} else {
|
|
headerMap = {};
|
|
}
|
|
|
|
try {
|
|
http.Response? res;
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
if (autoHeader) {
|
|
if (!headerMap.containsKey("authorization") && accessToken != null) {
|
|
headerMap["authorization"] = "Bearer $accessToken";
|
|
}
|
|
if (!headerMap.containsKey("user-agent") && userAgent != null) {
|
|
headerMap["user-agent"] = "$userAgent";
|
|
}
|
|
if (!headerMap.containsKey("content-type")) {
|
|
headerMap["content-type"] = "application/json";
|
|
}
|
|
if (url.contains('kommunikacio/uzenetek')) {
|
|
headerMap["X-Uzenet-Lokalizacio"] = "hu-HU";
|
|
}
|
|
}
|
|
|
|
res = await client.post(Uri.parse(url), headers: headerMap, body: body);
|
|
if (res.statusCode == 401) {
|
|
await refreshLogin();
|
|
headerMap.remove("authorization");
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (res == null) throw "Login error";
|
|
|
|
if (json) {
|
|
return jsonDecode(res.body);
|
|
} else {
|
|
return res.body;
|
|
}
|
|
} on http.ClientException catch (error) {
|
|
print(
|
|
"ERROR: KretaClient.postAPI ($url) ClientException: ${error.message}");
|
|
} catch (error) {
|
|
print("ERROR: KretaClient.postAPI ($url) ${error.runtimeType}: $error");
|
|
}
|
|
}
|
|
|
|
Future<void> refreshLogin() async {
|
|
if (_loginRefreshing) return;
|
|
_loginRefreshing = true;
|
|
|
|
User? loginUser = _user.user;
|
|
if (loginUser == null) return;
|
|
|
|
Map<String, String> headers = {
|
|
"content-type": "application/x-www-form-urlencoded",
|
|
};
|
|
|
|
String nonceStr = await getAPI(KretaAPI.nonce, json: false);
|
|
Nonce nonce =
|
|
getNonce(nonceStr, loginUser.username, loginUser.instituteCode);
|
|
headers.addAll(nonce.header());
|
|
|
|
if (_settings.presentationMode) {
|
|
print("DEBUG: refreshLogin: ${loginUser.id}");
|
|
} else {
|
|
print("DEBUG: refreshLogin: ${loginUser.id} ${loginUser.name}");
|
|
}
|
|
|
|
Map? loginRes = await postAPI(
|
|
KretaAPI.login,
|
|
headers: headers,
|
|
body: User.loginBody(
|
|
username: loginUser.username,
|
|
password: loginUser.password,
|
|
instituteCode: loginUser.instituteCode,
|
|
),
|
|
);
|
|
|
|
if (loginRes != null) {
|
|
if (loginRes.containsKey("access_token")) {
|
|
accessToken = loginRes["access_token"];
|
|
}
|
|
if (loginRes.containsKey("refresh_token")) {
|
|
refreshToken = loginRes["refresh_token"];
|
|
}
|
|
|
|
// Update role
|
|
loginUser.role =
|
|
JwtUtils.getRoleFromJWT(accessToken ?? "") ?? Role.student;
|
|
}
|
|
|
|
if (refreshToken != null) {
|
|
Map? refreshRes = await postAPI(KretaAPI.login,
|
|
headers: headers,
|
|
body: User.refreshBody(
|
|
refreshToken: refreshToken!,
|
|
instituteCode: loginUser.instituteCode));
|
|
if (refreshRes != null) {
|
|
if (refreshRes.containsKey("id_token")) {
|
|
idToken = refreshRes["id_token"];
|
|
}
|
|
}
|
|
}
|
|
|
|
_loginRefreshing = false;
|
|
}
|
|
|
|
Future<void> logout() async {
|
|
User? loginUser = _user.user;
|
|
if (loginUser == null) return;
|
|
|
|
Map<String, String> headers = {
|
|
"content-type": "application/x-www-form-urlencoded",
|
|
};
|
|
|
|
await postAPI(
|
|
KretaAPI.logout,
|
|
headers: headers,
|
|
body: User.logoutBody(
|
|
refreshToken: refreshToken!,
|
|
),
|
|
json: false,
|
|
);
|
|
}
|
|
}
|