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); }); }