2023-03-18 15:55:11 +01:00
|
|
|
import 'package:collection/collection.dart';
|
2023-03-03 23:38:30 +01:00
|
|
|
import 'package:immich_mobile/shared/models/user.dart';
|
2023-02-09 18:32:08 +01:00
|
|
|
import 'package:isar/isar.dart';
|
|
|
|
import 'dart:convert';
|
|
|
|
|
|
|
|
part 'store.g.dart';
|
|
|
|
|
|
|
|
/// Key-value store for individual items enumerated in StoreKey.
|
|
|
|
/// Supports String, int and JSON-serializable Objects
|
|
|
|
/// Can be used concurrently from multiple isolates
|
|
|
|
class Store {
|
|
|
|
static late final Isar _db;
|
2023-03-18 15:55:11 +01:00
|
|
|
static final List<dynamic> _cache =
|
|
|
|
List.filled(StoreKey.values.map((e) => e.id).max + 1, null);
|
2023-02-09 18:32:08 +01:00
|
|
|
|
|
|
|
/// Initializes the store (call exactly once per app start)
|
|
|
|
static void init(Isar db) {
|
|
|
|
_db = db;
|
|
|
|
_populateCache();
|
|
|
|
_db.storeValues.where().build().watch().listen(_onChangeListener);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// clears all values from this store (cache and DB), only for testing!
|
|
|
|
static Future<void> clear() {
|
|
|
|
_cache.fillRange(0, _cache.length, null);
|
|
|
|
return _db.writeTxn(() => _db.storeValues.clear());
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the stored value for the given key, or the default value if null
|
|
|
|
static T? get<T>(StoreKey key, [T? defaultValue]) =>
|
2023-03-03 23:38:30 +01:00
|
|
|
_cache[key.id] ?? defaultValue;
|
2023-02-09 18:32:08 +01:00
|
|
|
|
|
|
|
/// Stores the value synchronously in the cache and asynchronously in the DB
|
|
|
|
static Future<void> put<T>(StoreKey key, T value) {
|
2023-03-03 23:38:30 +01:00
|
|
|
_cache[key.id] = value;
|
|
|
|
return _db.writeTxn(
|
|
|
|
() async => _db.storeValues.put(await StoreValue._of(value, key)),
|
|
|
|
);
|
2023-02-09 18:32:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Removes the value synchronously from the cache and asynchronously from the DB
|
|
|
|
static Future<void> delete(StoreKey key) {
|
2023-03-03 23:38:30 +01:00
|
|
|
_cache[key.id] = null;
|
|
|
|
return _db.writeTxn(() => _db.storeValues.delete(key.id));
|
2023-02-09 18:32:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Fills the cache with the values from the DB
|
|
|
|
static _populateCache() {
|
|
|
|
for (StoreKey key in StoreKey.values) {
|
2023-03-03 23:38:30 +01:00
|
|
|
final StoreValue? value = _db.storeValues.getSync(key.id);
|
2023-02-09 18:32:08 +01:00
|
|
|
if (value != null) {
|
2023-03-03 23:38:30 +01:00
|
|
|
_cache[key.id] = value._extract(key);
|
2023-02-09 18:32:08 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// updates the state if a value is updated in any isolate
|
|
|
|
static void _onChangeListener(List<StoreValue>? data) {
|
|
|
|
if (data != null) {
|
|
|
|
for (StoreValue value in data) {
|
|
|
|
_cache[value.id] = value._extract(StoreKey.values[value.id]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Internal class for `Store`, do not use elsewhere.
|
|
|
|
@Collection(inheritance: false)
|
|
|
|
class StoreValue {
|
|
|
|
StoreValue(this.id, {this.intValue, this.strValue});
|
|
|
|
Id id;
|
|
|
|
int? intValue;
|
|
|
|
String? strValue;
|
|
|
|
|
2023-03-18 15:55:11 +01:00
|
|
|
dynamic _extract(StoreKey key) {
|
|
|
|
switch (key.type) {
|
|
|
|
case int:
|
|
|
|
return key.fromDb == null
|
|
|
|
? intValue
|
|
|
|
: key.fromDb!.call(Store._db, intValue!);
|
|
|
|
case bool:
|
|
|
|
return intValue == null ? null : intValue! == 1;
|
|
|
|
case DateTime:
|
|
|
|
return intValue == null
|
2023-02-09 18:32:08 +01:00
|
|
|
? null
|
2023-03-18 15:55:11 +01:00
|
|
|
: DateTime.fromMicrosecondsSinceEpoch(intValue!);
|
|
|
|
case String:
|
|
|
|
return key.fromJson != null
|
|
|
|
? key.fromJson!.call(json.decode(strValue!))
|
|
|
|
: strValue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<StoreValue> _of(dynamic value, StoreKey key) async {
|
|
|
|
int? i;
|
|
|
|
String? s;
|
|
|
|
switch (key.type) {
|
|
|
|
case int:
|
|
|
|
i = (key.toDb == null ? value : await key.toDb!.call(Store._db, value));
|
|
|
|
break;
|
|
|
|
case bool:
|
|
|
|
i = value == null ? null : (value ? 1 : 0);
|
|
|
|
break;
|
|
|
|
case DateTime:
|
|
|
|
i = value == null ? null : (value as DateTime).microsecondsSinceEpoch;
|
|
|
|
break;
|
|
|
|
case String:
|
|
|
|
s = key.fromJson == null ? value : json.encode(value.toJson());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return StoreValue(key.id, intValue: i, strValue: s);
|
|
|
|
}
|
2023-02-09 18:32:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Key for each possible value in the `Store`.
|
|
|
|
/// Defines the data type (int, String, JSON) for each value
|
|
|
|
enum StoreKey {
|
|
|
|
userRemoteId(0),
|
|
|
|
assetETag(1),
|
2023-03-18 15:55:11 +01:00
|
|
|
currentUser(2, type: int, fromDb: _getUser, toDb: _toUser),
|
|
|
|
deviceIdHash(3, type: int),
|
2023-03-03 23:38:30 +01:00
|
|
|
deviceId(4),
|
2023-03-18 15:55:11 +01:00
|
|
|
backupFailedSince(5, type: DateTime),
|
|
|
|
backupRequireWifi(6, type: bool),
|
|
|
|
backupRequireCharging(7, type: bool),
|
|
|
|
backupTriggerDelay(8, type: int);
|
2023-02-09 18:32:08 +01:00
|
|
|
|
2023-03-03 23:38:30 +01:00
|
|
|
const StoreKey(
|
|
|
|
this.id, {
|
2023-03-18 15:55:11 +01:00
|
|
|
this.type = String,
|
2023-03-03 23:38:30 +01:00
|
|
|
this.fromDb,
|
|
|
|
this.toDb,
|
|
|
|
// ignore: unused_element
|
|
|
|
this.fromJson,
|
|
|
|
});
|
|
|
|
final int id;
|
2023-03-18 15:55:11 +01:00
|
|
|
final Type type;
|
2023-03-03 23:38:30 +01:00
|
|
|
final dynamic Function(Isar, int)? fromDb;
|
|
|
|
final Future<int> Function(Isar, dynamic)? toDb;
|
|
|
|
final Function(dynamic)? fromJson;
|
|
|
|
}
|
|
|
|
|
|
|
|
User? _getUser(Isar db, int i) => db.users.getSync(i);
|
|
|
|
Future<int> _toUser(Isar db, dynamic u) {
|
|
|
|
User user = (u as User);
|
|
|
|
return db.users.put(user);
|
2023-02-09 18:32:08 +01:00
|
|
|
}
|