mirror of
https://github.com/KRTirtho/spotube.git
synced 2025-09-12 23:45:18 +00:00
230 lines
7.7 KiB
Dart
230 lines
7.7 KiB
Dart
library database;
|
|
|
|
import 'dart:convert';
|
|
import 'dart:io';
|
|
|
|
import 'package:drift/drift.dart';
|
|
import 'package:drift/remote.dart';
|
|
import 'package:encrypt/encrypt.dart';
|
|
import 'package:media_kit/media_kit.dart' hide Track;
|
|
import 'package:path/path.dart';
|
|
import 'package:path_provider/path_provider.dart';
|
|
import 'package:shadcn_flutter/shadcn_flutter.dart' show ThemeMode, Colors;
|
|
import 'package:spotube/models/database/database.steps.dart';
|
|
import 'package:spotube/models/lyrics.dart';
|
|
import 'package:spotube/models/metadata/market.dart';
|
|
import 'package:spotube/models/metadata/metadata.dart';
|
|
import 'package:spotube/services/kv_store/encrypted_kv_store.dart';
|
|
import 'package:spotube/services/kv_store/kv_store.dart';
|
|
import 'package:spotube/services/sourced_track/enums.dart';
|
|
import 'package:flutter/widgets.dart' hide Table, Key, View;
|
|
import 'package:spotube/modules/settings/color_scheme_picker_dialog.dart';
|
|
import 'package:drift/native.dart';
|
|
import 'package:spotube/services/youtube_engine/newpipe_engine.dart';
|
|
import 'package:spotube/services/youtube_engine/youtube_explode_engine.dart';
|
|
import 'package:spotube/services/youtube_engine/yt_dlp_engine.dart';
|
|
import 'package:sqlite3/sqlite3.dart';
|
|
import 'package:sqlite3_flutter_libs/sqlite3_flutter_libs.dart';
|
|
|
|
part 'database.g.dart';
|
|
|
|
part 'tables/authentication.dart';
|
|
part 'tables/blacklist.dart';
|
|
part 'tables/preferences.dart';
|
|
part 'tables/scrobbler.dart';
|
|
part 'tables/skip_segment.dart';
|
|
part 'tables/source_match.dart';
|
|
part 'tables/audio_player_state.dart';
|
|
part 'tables/history.dart';
|
|
part 'tables/lyrics.dart';
|
|
part 'tables/metadata_plugins.dart';
|
|
|
|
part 'typeconverters/color.dart';
|
|
part 'typeconverters/locale.dart';
|
|
part 'typeconverters/string_list.dart';
|
|
part 'typeconverters/encrypted_text.dart';
|
|
part 'typeconverters/map.dart';
|
|
part 'typeconverters/map_list.dart';
|
|
part 'typeconverters/subtitle.dart';
|
|
|
|
@DriftDatabase(
|
|
tables: [
|
|
AuthenticationTable,
|
|
BlacklistTable,
|
|
PreferencesTable,
|
|
ScrobblerTable,
|
|
SkipSegmentTable,
|
|
SourceMatchTable,
|
|
AudioPlayerStateTable,
|
|
HistoryTable,
|
|
LyricsTable,
|
|
MetadataPluginsTable,
|
|
],
|
|
)
|
|
class AppDatabase extends _$AppDatabase {
|
|
AppDatabase() : super(_openConnection());
|
|
|
|
@override
|
|
int get schemaVersion => 8;
|
|
|
|
@override
|
|
MigrationStrategy get migration {
|
|
return MigrationStrategy(
|
|
onUpgrade: stepByStep(
|
|
from1To2: (m, schema) async {
|
|
// Add invidiousInstance column to preferences table
|
|
await m.addColumn(
|
|
schema.preferencesTable,
|
|
schema.preferencesTable.invidiousInstance,
|
|
);
|
|
},
|
|
from2To3: (m, schema) async {
|
|
await m.addColumn(
|
|
schema.preferencesTable,
|
|
schema.preferencesTable.cacheMusic,
|
|
);
|
|
},
|
|
from3To4: (m, schema) async {
|
|
await m.addColumn(
|
|
schema.preferencesTable,
|
|
schema.preferencesTable.youtubeClientEngine,
|
|
);
|
|
},
|
|
from4To5: (m, schema) async {
|
|
final columnName = schema.preferencesTable.accentColorScheme
|
|
.escapedNameFor(SqlDialect.sqlite);
|
|
final columnNameOld =
|
|
'"${schema.preferencesTable.accentColorScheme.name}_old"';
|
|
final tableName = schema.preferencesTable.actualTableName;
|
|
await customStatement(
|
|
"ALTER TABLE $tableName "
|
|
"RENAME COLUMN $columnName to $columnNameOld",
|
|
);
|
|
await customStatement(
|
|
"ALTER TABLE $tableName "
|
|
"ADD COLUMN $columnName TEXT NOT NULL DEFAULT 'Orange:0xFFf97315'",
|
|
);
|
|
await customStatement(
|
|
"UPDATE $tableName "
|
|
"SET $columnName = $columnNameOld",
|
|
);
|
|
await customStatement(
|
|
"ALTER TABLE $tableName "
|
|
"DROP COLUMN $columnNameOld",
|
|
);
|
|
await customStatement(
|
|
"UPDATE $tableName "
|
|
"SET $columnName = 'Orange:0xFFf97315' WHERE $columnName = 'Blue:0xFF2196F3'",
|
|
);
|
|
},
|
|
from5To6: (m, schema) async {
|
|
try {
|
|
await m.addColumn(
|
|
schema.preferencesTable,
|
|
schema.preferencesTable.connectPort,
|
|
);
|
|
} on DriftRemoteException catch (e) {
|
|
// If the column already exists, ignore the error
|
|
if (e.remoteCause !=
|
|
'duplicate column name: ${schema.preferencesTable.connectPort.name}') {
|
|
rethrow;
|
|
}
|
|
}
|
|
},
|
|
from6To7: (m, schema) async {
|
|
await m.createTable(schema.metadataPluginsTable);
|
|
await m.addColumn(
|
|
schema.audioPlayerStateTable,
|
|
schema.audioPlayerStateTable.currentIndex,
|
|
);
|
|
await m.addColumn(
|
|
schema.audioPlayerStateTable,
|
|
schema.audioPlayerStateTable.tracks,
|
|
);
|
|
},
|
|
from7To8: (m, schema) async {
|
|
await m
|
|
.addColumn(
|
|
schema.metadataPluginsTable,
|
|
schema.metadataPluginsTable.entryPoint,
|
|
)
|
|
.catchError((error, stackTrace) {
|
|
// If the column already exists, ignore the error
|
|
if (!error.toString().contains('duplicate column name')) {
|
|
throw error;
|
|
}
|
|
});
|
|
await m
|
|
.addColumn(
|
|
schema.metadataPluginsTable,
|
|
schema.metadataPluginsTable.apis,
|
|
)
|
|
.catchError((error, stackTrace) {
|
|
// If the column already exists, ignore the error
|
|
if (!error.toString().contains('duplicate column name')) {
|
|
throw error;
|
|
}
|
|
});
|
|
await m
|
|
.addColumn(
|
|
schema.metadataPluginsTable,
|
|
schema.metadataPluginsTable.abilities,
|
|
)
|
|
.catchError((error, stackTrace) {
|
|
// If the column already exists, ignore the error
|
|
if (!error.toString().contains('duplicate column name')) {
|
|
throw error;
|
|
}
|
|
});
|
|
await m
|
|
.addColumn(
|
|
schema.metadataPluginsTable,
|
|
schema.metadataPluginsTable.repository,
|
|
)
|
|
.catchError((error, stackTrace) {
|
|
// If the column already exists, ignore the error
|
|
if (!error.toString().contains('duplicate column name')) {
|
|
throw error;
|
|
}
|
|
});
|
|
await m
|
|
.addColumn(
|
|
schema.metadataPluginsTable,
|
|
schema.metadataPluginsTable.pluginApiVersion,
|
|
)
|
|
.catchError((error, stackTrace) {
|
|
// If the column already exists, ignore the error
|
|
if (!error.toString().contains('duplicate column name')) {
|
|
throw error;
|
|
}
|
|
});
|
|
},
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
LazyDatabase _openConnection() {
|
|
// the LazyDatabase util lets us find the right location for the file async.
|
|
return LazyDatabase(() async {
|
|
// put the database file, called db.sqlite here, into the documents folder
|
|
// for your app.
|
|
final dbFolder = await getApplicationSupportDirectory();
|
|
final file = File(join(dbFolder.path, 'db.sqlite'));
|
|
|
|
// Also work around limitations on old Android versions
|
|
if (Platform.isAndroid) {
|
|
await applyWorkaroundToOpenSqlite3OnOldAndroidVersions();
|
|
}
|
|
|
|
// Make sqlite3 pick a more suitable location for temporary files - the
|
|
// one from the system may be inaccessible due to sandboxing.
|
|
final cacheBase = (await getTemporaryDirectory()).path;
|
|
// We can't access /tmp on Android, which sqlite3 would try by default.
|
|
// Explicitly tell it about the correct temporary directory.
|
|
sqlite3.tempDirectory = cacheBase;
|
|
|
|
return NativeDatabase.createInBackground(file);
|
|
});
|
|
}
|