package exh import android.content.Context import com.elvishew.xlog.XLog import com.pushtorefresh.storio.sqlite.queries.Query import com.pushtorefresh.storio.sqlite.queries.RawQuery import eu.kanade.tachiyomi.BuildConfig import eu.kanade.tachiyomi.data.backup.models.DHistory import eu.kanade.tachiyomi.data.database.DatabaseHelper import eu.kanade.tachiyomi.data.database.models.Chapter import eu.kanade.tachiyomi.data.database.models.Manga import eu.kanade.tachiyomi.data.database.models.MangaImpl import eu.kanade.tachiyomi.data.database.models.Track import eu.kanade.tachiyomi.data.database.resolvers.MangaUrlPutResolver import eu.kanade.tachiyomi.data.database.tables.MangaTable import eu.kanade.tachiyomi.data.library.LibraryUpdateJob import eu.kanade.tachiyomi.data.preference.PreferencesHelper import eu.kanade.tachiyomi.data.updater.UpdaterJob import eu.kanade.tachiyomi.extension.ExtensionUpdateJob import exh.source.BlacklistedSources import java.io.File import java.net.URI import java.net.URISyntaxException import uy.kohesive.injekt.injectLazy object EXHMigrations { private val db: DatabaseHelper by injectLazy() private val logger = XLog.tag("EXHMigrations") /** * Performs a migration when the application is updated. * * @param preferences Preferences of the application. * @return true if a migration is performed, false otherwise. */ fun upgrade(preferences: PreferencesHelper): Boolean { val context = preferences.context val oldVersion = preferences.eh_lastVersionCode().get() try { if (oldVersion < BuildConfig.VERSION_CODE) { // Fresh install if (oldVersion == 0) { // Set up default background tasks UpdaterJob.setupTask(context) ExtensionUpdateJob.setupTask(context) LibraryUpdateJob.setupTask(context) return false } if (oldVersion < 4) { db.inTransaction { // Migrate Tsumino source IDs db.lowLevel().executeSQL( RawQuery.builder() .query( """ UPDATE ${MangaTable.TABLE} SET ${MangaTable.COL_SOURCE} = $HBROWSE_SOURCE_ID WHERE ${MangaTable.COL_SOURCE} = 6912 """.trimIndent() ) .affectsTables(MangaTable.TABLE) .build() ) // Migrate BHrowse URLs val hBrowseManga = db.db.get() .listOfObjects(Manga::class.java) .withQuery( Query.builder() .table(MangaTable.TABLE) .where("${MangaTable.COL_SOURCE} = $HBROWSE_SOURCE_ID") .build() ) .prepare() .executeAsBlocking() hBrowseManga.forEach { it.url = it.url + "/c00001" } db.db.put() .objects(hBrowseManga) // Extremely slow without the resolver :/ .withPutResolver(MangaUrlPutResolver()) .prepare() .executeAsBlocking() } } // if (oldVersion < 1) { } // do stuff here when releasing changed crap // TODO BE CAREFUL TO NOT FUCK UP MergedSources IF CHANGING URLs preferences.eh_lastVersionCode().set(BuildConfig.VERSION_CODE) return true } } catch (e: Exception) { logger.e("Failed to migrate app from $oldVersion -> ${BuildConfig.VERSION_CODE}!", e) } return false } fun migrateBackupEntry(manga: MangaImpl): MangaImpl { // Migrate HentaiCafe source IDs if (manga.source == 6908L) { manga.source = HENTAI_CAFE_SOURCE_ID!! } // Migrate Tsumino source IDs if (manga.source == 6909L) { manga.source = TSUMINO_SOURCE_ID!! } if (manga.source == 6912L) { manga.source = HBROWSE_SOURCE_ID!! } // Migrate nhentai URLs if (manga.source == NHENTAI_SOURCE_ID) { manga.url = getUrlWithoutDomain(manga.url) } // Allow importing of nhentai extension backups if (manga.source in BlacklistedSources.NHENTAI_EXT_SOURCES) { manga.source = NHENTAI_SOURCE_ID } // Allow importing of English PervEden extension backups if (manga.source in BlacklistedSources.PERVEDEN_EN_EXT_SOURCES) { manga.source = PERV_EDEN_EN_SOURCE_ID } // Allow importing of Italian PervEden extension backups if (manga.source in BlacklistedSources.PERVEDEN_IT_EXT_SOURCES) { manga.source = PERV_EDEN_IT_SOURCE_ID } // Allow importing of EHentai extension backups if (manga.source in BlacklistedSources.EHENTAI_EXT_SOURCES) { manga.source = EH_SOURCE_ID } return manga } private fun backupDatabase(context: Context, oldMigrationVersion: Int) { val backupLocation = File(File(context.filesDir, "exh_db_bck"), "$oldMigrationVersion.bck.db") if (backupLocation.exists()) return // Do not backup same version twice val dbLocation = context.getDatabasePath(db.lowLevel().sqliteOpenHelper().databaseName) try { dbLocation.copyTo(backupLocation, overwrite = true) } catch (t: Throwable) { XLog.w("Failed to backup database!") } } private fun getUrlWithoutDomain(orig: String): String { return try { val uri = URI(orig) var out = uri.path if (uri.query != null) { out += "?" + uri.query } if (uri.fragment != null) { out += "#" + uri.fragment } out } catch (e: URISyntaxException) { orig } } } data class BackupEntry( val manga: Manga, val chapters: List, val categories: List, val history: List, val tracks: List )