2019-11-02 03:16:47 +00:00
|
|
|
import 'dart:async';
|
|
|
|
|
2019-11-10 13:11:30 +00:00
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
|
2019-10-31 02:20:07 +00:00
|
|
|
import 'types.dart';
|
|
|
|
import 'package:flutter/services.dart';
|
|
|
|
|
2020-05-11 00:48:41 +00:00
|
|
|
///Class that implements a singleton object (shared instance) which manages the shared HTTP auth credentials cache.
|
2019-11-10 13:11:30 +00:00
|
|
|
///On iOS, this class uses the [URLCredentialStorage](https://developer.apple.com/documentation/foundation/urlcredentialstorage) class.
|
2020-05-29 12:51:26 +00:00
|
|
|
///On Android, this class has a custom implementation using `android.database.sqlite.SQLiteDatabase` because
|
|
|
|
///[WebViewDatabase](https://developer.android.com/reference/android/webkit/WebViewDatabase)
|
2019-11-10 13:11:30 +00:00
|
|
|
///doesn't offer the same functionalities as iOS `URLCredentialStorage`.
|
2019-10-31 02:20:07 +00:00
|
|
|
class HttpAuthCredentialDatabase {
|
|
|
|
static HttpAuthCredentialDatabase _instance;
|
2019-12-01 11:55:06 +00:00
|
|
|
static const MethodChannel _channel = const MethodChannel(
|
|
|
|
'com.pichillilorenzo/flutter_inappwebview_credential_database');
|
2019-10-31 02:20:07 +00:00
|
|
|
|
2019-11-10 13:11:30 +00:00
|
|
|
///Gets the database shared instance.
|
2019-10-31 02:20:07 +00:00
|
|
|
static HttpAuthCredentialDatabase instance() {
|
|
|
|
return (_instance != null) ? _instance : _init();
|
|
|
|
}
|
|
|
|
|
|
|
|
static HttpAuthCredentialDatabase _init() {
|
|
|
|
_channel.setMethodCallHandler(_handleMethod);
|
2020-05-29 17:56:03 +00:00
|
|
|
_instance = HttpAuthCredentialDatabase();
|
2019-10-31 02:20:07 +00:00
|
|
|
return _instance;
|
|
|
|
}
|
|
|
|
|
2019-12-01 11:55:06 +00:00
|
|
|
static Future<dynamic> _handleMethod(MethodCall call) async {}
|
2019-10-31 02:20:07 +00:00
|
|
|
|
2019-11-10 13:11:30 +00:00
|
|
|
///Gets a map list of all HTTP auth credentials saved.
|
|
|
|
///Each map contains the key `protectionSpace` of type [ProtectionSpace]
|
|
|
|
///and the key `credentials` of type `List<HttpAuthCredential>` that contains all the HTTP auth credentials saved for that `protectionSpace`.
|
2020-06-21 22:09:35 +00:00
|
|
|
Future<List<ProtectionSpaceHttpAuthCredentials>>
|
|
|
|
getAllAuthCredentials() async {
|
2019-10-31 02:20:07 +00:00
|
|
|
Map<String, dynamic> args = <String, dynamic>{};
|
2019-12-01 11:55:06 +00:00
|
|
|
List<dynamic> allCredentials =
|
|
|
|
await _channel.invokeMethod('getAllAuthCredentials', args);
|
2020-06-20 19:58:29 +00:00
|
|
|
|
|
|
|
List<ProtectionSpaceHttpAuthCredentials> result = [];
|
|
|
|
|
2019-10-31 02:20:07 +00:00
|
|
|
for (Map<dynamic, dynamic> map in allCredentials) {
|
|
|
|
Map<dynamic, dynamic> protectionSpace = map["protectionSpace"];
|
|
|
|
List<dynamic> credentials = map["credentials"];
|
2020-06-21 22:09:35 +00:00
|
|
|
result.add(ProtectionSpaceHttpAuthCredentials(
|
|
|
|
protectionSpace: ProtectionSpace(
|
|
|
|
host: protectionSpace["host"],
|
|
|
|
protocol: protectionSpace["protocol"],
|
|
|
|
realm: protectionSpace["realm"],
|
|
|
|
port: protectionSpace["port"]),
|
|
|
|
credentials: credentials
|
|
|
|
.map((credential) => HttpAuthCredential(
|
2020-06-20 19:58:29 +00:00
|
|
|
username: credential["username"],
|
|
|
|
password: credential["password"]))
|
2020-06-21 22:09:35 +00:00
|
|
|
.toList()));
|
2019-10-31 02:20:07 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2019-11-10 13:11:30 +00:00
|
|
|
///Gets all the HTTP auth credentials saved for that [protectionSpace].
|
2019-12-01 11:55:06 +00:00
|
|
|
Future<List<HttpAuthCredential>> getHttpAuthCredentials(
|
|
|
|
{@required ProtectionSpace protectionSpace}) async {
|
2019-10-31 02:20:07 +00:00
|
|
|
Map<String, dynamic> args = <String, dynamic>{};
|
|
|
|
args.putIfAbsent("host", () => protectionSpace.host);
|
|
|
|
args.putIfAbsent("protocol", () => protectionSpace.protocol);
|
|
|
|
args.putIfAbsent("realm", () => protectionSpace.realm);
|
|
|
|
args.putIfAbsent("port", () => protectionSpace.port);
|
2019-12-01 11:55:06 +00:00
|
|
|
List<dynamic> credentialList =
|
|
|
|
await _channel.invokeMethod('getHttpAuthCredentials', args);
|
2019-10-31 02:20:07 +00:00
|
|
|
List<HttpAuthCredential> credentials = [];
|
|
|
|
for (Map<dynamic, dynamic> credential in credentialList) {
|
2019-12-01 11:55:06 +00:00
|
|
|
credentials.add(HttpAuthCredential(
|
|
|
|
username: credential["username"], password: credential["password"]));
|
2019-10-31 02:20:07 +00:00
|
|
|
}
|
|
|
|
return credentials;
|
|
|
|
}
|
|
|
|
|
2019-11-10 13:11:30 +00:00
|
|
|
///Saves an HTTP auth [credential] for that [protectionSpace].
|
2019-12-01 11:55:06 +00:00
|
|
|
Future<void> setHttpAuthCredential(
|
|
|
|
{@required ProtectionSpace protectionSpace,
|
|
|
|
@required HttpAuthCredential credential}) async {
|
2019-10-31 02:20:07 +00:00
|
|
|
Map<String, dynamic> args = <String, dynamic>{};
|
|
|
|
args.putIfAbsent("host", () => protectionSpace.host);
|
|
|
|
args.putIfAbsent("protocol", () => protectionSpace.protocol);
|
|
|
|
args.putIfAbsent("realm", () => protectionSpace.realm);
|
|
|
|
args.putIfAbsent("port", () => protectionSpace.port);
|
|
|
|
args.putIfAbsent("username", () => credential.username);
|
|
|
|
args.putIfAbsent("password", () => credential.password);
|
|
|
|
await _channel.invokeMethod('setHttpAuthCredential', args);
|
|
|
|
}
|
|
|
|
|
2019-11-10 13:11:30 +00:00
|
|
|
///Removes an HTTP auth [credential] for that [protectionSpace].
|
2019-12-01 11:55:06 +00:00
|
|
|
Future<void> removeHttpAuthCredential(
|
|
|
|
{@required ProtectionSpace protectionSpace,
|
|
|
|
@required HttpAuthCredential credential}) async {
|
2019-10-31 02:20:07 +00:00
|
|
|
Map<String, dynamic> args = <String, dynamic>{};
|
|
|
|
args.putIfAbsent("host", () => protectionSpace.host);
|
|
|
|
args.putIfAbsent("protocol", () => protectionSpace.protocol);
|
|
|
|
args.putIfAbsent("realm", () => protectionSpace.realm);
|
|
|
|
args.putIfAbsent("port", () => protectionSpace.port);
|
|
|
|
args.putIfAbsent("username", () => credential.username);
|
|
|
|
args.putIfAbsent("password", () => credential.password);
|
|
|
|
await _channel.invokeMethod('removeHttpAuthCredential', args);
|
|
|
|
}
|
|
|
|
|
2019-11-10 13:11:30 +00:00
|
|
|
///Removes all the HTTP auth credentials saved for that [protectionSpace].
|
2019-12-01 11:55:06 +00:00
|
|
|
Future<void> removeHttpAuthCredentials(
|
|
|
|
{@required ProtectionSpace protectionSpace}) async {
|
2019-10-31 02:20:07 +00:00
|
|
|
Map<String, dynamic> args = <String, dynamic>{};
|
|
|
|
args.putIfAbsent("host", () => protectionSpace.host);
|
|
|
|
args.putIfAbsent("protocol", () => protectionSpace.protocol);
|
|
|
|
args.putIfAbsent("realm", () => protectionSpace.realm);
|
|
|
|
args.putIfAbsent("port", () => protectionSpace.port);
|
|
|
|
await _channel.invokeMethod('removeHttpAuthCredentials', args);
|
|
|
|
}
|
|
|
|
|
2019-11-10 13:11:30 +00:00
|
|
|
///Removes all the HTTP auth credentials saved in the database.
|
2019-10-31 02:20:07 +00:00
|
|
|
Future<void> clearAllAuthCredentials() async {
|
|
|
|
Map<String, dynamic> args = <String, dynamic>{};
|
|
|
|
await _channel.invokeMethod('clearAllAuthCredentials', args);
|
|
|
|
}
|
2019-12-01 11:55:06 +00:00
|
|
|
}
|