- Removed some unneeded interface definitions. - Clean-slated some Services that need to be re-written.
654 lines
30 KiB
C#
654 lines
30 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Collections.Immutable;
|
|
using System.IO;
|
|
using System.Reflection;
|
|
using System.Security;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using System.Xml.Linq;
|
|
using Barotrauma.LuaCs.Configuration;
|
|
using Barotrauma.LuaCs.Networking;
|
|
using Barotrauma.Steam;
|
|
using FluentResults;
|
|
using FluentResults.LuaCs;
|
|
using Microsoft.CodeAnalysis;
|
|
using OneOf.Types;
|
|
using Error = FluentResults.Error;
|
|
using File = Barotrauma.IO.File;
|
|
using Path = Barotrauma.IO.Path;
|
|
using Success = OneOf.Types.Success;
|
|
|
|
namespace Barotrauma.LuaCs.Services;
|
|
|
|
public class StorageService : IStorageService
|
|
{
|
|
|
|
public StorageService(Lazy<IConfigService> configService)
|
|
{
|
|
_configService = configService;
|
|
}
|
|
private readonly Lazy<IConfigService> _configService;
|
|
private IConfigEntry<string> _kLocalStoragePath = null;
|
|
private IConfigEntry<string> _kLocalFilePathRules = null;
|
|
private const string _packagePathKeyword = "<PACKNAME>";
|
|
private readonly string _runLocation = Path.GetDirectoryName(Assembly.GetEntryAssembly()!.Location.CleanUpPath());
|
|
|
|
// TODO: Rewrite the config to get info from .ctor.
|
|
private IConfigEntry<string> LocalStoragePath => _kLocalStoragePath ??= GetOrCreateConfig(nameof(LocalStoragePath), "/Data/Mods");
|
|
private IConfigEntry<string> LocalFilePathRule => _kLocalFilePathRules ??= GetOrCreateConfig(nameof(LocalFilePathRule), _packagePathKeyword);
|
|
private IConfigEntry<string> GetOrCreateConfig(string name, string defaultValue)
|
|
{
|
|
var c = _configService.Value
|
|
.GetConfig<IConfigEntry<string>>(ModUtils.Definitions.LuaCsForBarotrauma, name);
|
|
if (c is not null)
|
|
return c;
|
|
|
|
var c1 = _configService.Value.AddConfigEntry(
|
|
ModUtils.Definitions.LuaCsForBarotrauma,
|
|
name, defaultValue, NetSync.None, valueChangePredicate: (value) => false);
|
|
if (c1.IsSuccess)
|
|
return c1.Value;
|
|
|
|
throw new KeyNotFoundException("Cannot find storage value for key: " + name);
|
|
|
|
}
|
|
public bool IsDisposed { get; private set; }
|
|
|
|
public void Dispose()
|
|
{
|
|
if (IsDisposed)
|
|
return;
|
|
IsDisposed = true;
|
|
_kLocalStoragePath = null;
|
|
_kLocalFilePathRules = null;
|
|
}
|
|
|
|
public FluentResults.Result<XDocument> LoadLocalXml(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? TryLoadXml(r.Value) : r.ToResult();
|
|
public FluentResults.Result<byte[]> LoadLocalBinary(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? TryLoadBinary(r.Value) : r.ToResult();
|
|
public FluentResults.Result<string> LoadLocalText(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? TryLoadText(r.Value) : r.ToResult();
|
|
public FluentResults.Result SaveLocalXml(ContentPackage package, string localFilePath, XDocument document) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? TrySaveXml(r.Value, document) : r.ToResult();
|
|
public FluentResults.Result SaveLocalBinary(ContentPackage package, string localFilePath, in byte[] bytes) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? TrySaveBinary(r.Value, bytes) : r.ToResult();
|
|
public FluentResults.Result SaveLocalText(ContentPackage package, string localFilePath, in string text) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? TrySaveText(r.Value, text) : r.ToResult();
|
|
public async Task<FluentResults.Result<XDocument>> LoadLocalXmlAsync(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? await TryLoadXmlAsync(r.Value) : r.ToResult();
|
|
public async Task<FluentResults.Result<byte[]>> LoadLocalBinaryAsync(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? await TryLoadBinaryAsync(r.Value) : r.ToResult();
|
|
public async Task<FluentResults.Result<string>> LoadLocalTextAsync(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? await TryLoadTextAsync(r.Value) : r.ToResult();
|
|
public async Task<FluentResults.Result> SaveLocalXmlAsync(ContentPackage package, string localFilePath, XDocument document) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? await TrySaveXmlAsync(r.Value, document) : r.ToResult();
|
|
public async Task<FluentResults.Result> SaveLocalBinaryAsync(ContentPackage package, string localFilePath, byte[] bytes) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? await TrySaveBinaryAsync(r.Value, bytes) : r.ToResult();
|
|
public async Task<FluentResults.Result> SaveLocalTextAsync(ContentPackage package, string localFilePath, string text) =>
|
|
GetAbsFromLocal(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? await TrySaveTextAsync(r.Value, text) : r.ToResult();
|
|
public FluentResults.Result<XDocument> LoadPackageXml(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromPackage(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? TryLoadXml(r.Value) : r.ToResult();
|
|
public FluentResults.Result<byte[]> LoadPackageBinary(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromPackage(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? TryLoadBinary(r.Value) : r.ToResult();
|
|
public FluentResults.Result<string> LoadPackageText(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromPackage(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? TryLoadText(r.Value) : r.ToResult();
|
|
|
|
public ImmutableArray<(string, FluentResults.Result<XDocument>)> LoadPackageXmlFiles(ContentPackage package, ImmutableArray<string> localFilePaths)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (localFilePaths.IsDefaultOrEmpty)
|
|
return ImmutableArray<(string, FluentResults.Result<XDocument>)>.Empty;
|
|
var builder = ImmutableArray.CreateBuilder<(string, FluentResults.Result<XDocument>)>(localFilePaths.Length);
|
|
foreach (var path in localFilePaths)
|
|
builder.Add((path, LoadPackageXml(package, path)));
|
|
return builder.MoveToImmutable();
|
|
}
|
|
|
|
public ImmutableArray<(string, FluentResults.Result<byte[]>)> LoadPackageBinaryFiles(ContentPackage package, ImmutableArray<string> localFilePaths)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (localFilePaths.IsDefaultOrEmpty)
|
|
return ImmutableArray<(string, FluentResults.Result<byte[]>)>.Empty;
|
|
var builder = ImmutableArray.CreateBuilder<(string, FluentResults.Result<byte[]>)>(localFilePaths.Length);
|
|
foreach (var path in localFilePaths)
|
|
builder.Add((path, LoadPackageBinary(package, path)));
|
|
return builder.MoveToImmutable();
|
|
}
|
|
|
|
public ImmutableArray<(string, FluentResults.Result<string>)> LoadPackageTextFiles(ContentPackage package, ImmutableArray<string> localFilePaths)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (localFilePaths.IsDefaultOrEmpty)
|
|
return ImmutableArray<(string, FluentResults.Result<string>)>.Empty;
|
|
var builder = ImmutableArray.CreateBuilder<(string, FluentResults.Result<string>)>(localFilePaths.Length);
|
|
foreach (var path in localFilePaths)
|
|
builder.Add((path, LoadPackageText(package, path)));
|
|
return builder.MoveToImmutable();
|
|
}
|
|
|
|
public FluentResults.Result<ImmutableArray<string>> FindFilesInPackage(ContentPackage package, string localSubfolder, string regexFilter, bool searchRecursively)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
var r = GetAbsFromPackage(package, localSubfolder);
|
|
if (r is { IsFailed: true })
|
|
return r.ToResult();
|
|
var builder = ImmutableArray.CreateBuilder<(string, FluentResults.Result<ImmutableArray<string>>)>();
|
|
var sOption = searchRecursively ? SearchOption.AllDirectories : SearchOption.TopDirectoryOnly;
|
|
string[] arr = Directory.GetFiles(localSubfolder, regexFilter.IsNullOrWhiteSpace() ? "*.*" : regexFilter, sOption);
|
|
return new FluentResults.Result<ImmutableArray<string>>().WithSuccess($"Files found.")
|
|
.WithValue(arr.ToImmutableArray());
|
|
}
|
|
|
|
public async Task<FluentResults.Result<XDocument>> LoadPackageXmlAsync(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromPackage(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? await TryLoadXmlAsync(r.Value) : r.ToResult();
|
|
|
|
public async Task<FluentResults.Result<byte[]>> LoadPackageBinaryAsync(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromPackage(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? await TryLoadBinaryAsync(r.Value) : r.ToResult();
|
|
|
|
public async Task<FluentResults.Result<string>> LoadPackageTextAsync(ContentPackage package, string localFilePath) =>
|
|
GetAbsFromPackage(package, localFilePath) is var r && r is { IsSuccess: true, Value: not null }
|
|
? await TryLoadTextAsync(r.Value) : r.ToResult();
|
|
|
|
public async Task<ImmutableArray<(string, FluentResults.Result<XDocument>)>> LoadPackageXmlFilesAsync(ContentPackage package, ImmutableArray<string> localFilePaths)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (localFilePaths.IsDefaultOrEmpty)
|
|
return ImmutableArray<(string, FluentResults.Result<XDocument>)>.Empty;
|
|
var builder = ImmutableArray.CreateBuilder<(string, FluentResults.Result<XDocument>)>(localFilePaths.Length);
|
|
foreach (var path in localFilePaths)
|
|
builder.Add((path, await LoadPackageXmlAsync(package, path)));
|
|
return builder.MoveToImmutable();
|
|
}
|
|
|
|
public async Task<ImmutableArray<(string, FluentResults.Result<byte[]>)>> LoadPackageBinaryFilesAsync(ContentPackage package, ImmutableArray<string> localFilePaths)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (localFilePaths.IsDefaultOrEmpty)
|
|
return ImmutableArray<(string, FluentResults.Result<byte[]>)>.Empty;
|
|
var builder = ImmutableArray.CreateBuilder<(string, FluentResults.Result<byte[]>)>(localFilePaths.Length);
|
|
foreach (var path in localFilePaths)
|
|
builder.Add((path, await LoadPackageBinaryAsync(package, path)));
|
|
return builder.MoveToImmutable();
|
|
}
|
|
|
|
public async Task<ImmutableArray<(string, FluentResults.Result<string>)>> LoadPackageTextFilesAsync(ContentPackage package, ImmutableArray<string> localFilePaths)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (localFilePaths.IsDefaultOrEmpty)
|
|
return ImmutableArray<(string, FluentResults.Result<string>)>.Empty;
|
|
var builder = ImmutableArray.CreateBuilder<(string, FluentResults.Result<string>)>(localFilePaths.Length);
|
|
foreach (var path in localFilePaths)
|
|
builder.Add((path, await LoadPackageTextAsync(package, path)));
|
|
return builder.MoveToImmutable();
|
|
}
|
|
|
|
|
|
public FluentResults.Result<XDocument> TryLoadXml(string filePath, Encoding encoding = null)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
var r = TryLoadText(filePath, encoding);
|
|
if (r is { IsSuccess: true, Value: not null })
|
|
return XDocument.Parse(r.Value);
|
|
else
|
|
{
|
|
return r.ToResult<XDocument>(s => null)
|
|
.WithError(GetGeneralError(nameof(LoadLocalXml), filePath));
|
|
}
|
|
}
|
|
|
|
public FluentResults.Result<string> TryLoadText(string filePath, Encoding encoding = null)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
return IOExceptionsOperationRunner(nameof(TryLoadText), filePath, () =>
|
|
{
|
|
var fp = filePath.CleanUpPath();
|
|
fp = System.IO.Path.IsPathRooted(fp) ? fp : System.IO.Path.GetFullPath(fp);
|
|
var fileText = encoding is null ? System.IO.File.ReadAllText(fp) : System.IO.File.ReadAllText(fp, encoding);
|
|
return new FluentResults.Result<string>().WithSuccess($"Loaded file successfully").WithValue(fileText);
|
|
});
|
|
}
|
|
|
|
public FluentResults.Result<byte[]> TryLoadBinary(string filePath)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
return IOExceptionsOperationRunner(nameof(TryLoadBinary), filePath, () =>
|
|
{
|
|
var fp = filePath.CleanUpPath();
|
|
fp = System.IO.Path.IsPathRooted(fp) ? fp : System.IO.Path.GetFullPath(fp);
|
|
var fileData = System.IO.File.ReadAllBytes(fp);
|
|
return new FluentResults.Result<byte[]>().WithSuccess($"Loaded file successfully").WithValue(fileData);
|
|
});
|
|
}
|
|
|
|
public FluentResults.Result TrySaveXml(string filePath, in XDocument document, Encoding encoding = null) => TrySaveText(filePath, document.ToString(), encoding);
|
|
public FluentResults.Result TrySaveText(string filePath, in string text, Encoding encoding = null)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (text.IsNullOrWhiteSpace())
|
|
{
|
|
return FluentResults.Result.Fail($"Contents are empty for {filePath}")
|
|
.WithError(new Error($"Contents are empty for {filePath}")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.Sources, filePath));
|
|
}
|
|
|
|
string t = text; //copy
|
|
return IOExceptionsOperationRunner(nameof(TrySaveText), filePath, () =>
|
|
{
|
|
var fp = filePath.CleanUpPath();
|
|
fp = System.IO.Path.IsPathRooted(fp) ? fp : System.IO.Path.GetFullPath(fp);
|
|
System.IO.File.WriteAllText(fp, t, encoding);
|
|
return new FluentResults.Result().WithSuccess($"Saved to file successfully");
|
|
});
|
|
}
|
|
|
|
public FluentResults.Result TrySaveBinary(string filePath, in byte[] bytes)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (bytes is null || bytes.Length == 0)
|
|
{
|
|
return FluentResults.Result.Fail($"Byte array is null or empty for {filePath}")
|
|
.WithError(new Error($"Byte array is null or empty for {filePath}")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.Sources, filePath));
|
|
}
|
|
byte[] b = new byte[bytes.Length];
|
|
System.Buffer.BlockCopy(bytes, 0, b, 0, bytes.Length);
|
|
return IOExceptionsOperationRunner(nameof(TrySaveBinary), filePath, () =>
|
|
{
|
|
var fp = filePath.CleanUpPath();
|
|
fp = System.IO.Path.IsPathRooted(fp) ? fp : System.IO.Path.GetFullPath(fp);
|
|
System.IO.File.WriteAllBytes(fp, b);
|
|
return new FluentResults.Result().WithSuccess($"Saved to file successfully");
|
|
});
|
|
}
|
|
|
|
public FluentResults.Result<bool> FileExists(string filePath)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
return IOExceptionsOperationRunner<bool>(nameof(FileExists), filePath, () =>
|
|
{
|
|
var fp = filePath.CleanUpPath();
|
|
fp = System.IO.Path.IsPathRooted(fp) ? fp : System.IO.Path.GetFullPath(fp);
|
|
return System.IO.File.Exists(fp);
|
|
});
|
|
}
|
|
|
|
public async Task<FluentResults.Result<XDocument>> TryLoadXmlAsync(string filePath, Encoding encoding = null)
|
|
{
|
|
var r = await TryLoadTextAsync(filePath, encoding);
|
|
if (r is { IsSuccess: true, Value: {} value } && !value.IsNullOrWhiteSpace())
|
|
return XDocument.Parse(value);
|
|
return FluentResults.Result.Fail<XDocument>(GetGeneralError(nameof(TryLoadXml), filePath));
|
|
}
|
|
|
|
public async Task<FluentResults.Result<string>> TryLoadTextAsync(string filePath, Encoding encoding = null)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
return await IOExceptionsOperationRunnerAsync<string>(nameof(TryLoadTextAsync), filePath, async () =>
|
|
{
|
|
var fp = filePath.CleanUpPath();
|
|
fp = System.IO.Path.IsPathRooted(fp) ? fp : System.IO.Path.GetFullPath(fp);
|
|
return await System.IO.File.ReadAllTextAsync(fp);
|
|
});
|
|
}
|
|
|
|
public async Task<FluentResults.Result<byte[]>> TryLoadBinaryAsync(string filePath)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
return await IOExceptionsOperationRunnerAsync<byte[]>(nameof(TryLoadTextAsync), filePath, async () =>
|
|
{
|
|
var fp = filePath.CleanUpPath();
|
|
fp = System.IO.Path.IsPathRooted(fp) ? fp : System.IO.Path.GetFullPath(fp);
|
|
return await System.IO.File.ReadAllBytesAsync(fp);
|
|
});
|
|
}
|
|
|
|
public async Task<FluentResults.Result> TrySaveXmlAsync(string filePath, XDocument document, Encoding encoding = null) => await TrySaveTextAsync(filePath, document.ToString(), encoding);
|
|
public async Task<FluentResults.Result> TrySaveTextAsync(string filePath, string text, Encoding encoding = null)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (text.IsNullOrWhiteSpace())
|
|
{
|
|
return FluentResults.Result.Fail($"Contents are empty for {filePath}")
|
|
.WithError(new Error($"Contents are empty for {filePath}")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.Sources, filePath));
|
|
}
|
|
|
|
string t = text.ToString(); //copy
|
|
return await IOExceptionsOperationRunnerAsync(nameof(TrySaveText), filePath, async () =>
|
|
{
|
|
var fp = filePath.CleanUpPath();
|
|
fp = System.IO.Path.IsPathRooted(fp) ? fp : System.IO.Path.GetFullPath(fp);
|
|
await System.IO.File.WriteAllTextAsync(fp, t, encoding);
|
|
return new FluentResults.Result().WithSuccess($"Saved to file successfully");
|
|
});
|
|
}
|
|
|
|
public async Task<FluentResults.Result> TrySaveBinaryAsync(string filePath, byte[] bytes)
|
|
{
|
|
((IService)this).CheckDisposed();
|
|
if (bytes is null || bytes.Length == 0)
|
|
{
|
|
return FluentResults.Result.Fail($"Byte array is null or empty for {filePath}")
|
|
.WithError(new Error($"Byte array is null or empty for {filePath}")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.Sources, filePath));
|
|
}
|
|
byte[] b = new byte[bytes.Length];
|
|
System.Buffer.BlockCopy(bytes, 0, b, 0, bytes.Length);
|
|
return await IOExceptionsOperationRunnerAsync(nameof(TrySaveBinary), filePath, async () =>
|
|
{
|
|
var fp = filePath.CleanUpPath();
|
|
fp = System.IO.Path.IsPathRooted(fp) ? fp : System.IO.Path.GetFullPath(fp);
|
|
await System.IO.File.WriteAllBytesAsync(fp, b);
|
|
return new FluentResults.Result().WithSuccess($"Saved to file successfully");
|
|
});
|
|
}
|
|
|
|
private async Task<FluentResults.Result<T>> IOExceptionsOperationRunnerAsync<T>(string funcName, string filepath, Func<Task<FluentResults.Result<T>>> operation)
|
|
{
|
|
try
|
|
{
|
|
return await operation?.Invoke()!;
|
|
}
|
|
catch (ArgumentNullException ane)
|
|
{
|
|
return ReturnException(ane, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (ArgumentException ae)
|
|
{
|
|
return ReturnException(ae, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (PathTooLongException ptle)
|
|
{
|
|
return ReturnException(ptle, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (NotSupportedException nse)
|
|
{
|
|
return ReturnException(nse, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (UnauthorizedAccessException uae)
|
|
{
|
|
return ReturnException(uae, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (DirectoryNotFoundException dnfe)
|
|
{
|
|
return ReturnException(dnfe, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (FileNotFoundException fnfe)
|
|
{
|
|
return ReturnException(fnfe, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (SecurityException se)
|
|
{
|
|
return ReturnException(se, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (IOException ioe)
|
|
{
|
|
return ReturnException(ioe, filepath).WithError(GetGeneralError(nameof(SaveLocalXml), filepath));
|
|
}
|
|
}
|
|
|
|
private async Task<FluentResults.Result> IOExceptionsOperationRunnerAsync(string funcName, string filepath, Func<Task<FluentResults.Result>> operation)
|
|
{
|
|
try
|
|
{
|
|
return await operation?.Invoke()!;
|
|
}
|
|
catch (ArgumentNullException ane)
|
|
{
|
|
return ReturnException(ane, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (ArgumentException ae)
|
|
{
|
|
return ReturnException(ae, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (PathTooLongException ptle)
|
|
{
|
|
return ReturnException(ptle, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (NotSupportedException nse)
|
|
{
|
|
return ReturnException(nse, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (UnauthorizedAccessException uae)
|
|
{
|
|
return ReturnException(uae, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (DirectoryNotFoundException dnfe)
|
|
{
|
|
return ReturnException(dnfe, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (FileNotFoundException fnfe)
|
|
{
|
|
return ReturnException(fnfe, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (SecurityException se)
|
|
{
|
|
return ReturnException(se, filepath).WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (IOException ioe)
|
|
{
|
|
return ReturnException(ioe, filepath).WithError(GetGeneralError(nameof(SaveLocalXml), filepath));
|
|
}
|
|
}
|
|
|
|
private FluentResults.Result<T> IOExceptionsOperationRunner<T>(string funcName, string filepath, Func<FluentResults.Result<T>> operation)
|
|
{
|
|
try
|
|
{
|
|
return operation?.Invoke();
|
|
}
|
|
catch (ArgumentNullException ane)
|
|
{
|
|
return ReturnException(ane, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (ArgumentException ae)
|
|
{
|
|
return ReturnException(ae, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (PathTooLongException ptle)
|
|
{
|
|
return ReturnException(ptle, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (NotSupportedException nse)
|
|
{
|
|
return ReturnException(nse, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (UnauthorizedAccessException uae)
|
|
{
|
|
return ReturnException(uae, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (DirectoryNotFoundException dnfe)
|
|
{
|
|
return ReturnException(dnfe, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (FileNotFoundException fnfe)
|
|
{
|
|
return ReturnException(fnfe, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (SecurityException se)
|
|
{
|
|
return ReturnException(se, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (IOException ioe)
|
|
{
|
|
return ReturnException(ioe, filepath)
|
|
.WithError(GetGeneralError(nameof(SaveLocalXml), filepath));
|
|
}
|
|
}
|
|
|
|
private FluentResults.Result IOExceptionsOperationRunner(string funcName, string filepath, Func<FluentResults.Result> operation)
|
|
{
|
|
try
|
|
{
|
|
return operation?.Invoke();
|
|
}
|
|
catch (ArgumentNullException ane)
|
|
{
|
|
return ReturnException(ane, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (ArgumentException ae)
|
|
{
|
|
return ReturnException(ae, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (PathTooLongException ptle)
|
|
{
|
|
return ReturnException(ptle, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (NotSupportedException nse)
|
|
{
|
|
return ReturnException(nse, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (UnauthorizedAccessException uae)
|
|
{
|
|
return ReturnException(uae, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (DirectoryNotFoundException dnfe)
|
|
{
|
|
return ReturnException(dnfe, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (FileNotFoundException fnfe)
|
|
{
|
|
return ReturnException(fnfe, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (SecurityException se)
|
|
{
|
|
return ReturnException(se, filepath)
|
|
.WithError(GetGeneralError(funcName, filepath));
|
|
}
|
|
catch (IOException ioe)
|
|
{
|
|
return ReturnException(ioe, filepath)
|
|
.WithError(GetGeneralError(nameof(SaveLocalXml), filepath));
|
|
}
|
|
}
|
|
|
|
private Error GetGeneralError(string funcName, string localfp, ContentPackage package) =>
|
|
new Error($"{funcName}: Failed to load local file.")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.Sources, localfp)
|
|
.WithMetadata(MetadataType.RootObject, package);
|
|
|
|
private Error GetGeneralError(string funcName, string localfp) =>
|
|
new Error($"{funcName}: Failed to load local file.")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.Sources, localfp);
|
|
|
|
private FluentResults.Result<string> GetAbsFromLocal(ContentPackage package, string localFilePath)
|
|
{
|
|
if (Path.IsPathRooted(localFilePath))
|
|
{
|
|
return new FluentResults.Result<string>().WithError(
|
|
new Error($"The path '{localFilePath}' is a rooted path. Must be relative!")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.RootObject, localFilePath));
|
|
}
|
|
|
|
if (package is null)
|
|
{
|
|
return new FluentResults.Result<string>().WithError(
|
|
new Error($"{nameof(GetAbsFromPackage)} The package reference for {localFilePath} is null!")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.RootObject, localFilePath));
|
|
}
|
|
|
|
return new FluentResults.Result<string>().WithSuccess($"Path constructed")
|
|
.WithValue( System.IO.Path.GetFullPath(System.IO.Path.Combine(
|
|
_runLocation,
|
|
LocalStoragePath.Value,
|
|
LocalFilePathRule.Value.Replace(_packagePathKeyword, package.Name.IsNullOrWhiteSpace()
|
|
? package.TryExtractSteamWorkshopId(out var id)
|
|
? id.Value.ToString()
|
|
: "_fallbackFolder"
|
|
: package.Name),
|
|
localFilePath)));
|
|
}
|
|
|
|
private FluentResults.Result<string> GetAbsFromPackage(ContentPackage package, string localFilePath)
|
|
{
|
|
if (package is null)
|
|
{
|
|
return new FluentResults.Result<string>().WithError(
|
|
new Error($"{nameof(GetAbsFromPackage)} The package reference for {localFilePath} is null!")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.RootObject, localFilePath));
|
|
}
|
|
|
|
if (localFilePath.IsNullOrWhiteSpace())
|
|
{
|
|
return new FluentResults.Result<string>().WithValue(Path.GetFullPath(package.Path.CleanUpPath()));
|
|
}
|
|
|
|
var path = localFilePath.CleanUpPath();
|
|
|
|
if (Path.IsPathRooted(path))
|
|
{
|
|
return new FluentResults.Result<string>().WithError(
|
|
new Error($"The path '{localFilePath}' is a rooted path. Must be relative!")
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.RootObject, localFilePath));
|
|
}
|
|
|
|
return new FluentResults.Result<string>().WithSuccess($"Path constructed")
|
|
.WithValue(Path.Combine(Path.GetFullPath(package.Path.CleanUpPath()), path));
|
|
}
|
|
|
|
private FluentResults.Result<TReturn> ReturnException<TReturn, TException>(TException exception, ContentPackage package) where TException : Exception
|
|
{
|
|
return new FluentResults.Result<TReturn>().WithError(new ExceptionalError(exception)
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.RootObject, package));
|
|
}
|
|
|
|
private FluentResults.Result ReturnException<TException>(TException exception, ContentPackage package) where TException : Exception
|
|
{
|
|
return new FluentResults.Result().WithError(new ExceptionalError(exception)
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.RootObject, package));
|
|
}
|
|
|
|
private FluentResults.Result ReturnException<TException>(TException exception, string filePath) where TException : Exception
|
|
{
|
|
return new FluentResults.Result().WithError(new ExceptionalError(exception)
|
|
.WithMetadata(MetadataType.ExceptionObject, this)
|
|
.WithMetadata(MetadataType.RootObject, filePath));
|
|
}
|
|
}
|