mirror of
https://github.com/Kareadita/Kavita.git
synced 2025-06-04 06:04:37 -04:00
* Moved the Server Settings out into a button on nav header * Refactored Mange Users page to the new design (skeleton). Implemented skeleton code for Invite User. * Hashed out more of the code, but need to move all the email code to a Kavita controlled API server due to password credentials. * Cleaned up some warnings * When no user exists for an api key in Plugin controller, throw 401. * Hooked in the ability to check if the Kavita instance can be accessed externally so we can determine if the user can invite or not. * Hooked up some logic if the user's server isn't accessible, then default to old flow * Basic flow is working for confirm email. Needs validation, error handling, etc. * Refactored Password validation to account service * Cleaned up the code in confirm-email to work much better. * Refactored the login page to have a container functionality, so we can reuse the styles on multiple pages (registration pages). Hooked up the code for confirm email. * Messy code, but making progress. Refactored Register to be used only for first time user registration. Added a new register component to handle first time flow only. * Invite works much better, still needs a bit of work for non-accessible server setup. Started work on underlying manage users page to meet new design. * Changed (you) to a star to indicate who you're logged in as. * Inviting a user is now working and tested fully. * Removed the register member component as we now have invite and confirm components. * Editing a user is now working. Username change and Role/Library access from within one screen. Email changing is on hold. * Cleaned up code for edit user and disabled email field for now. * Cleaned up the code to indicate changing a user's email is not possible. * Implemented a migration for existing accounts so they can validate their emails and still login. * Change url for email server * Implemented the ability to resend an email confirmation code (or regenerate for non accessible servers). Fixed an overflow on the confirm dialog. * Took care of some code cleanup * Removed 3 db calls from cover refresh and some misc cleanup * Fixed a broken test
353 lines
11 KiB
C#
353 lines
11 KiB
C#
using System;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Text.Json;
|
|
using Kavita.Common.EnvironmentInfo;
|
|
using Microsoft.Extensions.Hosting;
|
|
|
|
namespace Kavita.Common
|
|
{
|
|
public static class Configuration
|
|
{
|
|
public static readonly string AppSettingsFilename = Path.Join("config", GetAppSettingFilename());
|
|
|
|
public static string Branch
|
|
{
|
|
get => GetBranch(GetAppSettingFilename());
|
|
set => SetBranch(GetAppSettingFilename(), value);
|
|
}
|
|
|
|
public static int Port
|
|
{
|
|
get => GetPort(GetAppSettingFilename());
|
|
set => SetPort(GetAppSettingFilename(), value);
|
|
}
|
|
|
|
public static string JwtToken
|
|
{
|
|
get => GetJwtToken(GetAppSettingFilename());
|
|
set => SetJwtToken(GetAppSettingFilename(), value);
|
|
}
|
|
|
|
public static string LogLevel
|
|
{
|
|
get => GetLogLevel(GetAppSettingFilename());
|
|
set => SetLogLevel(GetAppSettingFilename(), value);
|
|
}
|
|
|
|
public static string LogPath
|
|
{
|
|
get => GetLoggingFile(GetAppSettingFilename());
|
|
set => SetLoggingFile(GetAppSettingFilename(), value);
|
|
}
|
|
|
|
public static string DatabasePath
|
|
{
|
|
get => GetDatabasePath(GetAppSettingFilename());
|
|
set => SetDatabasePath(GetAppSettingFilename(), value);
|
|
}
|
|
|
|
private static string GetAppSettingFilename()
|
|
{
|
|
if (!string.IsNullOrEmpty(AppSettingsFilename))
|
|
{
|
|
return AppSettingsFilename;
|
|
}
|
|
|
|
var environment = Environment.GetEnvironmentVariable("ASPNETCORE_ENVIRONMENT");
|
|
var isDevelopment = environment == Environments.Development;
|
|
return "appsettings" + (isDevelopment ? ".Development" : string.Empty) + ".json";
|
|
}
|
|
|
|
#region JWT Token
|
|
|
|
private static string GetJwtToken(string filePath)
|
|
{
|
|
try
|
|
{
|
|
var json = File.ReadAllText(filePath);
|
|
var jsonObj = JsonSerializer.Deserialize<dynamic>(json);
|
|
const string key = "TokenKey";
|
|
|
|
if (jsonObj.TryGetProperty(key, out JsonElement tokenElement))
|
|
{
|
|
return tokenElement.GetString();
|
|
}
|
|
|
|
return string.Empty;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine("Error reading app settings: " + ex.Message);
|
|
}
|
|
|
|
return string.Empty;
|
|
}
|
|
|
|
private static void SetJwtToken(string filePath, string token)
|
|
{
|
|
try
|
|
{
|
|
var currentToken = GetJwtToken(filePath);
|
|
var json = File.ReadAllText(filePath)
|
|
.Replace("\"TokenKey\": \"" + currentToken, "\"TokenKey\": \"" + token);
|
|
File.WriteAllText(filePath, json);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
/* Swallow exception */
|
|
}
|
|
}
|
|
|
|
public static bool CheckIfJwtTokenSet()
|
|
{
|
|
try
|
|
{
|
|
return GetJwtToken(GetAppSettingFilename()) != "super secret unguessable key";
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine("Error writing app settings: " + ex.Message);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Port
|
|
|
|
private static void SetPort(string filePath, int port)
|
|
{
|
|
if (new OsInfo(Array.Empty<IOsVersionAdapter>()).IsDocker)
|
|
{
|
|
return;
|
|
}
|
|
|
|
try
|
|
{
|
|
var currentPort = GetPort(filePath);
|
|
var json = File.ReadAllText(filePath).Replace("\"Port\": " + currentPort, "\"Port\": " + port);
|
|
File.WriteAllText(filePath, json);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
/* Swallow Exception */
|
|
}
|
|
}
|
|
|
|
private static int GetPort(string filePath)
|
|
{
|
|
const int defaultPort = 5000;
|
|
if (new OsInfo(Array.Empty<IOsVersionAdapter>()).IsDocker)
|
|
{
|
|
return defaultPort;
|
|
}
|
|
|
|
try
|
|
{
|
|
var json = File.ReadAllText(filePath);
|
|
var jsonObj = JsonSerializer.Deserialize<dynamic>(json);
|
|
const string key = "Port";
|
|
|
|
if (jsonObj.TryGetProperty(key, out JsonElement tokenElement))
|
|
{
|
|
return tokenElement.GetInt32();
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine("Error writing app settings: " + ex.Message);
|
|
}
|
|
|
|
return defaultPort;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region LogLevel
|
|
|
|
private static void SetLogLevel(string filePath, string logLevel)
|
|
{
|
|
try
|
|
{
|
|
var currentLevel = GetLogLevel(filePath);
|
|
var json = File.ReadAllText(filePath)
|
|
.Replace($"\"Default\": \"{currentLevel}\"", $"\"Default\": \"{logLevel}\"");
|
|
File.WriteAllText(filePath, json);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
/* Swallow Exception */
|
|
}
|
|
}
|
|
|
|
private static string GetLogLevel(string filePath)
|
|
{
|
|
try
|
|
{
|
|
var json = File.ReadAllText(filePath);
|
|
var jsonObj = JsonSerializer.Deserialize<dynamic>(json);
|
|
|
|
if (jsonObj.TryGetProperty("Logging", out JsonElement tokenElement))
|
|
{
|
|
foreach (var property in tokenElement.EnumerateObject())
|
|
{
|
|
if (!property.Name.Equals("LogLevel")) continue;
|
|
foreach (var logProperty in property.Value.EnumerateObject().Where(logProperty => logProperty.Name.Equals("Default")))
|
|
{
|
|
return logProperty.Value.GetString();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine("Error writing app settings: " + ex.Message);
|
|
}
|
|
|
|
return "Information";
|
|
}
|
|
|
|
#endregion
|
|
|
|
private static string GetBranch(string filePath)
|
|
{
|
|
const string defaultBranch = "main";
|
|
|
|
try
|
|
{
|
|
var json = File.ReadAllText(filePath);
|
|
var jsonObj = JsonSerializer.Deserialize<dynamic>(json);
|
|
const string key = "Branch";
|
|
|
|
if (jsonObj.TryGetProperty(key, out JsonElement tokenElement))
|
|
{
|
|
return tokenElement.GetString();
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine("Error reading app settings: " + ex.Message);
|
|
}
|
|
|
|
return defaultBranch;
|
|
}
|
|
|
|
private static void SetBranch(string filePath, string updatedBranch)
|
|
{
|
|
try
|
|
{
|
|
var currentBranch = GetBranch(filePath);
|
|
var json = File.ReadAllText(filePath)
|
|
.Replace("\"Branch\": " + currentBranch, "\"Branch\": " + updatedBranch);
|
|
File.WriteAllText(filePath, json);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
/* Swallow Exception */
|
|
}
|
|
}
|
|
|
|
private static string GetLoggingFile(string filePath)
|
|
{
|
|
const string defaultFile = "config/logs/kavita.log";
|
|
|
|
try
|
|
{
|
|
var json = File.ReadAllText(filePath);
|
|
var jsonObj = JsonSerializer.Deserialize<dynamic>(json);
|
|
|
|
if (jsonObj.TryGetProperty("Logging", out JsonElement tokenElement))
|
|
{
|
|
foreach (var property in tokenElement.EnumerateObject())
|
|
{
|
|
if (!property.Name.Equals("File")) continue;
|
|
foreach (var logProperty in property.Value.EnumerateObject())
|
|
{
|
|
if (logProperty.Name.Equals("Path"))
|
|
{
|
|
return logProperty.Value.GetString();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine("Error writing app settings: " + ex.Message);
|
|
}
|
|
|
|
return defaultFile;
|
|
}
|
|
|
|
/// <summary>
|
|
/// This should NEVER be called except by <see cref="MigrateConfigFiles"/>
|
|
/// </summary>
|
|
/// <param name="filePath"></param>
|
|
/// <param name="directory"></param>
|
|
private static void SetLoggingFile(string filePath, string directory)
|
|
{
|
|
try
|
|
{
|
|
var currentFile = GetLoggingFile(filePath);
|
|
var json = File.ReadAllText(filePath)
|
|
.Replace("\"Path\": \"" + currentFile + "\"", "\"Path\": \"" + directory + "\"");
|
|
File.WriteAllText(filePath, json);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
/* Swallow Exception */
|
|
Console.WriteLine(ex);
|
|
}
|
|
}
|
|
|
|
private static string GetDatabasePath(string filePath)
|
|
{
|
|
const string defaultFile = "config/kavita.db";
|
|
|
|
try
|
|
{
|
|
var json = File.ReadAllText(filePath);
|
|
var jsonObj = JsonSerializer.Deserialize<dynamic>(json);
|
|
|
|
if (jsonObj.TryGetProperty("ConnectionStrings", out JsonElement tokenElement))
|
|
{
|
|
foreach (var property in tokenElement.EnumerateObject())
|
|
{
|
|
if (!property.Name.Equals("DefaultConnection")) continue;
|
|
return property.Value.GetString();
|
|
}
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine("Error writing app settings: " + ex.Message);
|
|
}
|
|
|
|
return defaultFile;
|
|
}
|
|
|
|
/// <summary>
|
|
/// This should NEVER be called except by MigrateConfigFiles
|
|
/// </summary>
|
|
/// <param name="filePath"></param>
|
|
/// <param name="updatedPath"></param>
|
|
private static void SetDatabasePath(string filePath, string updatedPath)
|
|
{
|
|
try
|
|
{
|
|
var existingString = GetDatabasePath(filePath);
|
|
var json = File.ReadAllText(filePath)
|
|
.Replace(existingString,
|
|
"Data source=" + updatedPath);
|
|
File.WriteAllText(filePath, json);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
/* Swallow Exception */
|
|
}
|
|
}
|
|
}
|
|
}
|