2020-04-08 15:19:00 +08:00
// Copyright (c) Microsoft Corporation
// The Microsoft Corporation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
using System ;
2020-10-22 03:32:53 +08:00
using System.Diagnostics.CodeAnalysis ;
using System.IO ;
2020-03-25 10:55:02 +08:00
using System.Text.Json ;
2020-10-23 00:45:48 +08:00
using Microsoft.PowerToys.Settings.UI.Library.Interfaces ;
using Microsoft.PowerToys.Settings.UI.Library.Utilities ;
2020-03-25 10:55:02 +08:00
2020-10-23 00:45:48 +08:00
namespace Microsoft.PowerToys.Settings.UI.Library
2020-03-25 10:55:02 +08:00
{
2020-09-22 01:14:44 +08:00
public class SettingsUtils : ISettingsUtils
2020-03-25 10:55:02 +08:00
{
2020-04-18 06:25:08 +08:00
private const string DefaultFileName = "settings.json" ;
2020-04-20 21:03:26 +08:00
private const string DefaultModuleName = "" ;
2020-09-22 01:14:44 +08:00
private IIOProvider _ioProvider ;
2020-04-18 06:25:08 +08:00
2020-09-22 01:14:44 +08:00
public SettingsUtils ( IIOProvider ioProvider )
2020-09-09 01:04:17 +08:00
{
2020-09-22 01:14:44 +08:00
_ioProvider = ioProvider ? ? throw new ArgumentNullException ( nameof ( ioProvider ) ) ;
2020-09-09 01:04:17 +08:00
}
2020-09-22 01:14:44 +08:00
private bool SettingsFolderExists ( string powertoy )
2020-04-04 10:02:38 +08:00
{
2020-09-22 01:14:44 +08:00
return _ioProvider . DirectoryExists ( System . IO . Path . Combine ( LocalApplicationDataFolder ( ) , $"Microsoft\\PowerToys\\{powertoy}" ) ) ;
2020-04-04 10:02:38 +08:00
}
2020-09-22 01:14:44 +08:00
private void CreateSettingsFolder ( string powertoy )
2020-04-04 10:02:38 +08:00
{
2020-09-22 01:14:44 +08:00
_ioProvider . CreateDirectory ( System . IO . Path . Combine ( LocalApplicationDataFolder ( ) , $"Microsoft\\PowerToys\\{powertoy}" ) ) ;
}
public void DeleteSettings ( string powertoy = "" )
{
_ioProvider . DeleteDirectory ( System . IO . Path . Combine ( LocalApplicationDataFolder ( ) , $"Microsoft\\PowerToys\\{powertoy}" ) ) ;
2020-04-04 10:02:38 +08:00
}
/// <summary>
/// Get path to the json settings file.
/// </summary>
/// <returns>string path.</returns>
2020-04-18 06:25:08 +08:00
public static string GetSettingsPath ( string powertoy , string fileName = DefaultFileName )
2020-03-25 10:55:02 +08:00
{
2020-04-08 01:19:14 +08:00
if ( string . IsNullOrWhiteSpace ( powertoy ) )
2020-03-25 10:55:02 +08:00
{
2020-09-22 01:14:44 +08:00
return System . IO . Path . Combine (
2020-04-04 10:02:38 +08:00
LocalApplicationDataFolder ( ) ,
2020-04-18 06:25:08 +08:00
$"Microsoft\\PowerToys\\{fileName}" ) ;
2020-03-25 10:55:02 +08:00
}
2020-04-08 01:19:14 +08:00
2020-09-22 01:14:44 +08:00
return System . IO . Path . Combine (
2020-04-04 10:02:38 +08:00
LocalApplicationDataFolder ( ) ,
2020-04-18 06:25:08 +08:00
$"Microsoft\\PowerToys\\{powertoy}\\{fileName}" ) ;
2020-03-25 10:55:02 +08:00
}
2020-09-22 01:14:44 +08:00
public bool SettingsExists ( string powertoy = DefaultModuleName , string fileName = DefaultFileName )
2020-04-04 10:02:38 +08:00
{
2020-09-22 01:14:44 +08:00
return _ioProvider . FileExists ( GetSettingsPath ( powertoy , fileName ) ) ;
2020-04-04 10:02:38 +08:00
}
/// <summary>
/// Get a Deserialized object of the json settings string.
2020-09-24 04:20:32 +08:00
/// This function creates a file in the powertoy folder if it does not exist and returns an object with default properties.
2020-04-04 10:02:38 +08:00
/// </summary>
/// <returns>Deserialized json settings object.</returns>
2020-09-22 01:14:44 +08:00
public T GetSettings < T > ( string powertoy = DefaultModuleName , string fileName = DefaultFileName )
2020-09-24 04:20:32 +08:00
where T : ISettingsConfig , new ( )
{
if ( SettingsExists ( powertoy , fileName ) )
{
2020-10-27 02:13:53 +08:00
try
2020-09-24 04:20:32 +08:00
{
2020-10-27 02:13:53 +08:00
// Given the file already exists, to deserialize the file and read it's content.
T deserializedSettings = GetFile < T > ( powertoy , fileName ) ;
// If the file needs to be modified, to save the new configurations accordingly.
if ( deserializedSettings . UpgradeSettingsConfiguration ( ) )
{
SaveSettings ( deserializedSettings . ToJsonString ( ) , powertoy , fileName ) ;
}
return deserializedSettings ;
2020-09-24 04:20:32 +08:00
}
2020-10-27 02:13:53 +08:00
// Catch json deserialization exceptions when the file is corrupt and has an invalid json.
// If there are any deserialization issues like in https://github.com/microsoft/PowerToys/issues/7500, log the error and create a new settings.json file.
// This is different from the case where we have trailing zeros following a valid json file, which we have handled by trimming the trailing zeros.
catch ( JsonException ex )
{
Logger . LogError ( $"Exception encountered while loading {powertoy} settings." , ex ) ;
}
2020-09-24 04:20:32 +08:00
}
2020-10-27 02:13:53 +08:00
// If the settings file does not exist or if the file is corrupt, to create a new object with default parameters and save it to a newly created settings file.
T newSettingsItem = new T ( ) ;
SaveSettings ( newSettingsItem . ToJsonString ( ) , powertoy , fileName ) ;
return newSettingsItem ;
2020-09-24 04:20:32 +08:00
}
// Given the powerToy folder name and filename to be accessed, this function deserializes and returns the file.
private T GetFile < T > ( string powertoyFolderName = DefaultModuleName , string fileName = DefaultFileName )
2020-03-25 10:55:02 +08:00
{
2020-09-12 07:15:18 +08:00
// Adding Trim('\0') to overcome possible NTFS file corruption.
// Look at issue https://github.com/microsoft/PowerToys/issues/6413 you'll see the file has a large sum of \0 to fill up a 4096 byte buffer for writing to disk
// This, while not totally ideal, does work around the problem by trimming the end.
// The file itself did write the content correctly but something is off with the actual end of the file, hence the 0x00 bug
2020-09-24 04:20:32 +08:00
var jsonSettingsString = _ioProvider . ReadAllText ( GetSettingsPath ( powertoyFolderName , fileName ) ) . Trim ( '\0' ) ;
2020-03-25 10:55:02 +08:00
return JsonSerializer . Deserialize < T > ( jsonSettingsString ) ;
}
2020-04-08 01:19:14 +08:00
// Save settings to a json file.
2020-10-22 03:32:53 +08:00
[SuppressMessage("Design", "CA1031:Do not catch general exception types", Justification = "General exceptions will be logged until we can better understand runtime exception scenarios")]
2020-09-22 01:14:44 +08:00
public void SaveSettings ( string jsonSettings , string powertoy = DefaultModuleName , string fileName = DefaultFileName )
2020-03-25 10:55:02 +08:00
{
2020-04-18 06:25:08 +08:00
try
2020-04-04 10:02:38 +08:00
{
2020-04-18 06:25:08 +08:00
if ( jsonSettings ! = null )
2020-04-04 10:02:38 +08:00
{
2020-04-18 06:25:08 +08:00
if ( ! SettingsFolderExists ( powertoy ) )
{
CreateSettingsFolder ( powertoy ) ;
}
2020-04-08 15:19:00 +08:00
2020-09-22 01:14:44 +08:00
_ioProvider . WriteAllText ( GetSettingsPath ( powertoy , fileName ) , jsonSettings ) ;
2020-04-18 06:25:08 +08:00
}
}
2020-10-22 03:32:53 +08:00
catch ( Exception e )
2020-04-18 06:25:08 +08:00
{
2020-10-22 03:32:53 +08:00
Logger . LogError ( $"Exception encountered while saving {powertoy} settings." , e ) ;
#if DEBUG
if ( e is ArgumentException | | e is ArgumentNullException | | e is PathTooLongException )
{
2020-10-23 00:45:48 +08:00
throw ;
2020-10-22 03:32:53 +08:00
}
#endif
2020-04-04 10:02:38 +08:00
}
2020-03-25 10:55:02 +08:00
}
2020-04-08 15:19:00 +08:00
2020-09-22 01:14:44 +08:00
private static string LocalApplicationDataFolder ( )
2020-04-08 15:19:00 +08:00
{
return Environment . GetFolderPath ( Environment . SpecialFolder . LocalApplicationData ) ;
}
2020-03-25 10:55:02 +08:00
}
2020-05-06 05:28:44 +08:00
}