2021-02-03 19:51:07 +00:00
|
|
|
import 'dart:io';
|
2022-02-16 07:28:29 +00:00
|
|
|
|
2021-02-03 19:51:07 +00:00
|
|
|
import 'package:dio/dio.dart';
|
2021-03-25 23:30:34 +00:00
|
|
|
import 'package:selfprivacy/config/get_it_config.dart';
|
2021-02-03 19:51:07 +00:00
|
|
|
import 'package:selfprivacy/logic/api_maps/api_map.dart';
|
2022-05-14 02:54:40 +00:00
|
|
|
import 'package:selfprivacy/logic/models/hive/backblaze_credential.dart';
|
2021-12-06 18:31:19 +00:00
|
|
|
|
|
|
|
class BackblazeApiAuth {
|
|
|
|
BackblazeApiAuth({required this.authorizationToken, required this.apiUrl});
|
|
|
|
|
|
|
|
final String authorizationToken;
|
|
|
|
final String apiUrl;
|
|
|
|
}
|
|
|
|
|
|
|
|
class BackblazeApplicationKey {
|
|
|
|
BackblazeApplicationKey(
|
|
|
|
{required this.applicationKeyId, required this.applicationKey});
|
|
|
|
|
|
|
|
final String applicationKeyId;
|
|
|
|
final String applicationKey;
|
|
|
|
}
|
2021-02-03 19:51:07 +00:00
|
|
|
|
2021-03-25 23:30:34 +00:00
|
|
|
class BackblazeApi extends ApiMap {
|
2022-02-16 07:28:29 +00:00
|
|
|
BackblazeApi({this.hasLogger = false, this.isWithToken = true});
|
2021-03-25 23:30:34 +00:00
|
|
|
|
2022-05-24 18:55:39 +00:00
|
|
|
@override
|
2021-03-25 23:30:34 +00:00
|
|
|
BaseOptions get options {
|
|
|
|
var options = BaseOptions(baseUrl: rootAddress);
|
|
|
|
if (isWithToken) {
|
|
|
|
var backblazeCredential = getIt<ApiConfigModel>().backblazeCredential;
|
|
|
|
var token = backblazeCredential!.applicationKey;
|
|
|
|
options.headers = {'Authorization': 'Basic $token'};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (validateStatus != null) {
|
|
|
|
options.validateStatus = validateStatus!;
|
2021-02-03 19:51:07 +00:00
|
|
|
}
|
2021-03-25 23:30:34 +00:00
|
|
|
|
|
|
|
return options;
|
2021-02-03 19:51:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2021-03-25 23:30:34 +00:00
|
|
|
String rootAddress = 'https://api.backblazeb2.com/b2api/v2/';
|
2021-02-03 19:51:07 +00:00
|
|
|
|
2021-12-06 18:31:19 +00:00
|
|
|
String apiPrefix = '/b2api/v2';
|
|
|
|
|
|
|
|
Future<BackblazeApiAuth> getAuthorizationToken() async {
|
|
|
|
var client = await getClient();
|
|
|
|
var backblazeCredential = getIt<ApiConfigModel>().backblazeCredential;
|
|
|
|
if (backblazeCredential == null) {
|
|
|
|
throw Exception('Backblaze credential is null');
|
|
|
|
}
|
|
|
|
final String encodedApiKey = encodedBackblazeKey(
|
|
|
|
backblazeCredential.keyId, backblazeCredential.applicationKey);
|
|
|
|
var response = await client.get(
|
|
|
|
'b2_authorize_account',
|
|
|
|
options: Options(headers: {'Authorization': 'Basic $encodedApiKey'}),
|
|
|
|
);
|
|
|
|
if (response.statusCode != 200) {
|
|
|
|
throw Exception('code: ${response.statusCode}');
|
|
|
|
}
|
|
|
|
return BackblazeApiAuth(
|
|
|
|
authorizationToken: response.data['authorizationToken'],
|
|
|
|
apiUrl: response.data['apiUrl'],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-03-25 23:30:34 +00:00
|
|
|
Future<bool> isValid(String encodedApiKey) async {
|
|
|
|
var client = await getClient();
|
|
|
|
Response response = await client.get(
|
|
|
|
'b2_authorize_account',
|
|
|
|
options: Options(headers: {'Authorization': 'Basic $encodedApiKey'}),
|
|
|
|
);
|
2021-03-26 13:38:39 +00:00
|
|
|
close(client);
|
2021-02-03 19:51:07 +00:00
|
|
|
if (response.statusCode == HttpStatus.ok) {
|
2021-12-06 18:31:19 +00:00
|
|
|
if (response.data['allowed']['capabilities'].contains('listBuckets')) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2021-02-03 19:51:07 +00:00
|
|
|
} else if (response.statusCode == HttpStatus.unauthorized) {
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
throw Exception('code: ${response.statusCode}');
|
|
|
|
}
|
|
|
|
}
|
2021-03-25 23:30:34 +00:00
|
|
|
|
2021-12-06 18:31:19 +00:00
|
|
|
// Create bucket
|
|
|
|
Future<String> createBucket(String bucketName) async {
|
|
|
|
final auth = await getAuthorizationToken();
|
|
|
|
var backblazeCredential = getIt<ApiConfigModel>().backblazeCredential;
|
|
|
|
var client = await getClient();
|
|
|
|
client.options.baseUrl = auth.apiUrl;
|
|
|
|
var response = await client.post(
|
|
|
|
'$apiPrefix/b2_create_bucket',
|
|
|
|
data: {
|
|
|
|
'accountId': backblazeCredential!.keyId,
|
|
|
|
'bucketName': bucketName,
|
|
|
|
'bucketType': 'allPrivate',
|
|
|
|
'lifecycleRules': [
|
|
|
|
{
|
2022-05-24 18:55:39 +00:00
|
|
|
'daysFromHidingToDeleting': 30,
|
|
|
|
'daysFromUploadingToHiding': null,
|
|
|
|
'fileNamePrefix': ''
|
2021-12-06 18:31:19 +00:00
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
options: Options(
|
|
|
|
headers: {'Authorization': auth.authorizationToken},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
close(client);
|
|
|
|
if (response.statusCode == HttpStatus.ok) {
|
|
|
|
return response.data['bucketId'];
|
|
|
|
} else {
|
|
|
|
throw Exception('code: ${response.statusCode}');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a limited capability key with access to the given bucket
|
|
|
|
Future<BackblazeApplicationKey> createKey(String bucketId) async {
|
|
|
|
final auth = await getAuthorizationToken();
|
|
|
|
var client = await getClient();
|
|
|
|
client.options.baseUrl = auth.apiUrl;
|
|
|
|
var response = await client.post(
|
|
|
|
'$apiPrefix/b2_create_key',
|
|
|
|
data: {
|
|
|
|
'accountId': getIt<ApiConfigModel>().backblazeCredential!.keyId,
|
|
|
|
'bucketId': bucketId,
|
|
|
|
'capabilities': ['listBuckets', 'listFiles', 'readFiles', 'writeFiles'],
|
|
|
|
'keyName': 'selfprivacy-restricted-server-key',
|
|
|
|
},
|
|
|
|
options: Options(
|
|
|
|
headers: {'Authorization': auth.authorizationToken},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
close(client);
|
|
|
|
if (response.statusCode == HttpStatus.ok) {
|
|
|
|
return BackblazeApplicationKey(
|
|
|
|
applicationKeyId: response.data['applicationKeyId'],
|
|
|
|
applicationKey: response.data['applicationKey']);
|
|
|
|
} else {
|
|
|
|
throw Exception('code: ${response.statusCode}');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-25 23:30:34 +00:00
|
|
|
@override
|
2022-02-16 07:28:29 +00:00
|
|
|
bool hasLogger;
|
2021-03-25 23:30:34 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
bool isWithToken;
|
|
|
|
}
|