195 lines
5.1 KiB
Dart
195 lines
5.1 KiB
Dart
import 'dart:convert';
|
|
import 'package:http/http.dart' as http;
|
|
|
|
typedef TokenProvider = String? Function();
|
|
typedef UnauthorizedHandler = Future<void> Function();
|
|
|
|
class ApiService {
|
|
String _baseUrl;
|
|
final http.Client _client;
|
|
final Duration timeout;
|
|
TokenProvider? _getToken;
|
|
UnauthorizedHandler? _onUnauthorized;
|
|
|
|
ApiService({
|
|
required String baseUrl,
|
|
http.Client? client,
|
|
this.timeout = const Duration(seconds: 30),
|
|
}) : _baseUrl = baseUrl,
|
|
_client = client ?? http.Client();
|
|
|
|
String get baseUrl => _baseUrl;
|
|
|
|
void setBaseUrl(String url) {
|
|
_baseUrl = url;
|
|
}
|
|
|
|
void setTokenProvider(TokenProvider provider) {
|
|
_getToken = provider;
|
|
}
|
|
|
|
void setUnauthorizedHandler(UnauthorizedHandler handler) {
|
|
_onUnauthorized = handler;
|
|
}
|
|
|
|
void dispose() {
|
|
_client.close();
|
|
}
|
|
|
|
Map<String, String> _buildHeaders(Map<String, String>? extra) {
|
|
final token = _getToken?.call();
|
|
final headers = {'accept': 'application/json', ...?extra};
|
|
if (token != null && token.isNotEmpty) {
|
|
headers['Authorization'] = 'Bearer $token';
|
|
}
|
|
return headers;
|
|
}
|
|
|
|
Future<dynamic> get(String endpoint, {Map<String, String>? headers}) async {
|
|
final response = await _client
|
|
.get(
|
|
Uri.parse('$baseUrl$endpoint'),
|
|
headers: _buildHeaders(headers),
|
|
)
|
|
.timeout(timeout);
|
|
return _processResponse(response);
|
|
}
|
|
|
|
Future<dynamic> post(
|
|
String endpoint,
|
|
dynamic data, {
|
|
Map<String, String>? headers,
|
|
}) async {
|
|
final hasBody = data != null;
|
|
final response = await _client
|
|
.post(
|
|
Uri.parse('$baseUrl$endpoint'),
|
|
headers: _buildHeaders(hasBody ? _jsonHeaders(headers) : headers),
|
|
body: hasBody ? jsonEncode(data) : null,
|
|
)
|
|
.timeout(timeout);
|
|
return _processResponse(response);
|
|
}
|
|
|
|
Future<dynamic> postForm(String endpoint, Map<String, String> data) async {
|
|
final response = await _client
|
|
.post(
|
|
Uri.parse('$baseUrl$endpoint'),
|
|
headers: _buildHeaders({
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
'accept': 'application/json',
|
|
}),
|
|
body: data, // http package handles form-encoding for Map<String, String>
|
|
)
|
|
.timeout(timeout);
|
|
return _processResponse(response);
|
|
}
|
|
|
|
Future<dynamic> put(
|
|
String endpoint,
|
|
dynamic data, {
|
|
Map<String, String>? headers,
|
|
}) async {
|
|
final hasBody = data != null;
|
|
final response = await _client
|
|
.put(
|
|
Uri.parse('$baseUrl$endpoint'),
|
|
headers: _buildHeaders(hasBody ? _jsonHeaders(headers) : headers),
|
|
body: hasBody ? jsonEncode(data) : null,
|
|
)
|
|
.timeout(timeout);
|
|
return _processResponse(response);
|
|
}
|
|
|
|
Future<dynamic> delete(
|
|
String endpoint, {
|
|
Map<String, String>? headers,
|
|
}) async {
|
|
final response = await _client
|
|
.delete(
|
|
Uri.parse('$baseUrl$endpoint'),
|
|
headers: _buildHeaders(headers),
|
|
)
|
|
.timeout(timeout);
|
|
return _processResponse(response);
|
|
}
|
|
|
|
Map<String, String> _jsonHeaders(Map<String, String>? extra) {
|
|
return {'Content-Type': 'application/json', if (extra != null) ...extra};
|
|
}
|
|
|
|
Future<dynamic> _processResponse(http.Response res) async {
|
|
final body = _decodeBody(res);
|
|
if (res.statusCode >= 200 && res.statusCode < 300) {
|
|
return body;
|
|
}
|
|
|
|
if (res.statusCode == 401 && _onUnauthorized != null) {
|
|
await _onUnauthorized!();
|
|
}
|
|
|
|
final message = _extractErrorMessage(body);
|
|
throw ApiException(
|
|
statusCode: res.statusCode,
|
|
message: message,
|
|
body: body,
|
|
);
|
|
}
|
|
|
|
dynamic _decodeBody(http.Response res) {
|
|
if (res.body.isEmpty) return null;
|
|
|
|
final contentType = res.headers['content-type'] ?? '';
|
|
final shouldTryJson = contentType.contains('application/json') ||
|
|
contentType.contains('+json') ||
|
|
res.body.trimLeft().startsWith('{') ||
|
|
res.body.trimLeft().startsWith('[');
|
|
|
|
if (!shouldTryJson) return res.body;
|
|
|
|
try {
|
|
return jsonDecode(res.body);
|
|
} catch (_) {
|
|
// Avoid turning a server-side error body into a client-side crash.
|
|
return res.body;
|
|
}
|
|
}
|
|
|
|
String _extractErrorMessage(dynamic body) {
|
|
if (body == null) return 'No response body';
|
|
if (body is String) return body;
|
|
if (body is Map<String, dynamic>) {
|
|
for (final key in ['message', 'error', 'detail', 'msg']) {
|
|
final val = body[key];
|
|
if (val is String && val.trim().isNotEmpty) return val;
|
|
}
|
|
return body.toString();
|
|
}
|
|
if (body is List) {
|
|
final parts = body
|
|
.map((e) => e is Map
|
|
? _extractErrorMessage(Map<String, dynamic>.from(e))
|
|
: e.toString())
|
|
.where((e) => e.trim().isNotEmpty)
|
|
.toList();
|
|
if (parts.isNotEmpty) return parts.join('; ');
|
|
}
|
|
return body.toString();
|
|
}
|
|
}
|
|
|
|
class ApiException implements Exception {
|
|
final int statusCode;
|
|
final String message;
|
|
final dynamic body;
|
|
|
|
ApiException({
|
|
required this.statusCode,
|
|
required this.message,
|
|
this.body,
|
|
});
|
|
|
|
@override
|
|
String toString() => 'API error $statusCode: $message';
|
|
}
|