| | 1 | | using System; |
| | 2 | | using System.Collections.Generic; |
| | 3 | | using System.IO; |
| | 4 | | using System.Linq; |
| | 5 | | using MediaBrowser.Common.Configuration; |
| | 6 | | using Newtonsoft.Json.Linq; |
| | 7 | |
|
| | 8 | | namespace Jellyfin.Server.Migrations.Routines |
| | 9 | | { |
| | 10 | | /// <summary> |
| | 11 | | /// Migration to initialize the user logging configuration file "logging.user.json". |
| | 12 | | /// If the deprecated logging.json file exists and has a custom config, it will be used as logging.user.json, |
| | 13 | | /// otherwise a blank file will be created. |
| | 14 | | /// </summary> |
| | 15 | | internal class CreateUserLoggingConfigFile : IMigrationRoutine |
| | 16 | | { |
| | 17 | | /// <summary> |
| | 18 | | /// File history for logging.json as existed during this migration creation. The contents for each has been mini |
| | 19 | | /// </summary> |
| 0 | 20 | | private readonly List<string> _defaultConfigHistory = new List<string> |
| 0 | 21 | | { |
| 0 | 22 | | // 9a6c27947353585391e211aa88b925f81e8cd7b9 |
| 0 | 23 | | @"{""Serilog"":{""MinimumLevel"":{""Default"":""Information"",""Override"":{""Microsoft"":""Warning"",""Syst |
| 0 | 24 | | // 71bdcd730705a714ee208eaad7290b7c68df3885 |
| 0 | 25 | | @"{""Serilog"":{""MinimumLevel"":""Information"",""WriteTo"":[{""Name"":""Console"",""Args"":{""outputTempla |
| 0 | 26 | | // a44936f97f8afc2817d3491615a7cfe1e31c251c |
| 0 | 27 | | @"{""Serilog"":{""MinimumLevel"":""Information"",""WriteTo"":[{""Name"":""Console"",""Args"":{""outputTempla |
| 0 | 28 | | // 7af3754a11ad5a4284f107997fb5419a010ce6f3 |
| 0 | 29 | | @"{""Serilog"":{""MinimumLevel"":""Information"",""WriteTo"":[{""Name"":""Console"",""Args"":{""outputTempla |
| 0 | 30 | | // 60691349a11f541958e0b2247c9abc13cb40c9fb |
| 0 | 31 | | @"{""Serilog"":{""MinimumLevel"":""Information"",""WriteTo"":[{""Name"":""Console"",""Args"":{""outputTempla |
| 0 | 32 | | // 65fe243afbcc4b596cf8726708c1965cd34b5f68 |
| 0 | 33 | | @"{""Serilog"":{""MinimumLevel"":""Information"",""WriteTo"":[{""Name"":""Console"",""Args"":{""outputTempla |
| 0 | 34 | | // 96c9af590494aa8137d5a061aaf1e68feee60b67 |
| 0 | 35 | | @"{""Serilog"":{""MinimumLevel"":""Information"",""WriteTo"":[{""Name"":""Console"",""Args"":{""outputTempla |
| 0 | 36 | | }; |
| | 37 | |
|
| | 38 | | private readonly IApplicationPaths _appPaths; |
| | 39 | |
|
| | 40 | | public CreateUserLoggingConfigFile(IApplicationPaths appPaths) |
| | 41 | | { |
| 0 | 42 | | _appPaths = appPaths; |
| 0 | 43 | | } |
| | 44 | |
|
| | 45 | | /// <inheritdoc/> |
| 0 | 46 | | public Guid Id => Guid.Parse("{EF103419-8451-40D8-9F34-D1A8E93A1679}"); |
| | 47 | |
|
| | 48 | | /// <inheritdoc/> |
| 0 | 49 | | public string Name => "CreateLoggingConfigHierarchy"; |
| | 50 | |
|
| | 51 | | /// <inheritdoc/> |
| 0 | 52 | | public bool PerformOnNewInstall => false; |
| | 53 | |
|
| | 54 | | /// <inheritdoc/> |
| | 55 | | public void Perform() |
| | 56 | | { |
| 0 | 57 | | var logDirectory = _appPaths.ConfigurationDirectoryPath; |
| 0 | 58 | | var existingConfigPath = Path.Combine(logDirectory, "logging.json"); |
| | 59 | |
|
| | 60 | | // If the existing logging.json config file is unmodified, then 'reset' it by moving it to 'logging.old.json |
| | 61 | | // NOTE: This config file has 'reloadOnChange: true', so this change will take effect immediately even thoug |
| 0 | 62 | | if (File.Exists(existingConfigPath) && ExistingConfigUnmodified(existingConfigPath)) |
| | 63 | | { |
| 0 | 64 | | File.Move(existingConfigPath, Path.Combine(logDirectory, "logging.old.json")); |
| | 65 | | } |
| 0 | 66 | | } |
| | 67 | |
|
| | 68 | | /// <summary> |
| | 69 | | /// Check if the existing logging.json file has not been modified by the user by comparing it to all the |
| | 70 | | /// versions in our git history. Until now, the file has never been migrated after first creation so users |
| | 71 | | /// could have any version from the git history. |
| | 72 | | /// </summary> |
| | 73 | | /// <exception cref="IOException"><paramref name="oldConfigPath"/> does not exist or could not be read.</excepti |
| | 74 | | private bool ExistingConfigUnmodified(string oldConfigPath) |
| | 75 | | { |
| 0 | 76 | | var existingConfigJson = JToken.Parse(File.ReadAllText(oldConfigPath)); |
| 0 | 77 | | return _defaultConfigHistory |
| 0 | 78 | | .Select(JToken.Parse) |
| 0 | 79 | | .Any(historicalConfigJson => JToken.DeepEquals(existingConfigJson, historicalConfigJson)); |
| | 80 | | } |
| | 81 | | } |
| | 82 | | } |