2021-11-17 12:20:15 +01:00
|
|
|
import 'dart:io';
|
|
|
|
import 'dart:typed_data';
|
|
|
|
|
|
|
|
import 'package:flutter/foundation.dart' hide Key;
|
|
|
|
import 'package:flutter/services.dart';
|
|
|
|
|
|
|
|
import 'package:encrypt/encrypt.dart';
|
|
|
|
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
|
|
|
|
import 'package:matrix/matrix.dart';
|
|
|
|
import 'package:path_provider/path_provider.dart';
|
2021-11-17 19:17:40 +01:00
|
|
|
import 'package:sqflite/sqflite.dart' as sqflite;
|
2021-11-17 12:20:15 +01:00
|
|
|
|
|
|
|
import '../platform_infos.dart';
|
|
|
|
|
2021-11-17 19:17:40 +01:00
|
|
|
class FlutterFluffyBoxDatabase extends FluffyBoxDatabase {
|
|
|
|
FlutterFluffyBoxDatabase(
|
2021-11-17 12:20:15 +01:00
|
|
|
String name, {
|
|
|
|
String path,
|
2021-11-17 19:17:40 +01:00
|
|
|
Future<sqflite.Database> Function() openSqlDatabase,
|
2021-11-17 12:20:15 +01:00
|
|
|
}) : super(
|
|
|
|
name,
|
2021-11-17 19:17:40 +01:00
|
|
|
openSqlDatabase: openSqlDatabase,
|
2021-11-17 12:20:15 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
static const String _cipherStorageKey = 'database_encryption_key';
|
|
|
|
static const int _cipherStorageKeyLength = 512;
|
|
|
|
|
|
|
|
static Future<FluffyBoxDatabase> databaseBuilder(Client client) async {
|
2021-11-17 12:48:45 +01:00
|
|
|
Logs().d('Open FluffyBox...');
|
2021-11-17 12:20:15 +01:00
|
|
|
try {
|
|
|
|
// Workaround for secure storage is calling Platform.operatingSystem on web
|
|
|
|
if (kIsWeb) throw MissingPluginException();
|
|
|
|
|
|
|
|
const secureStorage = FlutterSecureStorage();
|
|
|
|
final containsEncryptionKey =
|
|
|
|
await secureStorage.containsKey(key: _cipherStorageKey);
|
|
|
|
if (!containsEncryptionKey) {
|
|
|
|
final key = SecureRandom(_cipherStorageKeyLength).base64;
|
|
|
|
await secureStorage.write(
|
|
|
|
key: _cipherStorageKey,
|
|
|
|
value: key,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// workaround for if we just wrote to the key and it still doesn't exist
|
|
|
|
final rawEncryptionKey = await secureStorage.read(key: _cipherStorageKey);
|
|
|
|
if (rawEncryptionKey == null) throw MissingPluginException();
|
|
|
|
} on MissingPluginException catch (_) {
|
2021-11-17 12:48:45 +01:00
|
|
|
Logs().i('FluffyBox encryption is not supported on this platform');
|
2021-11-17 12:20:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
final db = FluffyBoxDatabase(
|
2021-11-17 19:17:40 +01:00
|
|
|
'fluffybox_${client.clientName.replaceAll(' ', '_').toLowerCase()}',
|
|
|
|
openSqlDatabase: () => _openSqlDatabase(client),
|
2021-11-17 12:20:15 +01:00
|
|
|
);
|
|
|
|
await db.open();
|
2021-11-17 12:48:45 +01:00
|
|
|
Logs().d('FluffyBox is ready');
|
2021-11-17 12:20:15 +01:00
|
|
|
return db;
|
|
|
|
}
|
|
|
|
|
2021-11-17 19:17:40 +01:00
|
|
|
static Future<sqflite.Database> _openSqlDatabase(Client client) async {
|
|
|
|
final path = await _findDatabasePath(client);
|
|
|
|
return await sqflite.openDatabase(path);
|
2021-11-17 12:20:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static Future<String> _findDatabasePath(Client client) async {
|
|
|
|
String path = client.clientName;
|
|
|
|
if (!kIsWeb) {
|
|
|
|
Directory directory;
|
|
|
|
try {
|
|
|
|
directory = await getApplicationSupportDirectory();
|
|
|
|
} catch (_) {
|
|
|
|
try {
|
|
|
|
directory = await getLibraryDirectory();
|
|
|
|
} catch (_) {
|
|
|
|
directory = Directory.current;
|
|
|
|
}
|
|
|
|
}
|
2021-11-17 19:17:40 +01:00
|
|
|
path =
|
|
|
|
'${directory.path}${client.clientName.replaceAll(' ', '-')}.sqflite';
|
2021-11-17 12:20:15 +01:00
|
|
|
}
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
int get maxFileSize => supportsFileStoring ? 100 * 1024 * 1024 : 0;
|
|
|
|
@override
|
|
|
|
bool get supportsFileStoring => (PlatformInfos.isIOS ||
|
|
|
|
PlatformInfos.isAndroid ||
|
|
|
|
PlatformInfos.isDesktop);
|
|
|
|
|
|
|
|
Future<String> _getFileStoreDirectory() async {
|
|
|
|
try {
|
|
|
|
try {
|
|
|
|
return (await getApplicationSupportDirectory()).path;
|
|
|
|
} catch (_) {
|
|
|
|
return (await getApplicationDocumentsDirectory()).path;
|
|
|
|
}
|
|
|
|
} catch (_) {
|
|
|
|
return (await getDownloadsDirectory()).path;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<Uint8List> getFile(Uri mxcUri) async {
|
|
|
|
if (!supportsFileStoring) return null;
|
|
|
|
final tempDirectory = await _getFileStoreDirectory();
|
|
|
|
final file =
|
|
|
|
File('$tempDirectory/${Uri.encodeComponent(mxcUri.toString())}');
|
|
|
|
if (await file.exists() == false) return null;
|
|
|
|
final bytes = await file.readAsBytes();
|
|
|
|
return bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future storeFile(Uri mxcUri, Uint8List bytes, int time) async {
|
|
|
|
if (!supportsFileStoring) return null;
|
|
|
|
final tempDirectory = await _getFileStoreDirectory();
|
|
|
|
final file =
|
|
|
|
File('$tempDirectory/${Uri.encodeComponent(mxcUri.toString())}');
|
|
|
|
if (await file.exists()) return;
|
|
|
|
await file.writeAsBytes(bytes);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|