mirror of
https://github.com/microsoft/PowerToys.git
synced 2026-04-06 19:26:39 +02:00
* [PTRun] Implemented a new JSON storage method for PTRun settings files. * [PTRun] Removed uncessary parts. * [PTRun] Spell checks. * [PTRun] New JsonSerializerOptions added for information files. * [PTRun] Unnecessary null check is removed. * [PT Run] - ExtractFields function removed. - Creating instance is used instead of deserializing. * [PTRun] Build fix * [PTRun] Removed unncessary parts * [PTRun] CheckWithInformationFileToClear reversed. * [PTRun] Build fix. * [PTRun] Deserialization is used instead of key by key comparison. * [PTRun] Removed unncessary parts. * [PTRun] Removed unncessary parts. * [PTRun] Remove entry if query is null or empty.
184 lines
5.8 KiB
C#
184 lines
5.8 KiB
C#
// 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;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.IO.Abstractions;
|
|
using System.Linq;
|
|
using System.Text.Json;
|
|
using Wox.Plugin.Logger;
|
|
|
|
namespace Wox.Infrastructure.Storage
|
|
{
|
|
/// <summary>
|
|
/// Serialize object using json format.
|
|
/// </summary>
|
|
public class JsonStorage<T>
|
|
{
|
|
private static readonly IFileSystem FileSystem = new FileSystem();
|
|
private static readonly IPath Path = FileSystem.Path;
|
|
private static readonly IFile File = FileSystem.File;
|
|
|
|
private readonly object _saveLock = new object();
|
|
|
|
// use property initialization instead of DefaultValueAttribute
|
|
// easier and flexible for default value of object
|
|
private static readonly JsonSerializerOptions _serializerOptions = new JsonSerializerOptions
|
|
{
|
|
DefaultIgnoreCondition = System.Text.Json.Serialization.JsonIgnoreCondition.WhenWritingNull,
|
|
IncludeFields = true,
|
|
PropertyNameCaseInsensitive = true,
|
|
WriteIndented = true,
|
|
UnmappedMemberHandling = System.Text.Json.Serialization.JsonUnmappedMemberHandling.Disallow,
|
|
};
|
|
|
|
private static readonly JsonSerializerOptions _deserializerOptions = new JsonSerializerOptions
|
|
{
|
|
DefaultIgnoreCondition = System.Text.Json.Serialization.JsonIgnoreCondition.Never,
|
|
IncludeFields = true,
|
|
PropertyNameCaseInsensitive = true,
|
|
WriteIndented = true,
|
|
};
|
|
|
|
private T _data;
|
|
|
|
// need a new directory name
|
|
public const string DirectoryName = "Settings";
|
|
public const string FileSuffix = ".json";
|
|
|
|
public string FilePath { get; set; }
|
|
|
|
public string DirectoryPath { get; set; }
|
|
|
|
// This storage helper returns whether or not to delete the json storage items
|
|
private const int _jsonStorage = 1;
|
|
private StoragePowerToysVersionInfo _storageHelper;
|
|
|
|
public virtual T Load()
|
|
{
|
|
_storageHelper = new StoragePowerToysVersionInfo(FilePath, _jsonStorage);
|
|
|
|
if (File.Exists(FilePath))
|
|
{
|
|
var serialized = File.ReadAllText(FilePath);
|
|
if (!string.IsNullOrWhiteSpace(serialized))
|
|
{
|
|
Deserialize(serialized);
|
|
}
|
|
else
|
|
{
|
|
LoadDefault();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
LoadDefault();
|
|
}
|
|
|
|
return _data.NonNull();
|
|
}
|
|
|
|
private void Deserialize(string serialized)
|
|
{
|
|
try
|
|
{
|
|
_data = JsonSerializer.Deserialize<T>(serialized, _deserializerOptions);
|
|
}
|
|
catch (JsonException e)
|
|
{
|
|
LoadDefault();
|
|
Log.Exception($"Deserialize error for json <{FilePath}>", e, GetType());
|
|
}
|
|
|
|
if (_data == null)
|
|
{
|
|
LoadDefault();
|
|
}
|
|
}
|
|
|
|
private void LoadDefault()
|
|
{
|
|
if (File.Exists(FilePath))
|
|
{
|
|
BackupOriginFile();
|
|
}
|
|
|
|
_data = JsonSerializer.Deserialize<T>("{}", _serializerOptions);
|
|
Save();
|
|
}
|
|
|
|
private void BackupOriginFile()
|
|
{
|
|
// Using InvariantCulture since this is internal
|
|
var timestamp = DateTime.Now.ToString("yyyy-MM-dd-HH-mm-ss-fffffff", CultureInfo.InvariantCulture);
|
|
var directory = Path.GetDirectoryName(FilePath).NonNull();
|
|
var originName = Path.GetFileNameWithoutExtension(FilePath);
|
|
var backupName = $"{originName}-{timestamp}{FileSuffix}";
|
|
var backupPath = Path.Combine(directory, backupName);
|
|
File.Copy(FilePath, backupPath, true);
|
|
|
|
// todo give user notification for the backup process
|
|
}
|
|
|
|
public void Save()
|
|
{
|
|
lock (_saveLock)
|
|
{
|
|
try
|
|
{
|
|
string serialized = JsonSerializer.Serialize(_data, _serializerOptions);
|
|
File.WriteAllText(FilePath, serialized);
|
|
_storageHelper.Close();
|
|
|
|
Log.Info($"Saving cached data at <{FilePath}>", GetType());
|
|
}
|
|
catch (IOException e)
|
|
{
|
|
Log.Exception($"Error in saving data at <{FilePath}>", e, GetType());
|
|
}
|
|
}
|
|
}
|
|
|
|
public void Clear()
|
|
{
|
|
if (File.Exists(FilePath))
|
|
{
|
|
File.Delete(FilePath);
|
|
LoadDefault();
|
|
Log.Info($"Deleting cached data at <{FilePath}>", GetType());
|
|
}
|
|
}
|
|
|
|
public bool CheckVersionMismatch()
|
|
{
|
|
// Skip the fields check if the version hasn't changed.
|
|
// This optimization prevents unnecessary fields processing when the cache
|
|
// is already up to date, enhancing performance and reducing IO operations
|
|
if (!_storageHelper.ClearCache)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
_storageHelper.ClearCache = false;
|
|
return true;
|
|
}
|
|
|
|
public bool TryLoadData()
|
|
{
|
|
try
|
|
{
|
|
JsonSerializer.Deserialize<T>(File.ReadAllText(FilePath), _serializerOptions);
|
|
return true;
|
|
}
|
|
catch (JsonException e)
|
|
{
|
|
Log.Exception($"Error in TryLoadData at <{FilePath}>", e, GetType());
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|