
* Removed some extra spam for the console * Implemented the code to update RowVersion, which is our concurrency check * Fixed a critical issue where more than one bookmark could occur for a given chapter due to a race condition. Now we use concurrency checks and we also gracefully allow more than one row, by only grabbing first. * Cleaned up the logic for IHasConcurencyToken and updated the setters to be private. * Lots of comments and when deleting a library, remove any user progress items for which chapters don't exist. * When deleting a Series, cleanup user progress rows. * Now after a scan of library, if a series is removed, collection tags are pruned as well if there are no longer any series bound to it. * Updated the image on the Readme to show a better picture * Small code cleanup to remove null check modifier as I check for null just before then * Fixed images loading multiple times due to using function in binding with random. You can now click chapter images to read that chapter specifically. * Fixed cards being different sizes when read vs unread * Moved over Robbie's workflow changes from notifier. Commented out activity indicators as that is not shipping with this release. * Remove code that isn't needed * Reverted GA * Changed GA to trigger only when HEAD is updated
259 lines
7 KiB
C#
259 lines
7 KiB
C#
using System;
|
|
using System.IO;
|
|
using System.Text.Json;
|
|
using Kavita.Common.EnvironmentInfo;
|
|
using Microsoft.Extensions.Hosting;
|
|
|
|
namespace Kavita.Common
|
|
{
|
|
public static class Configuration
|
|
{
|
|
private static string AppSettingsFilename = 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);
|
|
}
|
|
|
|
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" : "") + ".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 bool SetJwtToken(string filePath, string token)
|
|
{
|
|
try
|
|
{
|
|
var currentToken = GetJwtToken(filePath);
|
|
var json = File.ReadAllText(filePath)
|
|
.Replace("\"TokenKey\": \"" + currentToken, "\"TokenKey\": \"" + token);
|
|
File.WriteAllText(filePath, json);
|
|
return true;
|
|
}
|
|
catch (Exception)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public static bool CheckIfJwtTokenSet()
|
|
{
|
|
//string filePath
|
|
try
|
|
{
|
|
return GetJwtToken(GetAppSettingFilename()) != "super secret unguessable key";
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine("Error writing app settings: " + ex.Message);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public static bool UpdateJwtToken(string token)
|
|
{
|
|
try
|
|
{
|
|
var filePath = GetAppSettingFilename();
|
|
var json = File.ReadAllText(filePath).Replace("super secret unguessable key", token);
|
|
File.WriteAllText(GetAppSettingFilename(), json);
|
|
return true;
|
|
}
|
|
catch (Exception)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Port
|
|
|
|
public static bool SetPort(string filePath, int port)
|
|
{
|
|
if (new OsInfo(Array.Empty<IOsVersionAdapter>()).IsDocker)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
try
|
|
{
|
|
var currentPort = GetPort(filePath);
|
|
var json = File.ReadAllText(filePath).Replace("\"Port\": " + currentPort, "\"Port\": " + port);
|
|
File.WriteAllText(filePath, json);
|
|
return true;
|
|
}
|
|
catch (Exception)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public 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
|
|
|
|
public static bool SetLogLevel(string filePath, string logLevel)
|
|
{
|
|
try
|
|
{
|
|
var currentLevel = GetLogLevel(filePath);
|
|
var json = File.ReadAllText(filePath)
|
|
.Replace($"\"Default\": \"{currentLevel}\"", $"\"Default\": \"{logLevel}\"");
|
|
File.WriteAllText(filePath, json);
|
|
return true;
|
|
}
|
|
catch (Exception)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public 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())
|
|
{
|
|
if (logProperty.Name.Equals("Default"))
|
|
{
|
|
return logProperty.Value.GetString();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine("Error writing app settings: " + ex.Message);
|
|
}
|
|
|
|
return "Information";
|
|
}
|
|
|
|
#endregion
|
|
|
|
public 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;
|
|
}
|
|
|
|
public static bool SetBranch(string filePath, string updatedBranch)
|
|
{
|
|
try
|
|
{
|
|
var currentBranch = GetBranch(filePath);
|
|
var json = File.ReadAllText(filePath)
|
|
.Replace("\"Branch\": " + currentBranch, "\"Branch\": " + updatedBranch);
|
|
File.WriteAllText(filePath, json);
|
|
return true;
|
|
}
|
|
catch (Exception)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|