using System; using System.Collections.Generic; using System.Collections.Immutable; using System.Globalization; using System.IO; using System.Linq; using System.Reflection; using System.Text.Json; using System.Threading.Tasks; using API.Constants; using API.Data.Repositories; using API.DTOs.Settings; using API.Entities; using API.Entities.Enums; using API.Entities.Enums.Theme; using API.Entities.MetadataMatching; using API.Extensions; using API.Services; using Kavita.Common; using Kavita.Common.EnvironmentInfo; using Microsoft.AspNetCore.Identity; using Microsoft.EntityFrameworkCore; namespace API.Data; public static class Seed { /// /// Generated on Startup. Seed.SeedSettings must run before /// public static ImmutableArray DefaultSettings; public static readonly ImmutableArray DefaultThemes = [ ..new List { new() { Name = "Dark", NormalizedName = "Dark".ToNormalized(), Provider = ThemeProvider.System, FileName = "dark.scss", IsDefault = true, Description = "Default theme shipped with Kavita" } }.ToArray() ]; public static readonly ImmutableArray DefaultStreams = ImmutableArray.Create( new List { new() { Name = "on-deck", StreamType = DashboardStreamType.OnDeck, Order = 0, IsProvided = true, Visible = true }, new() { Name = "recently-updated", StreamType = DashboardStreamType.RecentlyUpdated, Order = 1, IsProvided = true, Visible = true }, new() { Name = "newly-added", StreamType = DashboardStreamType.NewlyAdded, Order = 2, IsProvided = true, Visible = true }, new() { Name = "more-in-genre", StreamType = DashboardStreamType.MoreInGenre, Order = 3, IsProvided = true, Visible = false }, }.ToArray()); public static readonly ImmutableArray DefaultSideNavStreams = ImmutableArray.Create( new AppUserSideNavStream() { Name = "want-to-read", StreamType = SideNavStreamType.WantToRead, Order = 1, IsProvided = true, Visible = true }, new AppUserSideNavStream() { Name = "collections", StreamType = SideNavStreamType.Collections, Order = 2, IsProvided = true, Visible = true }, new AppUserSideNavStream() { Name = "reading-lists", StreamType = SideNavStreamType.ReadingLists, Order = 3, IsProvided = true, Visible = true }, new AppUserSideNavStream() { Name = "bookmarks", StreamType = SideNavStreamType.Bookmarks, Order = 4, IsProvided = true, Visible = true }, new AppUserSideNavStream() { Name = "all-series", StreamType = SideNavStreamType.AllSeries, Order = 5, IsProvided = true, Visible = true }, new AppUserSideNavStream() { Name = "browse-authors", StreamType = SideNavStreamType.BrowsePeople, Order = 6, IsProvided = true, Visible = true }); public static async Task SeedRoles(RoleManager roleManager) { var roles = typeof(PolicyConstants) .GetFields(BindingFlags.Public | BindingFlags.Static) .Where(f => f.FieldType == typeof(string)) .ToDictionary(f => f.Name, f => (string) f.GetValue(null)!).Values .Select(policyName => new AppRole() {Name = policyName}) .ToList(); foreach (var role in roles) { var exists = await roleManager.RoleExistsAsync(role.Name!); if (!exists) { await roleManager.CreateAsync(role); } } } public static async Task SeedThemes(DataContext context) { await context.Database.EnsureCreatedAsync(); foreach (var theme in DefaultThemes) { var existing = context.SiteTheme.FirstOrDefault(s => s.Name.Equals(theme.Name)); if (existing == null) { await context.SiteTheme.AddAsync(theme); } } await context.SaveChangesAsync(); } public static async Task SeedDefaultStreams(IUnitOfWork unitOfWork) { var allUsers = await unitOfWork.UserRepository.GetAllUsersAsync(AppUserIncludes.DashboardStreams); foreach (var user in allUsers) { if (user.DashboardStreams.Count != 0) continue; user.DashboardStreams ??= new List(); foreach (var defaultStream in DefaultStreams) { var newStream = new AppUserDashboardStream { Name = defaultStream.Name, IsProvided = defaultStream.IsProvided, Order = defaultStream.Order, StreamType = defaultStream.StreamType, Visible = defaultStream.Visible, }; user.DashboardStreams.Add(newStream); } unitOfWork.UserRepository.Update(user); await unitOfWork.CommitAsync(); } } public static async Task SeedDefaultSideNavStreams(IUnitOfWork unitOfWork) { var allUsers = await unitOfWork.UserRepository.GetAllUsersAsync(AppUserIncludes.SideNavStreams); foreach (var user in allUsers) { user.SideNavStreams ??= new List(); foreach (var defaultStream in DefaultSideNavStreams) { if (user.SideNavStreams.Any(s => s.Name == defaultStream.Name && s.StreamType == defaultStream.StreamType)) continue; var newStream = new AppUserSideNavStream() { Name = defaultStream.Name, IsProvided = defaultStream.IsProvided, Order = defaultStream.Order, StreamType = defaultStream.StreamType, Visible = defaultStream.Visible, }; user.SideNavStreams.Add(newStream); } unitOfWork.UserRepository.Update(user); await unitOfWork.CommitAsync(); } } public static async Task SeedSettings(DataContext context, IDirectoryService directoryService) { await context.Database.EnsureCreatedAsync(); DefaultSettings = ImmutableArray.Create(new List() { new() {Key = ServerSettingKey.CacheDirectory, Value = directoryService.CacheDirectory}, new() {Key = ServerSettingKey.TaskScan, Value = "daily"}, new() {Key = ServerSettingKey.TaskBackup, Value = "daily"}, new() {Key = ServerSettingKey.TaskCleanup, Value = "daily"}, new() {Key = ServerSettingKey.LoggingLevel, Value = "Debug"}, new() { Key = ServerSettingKey.BackupDirectory, Value = Path.GetFullPath(DirectoryService.BackupDirectory) }, new() { Key = ServerSettingKey.Port, Value = Configuration.DefaultHttpPort + string.Empty }, // Not used from DB, but DB is sync with appSettings.json new() { Key = ServerSettingKey.IpAddresses, Value = Configuration.DefaultIpAddresses }, // Not used from DB, but DB is sync with appSettings.json new() {Key = ServerSettingKey.AllowStatCollection, Value = "true"}, new() {Key = ServerSettingKey.EnableOpds, Value = "true"}, new() {Key = ServerSettingKey.BaseUrl, Value = "/"}, new() {Key = ServerSettingKey.InstallId, Value = HashUtil.AnonymousToken()}, new() {Key = ServerSettingKey.InstallVersion, Value = BuildInfo.Version.ToString()}, new() {Key = ServerSettingKey.BookmarkDirectory, Value = directoryService.BookmarkDirectory}, new() {Key = ServerSettingKey.TotalBackups, Value = "30"}, new() {Key = ServerSettingKey.TotalLogs, Value = "30"}, new() {Key = ServerSettingKey.EnableFolderWatching, Value = "false"}, new() {Key = ServerSettingKey.HostName, Value = string.Empty}, new() {Key = ServerSettingKey.EncodeMediaAs, Value = EncodeFormat.PNG.ToString()}, new() {Key = ServerSettingKey.LicenseKey, Value = string.Empty}, new() {Key = ServerSettingKey.OnDeckProgressDays, Value = "30"}, new() {Key = ServerSettingKey.OnDeckUpdateDays, Value = "7"}, new() {Key = ServerSettingKey.CoverImageSize, Value = CoverImageSize.Default.ToString()}, new() { Key = ServerSettingKey.CacheSize, Value = Configuration.DefaultCacheMemory + string.Empty }, // Not used from DB, but DB is sync with appSettings.json new() { Key = ServerSettingKey.OidcConfiguration, Value = JsonSerializer.Serialize(new OidcConfigDto())}, new() {Key = ServerSettingKey.EmailHost, Value = string.Empty}, new() {Key = ServerSettingKey.EmailPort, Value = string.Empty}, new() {Key = ServerSettingKey.EmailAuthPassword, Value = string.Empty}, new() {Key = ServerSettingKey.EmailAuthUserName, Value = string.Empty}, new() {Key = ServerSettingKey.EmailSenderAddress, Value = string.Empty}, new() {Key = ServerSettingKey.EmailSenderDisplayName, Value = string.Empty}, new() {Key = ServerSettingKey.EmailEnableSsl, Value = "true"}, new() {Key = ServerSettingKey.EmailSizeLimit, Value = 26_214_400 + string.Empty}, new() {Key = ServerSettingKey.EmailCustomizedTemplates, Value = "false"}, new() {Key = ServerSettingKey.FirstInstallVersion, Value = BuildInfo.Version.ToString()}, new() {Key = ServerSettingKey.FirstInstallDate, Value = DateTime.UtcNow.ToString(CultureInfo.InvariantCulture)}, }.ToArray()); foreach (var defaultSetting in DefaultSettings) { var existing = await context.ServerSetting.FirstOrDefaultAsync(s => s.Key == defaultSetting.Key); if (existing == null) { await context.ServerSetting.AddAsync(defaultSetting); } } await context.SaveChangesAsync(); // Port, IpAddresses and LoggingLevel are managed in appSettings.json. Update the DB values to match (await context.ServerSetting.FirstAsync(s => s.Key == ServerSettingKey.Port)).Value = Configuration.Port + string.Empty; (await context.ServerSetting.FirstAsync(s => s.Key == ServerSettingKey.IpAddresses)).Value = Configuration.IpAddresses; (await context.ServerSetting.FirstAsync(s => s.Key == ServerSettingKey.CacheDirectory)).Value = directoryService.CacheDirectory + string.Empty; (await context.ServerSetting.FirstAsync(s => s.Key == ServerSettingKey.BackupDirectory)).Value = DirectoryService.BackupDirectory + string.Empty; (await context.ServerSetting.FirstAsync(s => s.Key == ServerSettingKey.CacheSize)).Value = Configuration.CacheSize + string.Empty; await SetOidcSettingsFromDisk(context); await context.SaveChangesAsync(); } public static async Task SetOidcSettingsFromDisk(DataContext context) { var oidcSettingEntry = await context.ServerSetting .FirstOrDefaultAsync(setting => setting.Key == ServerSettingKey.OidcConfiguration); var storedOidcSettings = JsonSerializer.Deserialize(oidcSettingEntry!.Value)!; var diskOidcSettings = Configuration.OidcSettings; storedOidcSettings.Authority = diskOidcSettings.Authority; storedOidcSettings.ClientId = diskOidcSettings.ClientId; storedOidcSettings.Secret = diskOidcSettings.Secret; storedOidcSettings.CustomScopes = diskOidcSettings.CustomScopes; oidcSettingEntry.Value = JsonSerializer.Serialize(storedOidcSettings); } public static async Task SeedMetadataSettings(DataContext context) { await context.Database.EnsureCreatedAsync(); var existing = await context.MetadataSettings.FirstOrDefaultAsync(); if (existing == null) { existing = new MetadataSettings() { Enabled = true, EnablePeople = true, EnableRelationships = true, EnableSummary = true, EnablePublicationStatus = true, EnableStartDate = true, EnableTags = false, EnableGenres = true, EnableLocalizedName = false, FirstLastPeopleNaming = true, EnableCoverImage = true, EnableChapterTitle = false, EnableChapterSummary = true, EnableChapterPublisher = true, EnableChapterCoverImage = false, EnableChapterReleaseDate = true, PersonRoles = [PersonRole.Writer, PersonRole.CoverArtist, PersonRole.Character] }; await context.MetadataSettings.AddAsync(existing); } await context.SaveChangesAsync(); } public static async Task SeedUserApiKeys(DataContext context) { await context.Database.EnsureCreatedAsync(); var users = await context.AppUser.ToListAsync(); foreach (var user in users.Where(user => string.IsNullOrEmpty(user.ApiKey))) { user.ApiKey = HashUtil.ApiKey(); } await context.SaveChangesAsync(); } }