package main import ( "fmt" "os" "path/filepath" "strings" "gopkg.in/ini.v1" ) func runMigrations(app *appContext) { migrateProfiles(app) migrateBootstrap(app) migrateEmailStorage(app) migrateNotificationMethods(app) linkExistingOmbiDiscordTelegram(app) // migrateHyphens(app) } // Migrate pre-0.2.0 user templates to profiles func migrateProfiles(app *appContext) { if !(app.storage.policy.BlockedTags == nil && app.storage.configuration.GroupedFolders == nil && len(app.storage.displayprefs) == 0) { app.info.Println("Migrating user template files to new profile format") app.storage.migrateToProfile() for _, path := range [3]string{app.storage.policy_path, app.storage.configuration_path, app.storage.displayprefs_path} { if _, err := os.Stat(path); !os.IsNotExist(err) { dir, fname := filepath.Split(path) newFname := strings.Replace(fname, ".json", ".old.json", 1) err := os.Rename(path, filepath.Join(dir, newFname)) if err != nil { app.err.Fatalf("Failed to rename %s: %s", fname, err) } } } app.info.Println("In case of a problem, your original files have been renamed to .old.json") app.storage.storeProfiles() } } // Migrate pre-0.2.5 bootstrap theme choice to a17t version. func migrateBootstrap(app *appContext) { themes := map[string]string{ "Jellyfin (Dark)": "dark", "Default (Light)": "light", } if app.config.Section("ui").Key("theme").String() == "Bootstrap (Light)" { app.config.Section("ui").Key("theme").SetValue("Default (Light)") } if val, ok := themes[app.config.Section("ui").Key("theme").String()]; ok { app.cssClass = val } } func migrateEmailConfig(app *appContext) { tempConfig, _ := ini.Load(app.configPath) fmt.Println(warning("Part of your email configuration will be migrated to the new \"messages\" section.\nA backup will be made.")) err := tempConfig.SaveTo(app.configPath + "_" + commit + ".bak") if err != nil { app.err.Fatalf("Failed to backup config: %v", err) return } for _, setting := range []string{"use_24h", "date_format", "message"} { if val := app.config.Section("email").Key(setting).Value(); val != "" { tempConfig.Section("email").Key(setting).SetValue("") tempConfig.Section("messages").Key(setting).SetValue(val) } } if app.config.Section("messages").Key("enabled").MustBool(false) || app.config.Section("telegram").Key("enabled").MustBool(false) { tempConfig.Section("messages").Key("enabled").SetValue("true") } err = tempConfig.SaveTo(app.configPath) if err != nil { app.err.Fatalf("Failed to save config: %v", err) return } app.loadConfig() } // Migrate pre-0.3.6 email settings to the new messages section. // Called just after loading email storage in main.go. func migrateEmailStorage(app *appContext) error { // use_24h was moved to messages, so this checks if migration has already occurred or not. if app.config.Section("email").Key("use_24h").Value() == "" { return nil } var emails map[string]interface{} err := loadJSON(app.storage.emails_path, &emails) if err != nil { return err } newEmails := map[string]EmailAddress{} for jfID, addr := range emails { switch addr.(type) { case string: newEmails[jfID] = EmailAddress{ Addr: addr.(string), Contact: true, } // In case email settings still persist after migration has already happened case map[string]interface{}: return nil default: return fmt.Errorf("email address was type %T, not string: \"%+v\"\n", addr, addr) } } config, err := ini.Load(app.configPath) if err != nil { return err } config.Section("email").Key("use_24h").SetValue("") if err := config.SaveTo(app.configPath); err != nil { return err } err = storeJSON(app.storage.emails_path+".bak", emails) if err != nil { return err } err = storeJSON(app.storage.emails_path, newEmails) if err != nil { return err } app.info.Println("Migrated to new email format. A backup has also been made.") return nil } // Pre-0.4.0, Admin notifications for invites were indexed by and only sent to email addresses. Now, when Jellyfin Login is enabled, They are indexed by the admin's Jellyfin ID, and send by any method enabled for them. This migrates storage to that format. func migrateNotificationMethods(app *appContext) error { if !app.config.Section("ui").Key("jellyfin_login").MustBool(false) { return nil } changes := false for code, invite := range app.storage.invites { if invite.Notify == nil { continue } for address, notifyPrefs := range invite.Notify { if !strings.Contains(address, "@") { continue } for _, email := range app.storage.GetEmails() { if email.Addr == address { invite.Notify[email.JellyfinID] = notifyPrefs delete(invite.Notify, address) changes = true break } } } if changes { app.storage.invites[code] = invite } } if changes { app.info.Printf("Migrated to modified invite storage format.") return app.storage.storeInvites() } return nil } // Pre-0.4.0, Ombi users were created without linking their Discord & Telegram accounts. This will add them. func linkExistingOmbiDiscordTelegram(app *appContext) error { if !discordEnabled && !telegramEnabled { return nil } if !app.config.Section("ombi").Key("enabled").MustBool(false) { return nil } idList := map[string][2]string{} for _, user := range app.storage.GetDiscord() { idList[user.JellyfinID] = [2]string{user.ID, ""} } for _, user := range app.storage.GetTelegram() { vals, ok := idList[user.JellyfinID] if !ok { vals = [2]string{"", ""} } vals[1] = user.Username idList[user.JellyfinID] = vals } for jfID, ids := range idList { ombiUser, status, err := app.getOmbiUser(jfID) if status != 200 || err != nil { app.debug.Printf("Failed to get Ombi user with Discord/Telegram \"%s\"/\"%s\" (%d): %v", ids[0], ids[1], status, err) continue } _, status, err = app.ombi.SetNotificationPrefs(ombiUser, ids[0], ids[1]) if status != 200 || err != nil { app.debug.Printf("Failed to set prefs for Ombi user \"%s\" (%d): %v", ombiUser["userName"].(string), status, err) continue } } return nil } func migrateToBadger(app *appContext) { app.info.Println("Migrating to Badger(hold)") app.storage.loadAnnouncements() for k, v := range app.storage.announcements { app.storage.SetAnnouncementsKey(k, v) } app.storage.loadDiscordUsers() for jfID, v := range app.storage.discord { app.storage.SetDiscordKey(jfID, v) } app.storage.loadTelegramUsers() for jfID, v := range app.storage.telegram { app.storage.SetTelegramKey(jfID, v) } app.storage.loadMatrixUsers() for jfID, v := range app.storage.matrix { app.storage.SetMatrixKey(jfID, v) } app.storage.loadEmails() for jfID, v := range app.storage.emails { app.storage.SetEmailsKey(jfID, v) } } // Migrate between hyphenated & non-hyphenated user IDs. Doesn't seem to happen anymore, so disabled. // func migrateHyphens(app *appContext) { // checkVersion := func(version string) int { // numberStrings := strings.Split(version, ".") // n := 0 // for _, s := range numberStrings { // num, err := strconv.Atoi(s) // if err == nil { // n += num // } // } // return n // } // if serverType == mediabrowser.JellyfinServer && checkVersion(app.jf.ServerInfo.Version) >= checkVersion("10.7.0") { // // Get users to check if server uses hyphenated userIDs // app.jf.GetUsers(false) // // noHyphens := true // for _, e := range app.storage.GetEmails() { // if strings.Contains(e.JellyfinID, "-") { // noHyphens = false // break // } // } // if noHyphens == app.jf.Hyphens { // var newEmails map[string]interface{} // var newUsers map[string]time.Time // var status, status2 int // var err, err2 error // if app.jf.Hyphens { // app.info.Println(info("Your build of Jellyfin appears to hypenate user IDs. Your emails.json/users.json file will be modified to match.")) // time.Sleep(time.Second * time.Duration(3)) // newEmails, status, err = app.hyphenateEmailStorage(app.storage.emails) // newUsers, status2, err2 = app.hyphenateUserStorage(app.storage.users) // } else { // app.info.Println(info("Your emails.json/users.json file uses hyphens, but the Jellyfin server no longer does. It will be modified.")) // time.Sleep(time.Second * time.Duration(3)) // newEmails, status, err = app.deHyphenateEmailStorage(app.storage.emails) // newUsers, status2, err2 = app.deHyphenateUserStorage(app.storage.users) // } // if status != 200 || err != nil { // app.err.Printf("Failed to get users from Jellyfin (%d): %v", status, err) // app.err.Fatalf("Couldn't upgrade emails.json") // } // if status2 != 200 || err2 != nil { // app.err.Printf("Failed to get users from Jellyfin (%d): %v", status, err) // app.err.Fatalf("Couldn't upgrade users.json") // } // emailBakFile := app.storage.emails_path + ".bak" // usersBakFile := app.storage.users_path + ".bak" // err = storeJSON(emailBakFile, app.storage.emails) // err2 = storeJSON(usersBakFile, app.storage.users) // if err != nil { // app.err.Fatalf("couldn't store emails.json backup: %v", err) // } // if err2 != nil { // app.err.Fatalf("couldn't store users.json backup: %v", err) // } // app.storage.emails = newEmails // app.storage.users = newUsers // err = app.storage.storeEmails() // err2 = app.storage.storeUsers() // if err != nil { // app.err.Fatalf("couldn't store emails.json: %v", err) // } // if err2 != nil { // app.err.Fatalf("couldn't store users.json: %v", err) // } // } // } // }