2024-05-29 19:29:45 +00:00
|
|
|
import 'dart:async';
|
|
|
|
import 'dart:io';
|
2024-05-31 22:37:25 +00:00
|
|
|
import 'dart:isolate';
|
2024-05-29 19:29:45 +00:00
|
|
|
|
|
|
|
import 'package:flutter/foundation.dart';
|
2024-05-30 21:10:56 +00:00
|
|
|
import 'package:flutter_libsparkmobile/flutter_libsparkmobile.dart';
|
2024-05-31 22:37:25 +00:00
|
|
|
import 'package:mutex/mutex.dart';
|
2024-05-29 19:29:45 +00:00
|
|
|
import 'package:sqlite3/sqlite3.dart';
|
2024-05-31 22:37:25 +00:00
|
|
|
import 'package:uuid/uuid.dart';
|
2024-05-29 19:29:45 +00:00
|
|
|
|
|
|
|
import '../../electrumx_rpc/electrumx_client.dart';
|
2024-12-14 21:29:17 +00:00
|
|
|
import '../../models/electrumx_response/spark_models.dart';
|
2024-05-30 21:10:56 +00:00
|
|
|
import '../../utilities/extensions/extensions.dart';
|
2024-05-29 19:29:45 +00:00
|
|
|
import '../../utilities/logger.dart';
|
|
|
|
import '../../utilities/stack_file_system.dart';
|
2024-06-26 18:10:32 +00:00
|
|
|
import '../../wallets/crypto_currency/crypto_currency.dart';
|
2024-05-29 19:29:45 +00:00
|
|
|
|
2024-05-31 22:37:25 +00:00
|
|
|
part 'firo_cache_coordinator.dart';
|
|
|
|
part 'firo_cache_reader.dart';
|
|
|
|
part 'firo_cache_worker.dart';
|
2024-06-05 19:38:20 +00:00
|
|
|
part 'firo_cache_writer.dart';
|
2024-05-31 22:37:25 +00:00
|
|
|
|
2024-05-29 19:29:45 +00:00
|
|
|
/// Temporary debugging log function for this file
|
|
|
|
void _debugLog(Object? object) {
|
|
|
|
if (kDebugMode) {
|
|
|
|
Logging.instance.log(
|
|
|
|
object,
|
2024-05-30 21:17:50 +00:00
|
|
|
level: LogLevel.Debug,
|
2024-05-29 19:29:45 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
abstract class _FiroCache {
|
2024-06-05 19:38:20 +00:00
|
|
|
static const int _setCacheVersion = 1;
|
2024-06-10 17:24:05 +00:00
|
|
|
static const int _tagsCacheVersion = 2;
|
2024-06-26 18:10:32 +00:00
|
|
|
|
|
|
|
static final networks = [
|
|
|
|
CryptoCurrencyNetwork.main,
|
|
|
|
CryptoCurrencyNetwork.test,
|
|
|
|
];
|
|
|
|
|
|
|
|
static String sparkSetCacheFileName(CryptoCurrencyNetwork network) =>
|
|
|
|
network == CryptoCurrencyNetwork.main
|
|
|
|
? "spark_set_v$_setCacheVersion.sqlite3"
|
|
|
|
: "spark_set_v${_setCacheVersion}_${network.name}.sqlite3";
|
2024-12-14 21:29:17 +00:00
|
|
|
static String sparkSetMetaCacheFileName(CryptoCurrencyNetwork network) =>
|
|
|
|
network == CryptoCurrencyNetwork.main
|
|
|
|
? "spark_set_meta_v$_setCacheVersion.sqlite3"
|
|
|
|
: "spark_set_meta_v${_setCacheVersion}_${network.name}.sqlite3";
|
2024-06-26 18:10:32 +00:00
|
|
|
static String sparkUsedTagsCacheFileName(CryptoCurrencyNetwork network) =>
|
|
|
|
network == CryptoCurrencyNetwork.main
|
|
|
|
? "spark_tags_v$_tagsCacheVersion.sqlite3"
|
|
|
|
: "spark_tags_v${_tagsCacheVersion}_${network.name}.sqlite3";
|
|
|
|
|
|
|
|
static final Map<CryptoCurrencyNetwork, Database> _setCacheDB = {};
|
2024-12-14 21:29:17 +00:00
|
|
|
static final Map<CryptoCurrencyNetwork, Database> _setMetaCacheDB = {};
|
2024-06-26 18:10:32 +00:00
|
|
|
static final Map<CryptoCurrencyNetwork, Database> _usedTagsCacheDB = {};
|
|
|
|
static Database setCacheDB(CryptoCurrencyNetwork network) {
|
|
|
|
if (_setCacheDB[network] == null) {
|
2024-06-05 19:38:20 +00:00
|
|
|
throw Exception(
|
|
|
|
"FiroCache.init() must be called before accessing FiroCache.db!",
|
|
|
|
);
|
|
|
|
}
|
2024-06-26 18:10:32 +00:00
|
|
|
return _setCacheDB[network]!;
|
2024-06-05 19:38:20 +00:00
|
|
|
}
|
2024-05-29 19:29:45 +00:00
|
|
|
|
2024-12-14 21:29:17 +00:00
|
|
|
static Database setMetaCacheDB(CryptoCurrencyNetwork network) {
|
|
|
|
if (_setMetaCacheDB[network] == null) {
|
|
|
|
throw Exception(
|
|
|
|
"FiroCache.init() must be called before accessing FiroCache.db!",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return _setMetaCacheDB[network]!;
|
|
|
|
}
|
|
|
|
|
2024-06-26 18:10:32 +00:00
|
|
|
static Database usedTagsCacheDB(CryptoCurrencyNetwork network) {
|
|
|
|
if (_usedTagsCacheDB[network] == null) {
|
2024-05-29 19:29:45 +00:00
|
|
|
throw Exception(
|
|
|
|
"FiroCache.init() must be called before accessing FiroCache.db!",
|
|
|
|
);
|
|
|
|
}
|
2024-06-26 18:10:32 +00:00
|
|
|
return _usedTagsCacheDB[network]!;
|
2024-05-29 19:29:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static Future<void>? _initFuture;
|
|
|
|
static Future<void> init() => _initFuture ??= _init();
|
|
|
|
|
|
|
|
static Future<void> _init() async {
|
2024-06-05 19:38:20 +00:00
|
|
|
final sqliteDir =
|
|
|
|
await StackFileSystem.applicationFiroCacheSQLiteDirectory();
|
2024-05-29 19:29:45 +00:00
|
|
|
|
2024-06-26 18:10:32 +00:00
|
|
|
for (final network in networks) {
|
|
|
|
final sparkSetCacheFile =
|
|
|
|
File("${sqliteDir.path}/${sparkSetCacheFileName(network)}");
|
2024-05-29 19:29:45 +00:00
|
|
|
|
2024-12-14 21:29:17 +00:00
|
|
|
final sparkSetMetaCacheFile =
|
|
|
|
File("${sqliteDir.path}/${sparkSetMetaCacheFileName(network)}");
|
|
|
|
|
2024-06-26 18:10:32 +00:00
|
|
|
final sparkUsedTagsCacheFile =
|
|
|
|
File("${sqliteDir.path}/${sparkUsedTagsCacheFileName(network)}");
|
2024-05-29 19:29:45 +00:00
|
|
|
|
2024-06-26 18:10:32 +00:00
|
|
|
if (!(await sparkSetCacheFile.exists())) {
|
|
|
|
await _createSparkSetCacheDb(sparkSetCacheFile.path);
|
|
|
|
}
|
2024-12-14 21:29:17 +00:00
|
|
|
if (!(await sparkSetMetaCacheFile.exists())) {
|
|
|
|
await _createSparkSetMetaCacheDb(sparkSetMetaCacheFile.path);
|
|
|
|
}
|
2024-06-26 18:10:32 +00:00
|
|
|
if (!(await sparkUsedTagsCacheFile.exists())) {
|
|
|
|
await _createSparkUsedTagsCacheDb(sparkUsedTagsCacheFile.path);
|
|
|
|
}
|
|
|
|
|
|
|
|
_setCacheDB[network] = sqlite3.open(
|
|
|
|
sparkSetCacheFile.path,
|
|
|
|
mode: OpenMode.readWrite,
|
|
|
|
);
|
2024-12-14 21:29:17 +00:00
|
|
|
_setMetaCacheDB[network] = sqlite3.open(
|
|
|
|
sparkSetMetaCacheFile.path,
|
|
|
|
mode: OpenMode.readWrite,
|
|
|
|
);
|
2024-06-26 18:10:32 +00:00
|
|
|
_usedTagsCacheDB[network] = sqlite3.open(
|
|
|
|
sparkUsedTagsCacheFile.path,
|
|
|
|
mode: OpenMode.readWrite,
|
|
|
|
);
|
|
|
|
}
|
2024-05-29 19:29:45 +00:00
|
|
|
}
|
|
|
|
|
2024-06-26 18:10:32 +00:00
|
|
|
static Future<void> _deleteAllCache(CryptoCurrencyNetwork network) async {
|
2024-05-31 17:59:43 +00:00
|
|
|
final start = DateTime.now();
|
2024-06-26 18:10:32 +00:00
|
|
|
setCacheDB(network).execute(
|
2024-05-31 17:59:43 +00:00
|
|
|
"""
|
|
|
|
DELETE FROM SparkSet;
|
|
|
|
DELETE FROM SparkCoin;
|
|
|
|
DELETE FROM SparkSetCoins;
|
2024-06-05 19:38:20 +00:00
|
|
|
VACUUM;
|
|
|
|
""",
|
|
|
|
);
|
2024-12-14 21:29:17 +00:00
|
|
|
setMetaCacheDB(network).execute(
|
|
|
|
"""
|
|
|
|
DELETE FROM PreviousMetaFetchResult;
|
|
|
|
VACUUM;
|
|
|
|
""",
|
|
|
|
);
|
2024-06-26 18:10:32 +00:00
|
|
|
usedTagsCacheDB(network).execute(
|
2024-06-05 19:38:20 +00:00
|
|
|
"""
|
2024-05-31 17:59:43 +00:00
|
|
|
DELETE FROM SparkUsedCoinTags;
|
|
|
|
VACUUM;
|
|
|
|
""",
|
|
|
|
);
|
|
|
|
_debugLog(
|
|
|
|
"_deleteAllCache() "
|
|
|
|
"duration = ${DateTime.now().difference(start)}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-06-05 19:38:20 +00:00
|
|
|
static Future<void> _createSparkSetCacheDb(String file) async {
|
2024-05-29 19:29:45 +00:00
|
|
|
final db = sqlite3.open(
|
|
|
|
file,
|
|
|
|
mode: OpenMode.readWriteCreate,
|
|
|
|
);
|
|
|
|
|
|
|
|
db.execute(
|
|
|
|
"""
|
|
|
|
CREATE TABLE SparkSet (
|
2024-05-31 22:37:25 +00:00
|
|
|
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT UNIQUE,
|
|
|
|
blockHash TEXT NOT NULL,
|
|
|
|
setHash TEXT NOT NULL,
|
|
|
|
groupId INTEGER NOT NULL,
|
|
|
|
timestampUTC INTEGER NOT NULL,
|
|
|
|
UNIQUE (blockHash, setHash, groupId)
|
2024-05-29 19:29:45 +00:00
|
|
|
);
|
2024-05-31 22:37:25 +00:00
|
|
|
|
2024-05-29 19:29:45 +00:00
|
|
|
CREATE TABLE SparkCoin (
|
|
|
|
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT UNIQUE,
|
|
|
|
serialized TEXT NOT NULL,
|
|
|
|
txHash TEXT NOT NULL,
|
|
|
|
context TEXT NOT NULL,
|
|
|
|
UNIQUE(serialized, txHash, context)
|
|
|
|
);
|
2024-05-31 22:37:25 +00:00
|
|
|
|
2024-05-29 19:29:45 +00:00
|
|
|
CREATE TABLE SparkSetCoins (
|
2024-05-31 22:37:25 +00:00
|
|
|
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT UNIQUE,
|
|
|
|
setId INTEGER NOT NULL,
|
|
|
|
coinId INTEGER NOT NULL,
|
|
|
|
FOREIGN KEY (setId) REFERENCES SparkSet(id),
|
|
|
|
FOREIGN KEY (coinId) REFERENCES SparkCoin(id)
|
2024-05-29 19:29:45 +00:00
|
|
|
);
|
2024-06-05 19:38:20 +00:00
|
|
|
""",
|
|
|
|
);
|
|
|
|
|
|
|
|
db.dispose();
|
|
|
|
}
|
|
|
|
|
2024-12-14 21:29:17 +00:00
|
|
|
static Future<void> _createSparkSetMetaCacheDb(String file) async {
|
|
|
|
final db = sqlite3.open(
|
|
|
|
file,
|
|
|
|
mode: OpenMode.readWriteCreate,
|
|
|
|
);
|
|
|
|
|
|
|
|
db.execute(
|
|
|
|
"""
|
|
|
|
CREATE TABLE PreviousMetaFetchResult (
|
|
|
|
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT UNIQUE,
|
|
|
|
coinGroupId INTEGER NOT NULL UNIQUE,
|
|
|
|
blockHash TEXT NOT NULL,
|
|
|
|
setHash TEXT NOT NULL,
|
|
|
|
size INTEGER NOT NULL
|
|
|
|
);
|
|
|
|
""",
|
|
|
|
);
|
|
|
|
|
|
|
|
db.dispose();
|
|
|
|
}
|
|
|
|
|
2024-06-05 19:38:20 +00:00
|
|
|
static Future<void> _createSparkUsedTagsCacheDb(String file) async {
|
|
|
|
final db = sqlite3.open(
|
|
|
|
file,
|
|
|
|
mode: OpenMode.readWriteCreate,
|
|
|
|
);
|
|
|
|
|
|
|
|
db.execute(
|
|
|
|
"""
|
2024-05-30 21:10:56 +00:00
|
|
|
CREATE TABLE SparkUsedCoinTags (
|
|
|
|
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT UNIQUE,
|
2024-06-10 17:24:05 +00:00
|
|
|
tag TEXT NOT NULL UNIQUE,
|
|
|
|
txid TEXT NOT NULL
|
2024-05-30 21:10:56 +00:00
|
|
|
);
|
2024-05-29 19:29:45 +00:00
|
|
|
""",
|
|
|
|
);
|
|
|
|
|
2024-05-31 22:37:25 +00:00
|
|
|
db.dispose();
|
2024-05-29 19:29:45 +00:00
|
|
|
}
|
|
|
|
}
|