From 749000a090b70bbfd8f18a45c4dd0e8f3c767b78 Mon Sep 17 00:00:00 2001 From: Pavel Gross Date: Mon, 18 Nov 2024 16:57:20 +0300 Subject: [PATCH] [#28] Client: Apply code optimizations Signed-off-by: Pavel Gross --- .editorconfig | 4 +- src/FrostFS.SDK.Client/CllientKey.cs | 4 + .../FrostFS.SDK.Client.csproj | 4 + src/FrostFS.SDK.Client/FrostFSClient.cs | 340 +++++++++--------- .../Interfaces/IFrostFSClient.cs | 4 +- src/FrostFS.SDK.Client/Mappers/ContainerId.cs | 2 +- .../Mappers/Netmap/NodeInfo.cs | 2 +- .../Mappers/Object/Object.cs | 2 +- .../Mappers/Object/ObjectHeaderMapper.cs | 2 +- .../Mappers/Object/ObjectId.cs | 2 +- src/FrostFS.SDK.Client/Mappers/OwnerId.cs | 2 +- .../Models/Client/ClientSettings.cs | 63 +--- .../Models/Containers/FrostFsContainerId.cs | 2 +- .../Models/Containers/FrostFsContainerInfo.cs | 2 +- .../Models/Netmap/FrostFsVersion.cs | 2 +- .../Models/Object/FrostFsObjectId.cs | 7 +- .../Models/Object/FrostFsOwner.cs | 1 + .../Models/Session/FrostFsSessionToken.cs | 114 +++++- .../Parameters/CallContext.cs | 31 +- src/FrostFS.SDK.Client/Pool/ClientWrapper.cs | 3 +- src/FrostFS.SDK.Client/Pool/InitParameters.cs | 6 + src/FrostFS.SDK.Client/Pool/Pool.cs | 194 +--------- src/FrostFS.SDK.Client/Pool/SessionCache.cs | 38 +- src/FrostFS.SDK.Client/Pool/WrapperPrm.cs | 9 +- .../Services/AccountingServiceProvider.cs | 4 +- .../Services/ApeManagerServiceProvider.cs | 22 +- .../Services/ContainerServiceProvider.cs | 84 ++--- .../Services/NetmapServiceProvider.cs | 87 +++-- .../Services/ObjectServiceProvider.cs | 234 +++++------- .../Services/SessionServiceProvider.cs | 8 +- .../Services/Shared/SessionProvider.cs | 15 +- src/FrostFS.SDK.Client/Tools/ClientContext.cs | 34 +- src/FrostFS.SDK.Client/Tools/ObjectTools.cs | 20 +- .../Tools/RequestConstructor.cs | 58 --- src/FrostFS.SDK.Client/Tools/RequestSigner.cs | 39 +- src/FrostFS.SDK.Client/Tools/SearchReader.cs | 10 +- src/FrostFS.SDK.Client/Tools/Verifier.cs | 1 + .../FrostFS.SDK.Cryptography.csproj | 8 +- src/FrostFS.SDK.Cryptography/UUID.cs | 6 +- .../FrostFS.SDK.Protos.csproj | 6 +- src/FrostFS.SDK.Tests/ContainerTestsBase.cs | 4 +- .../FrostFS.SDK.Tests.csproj | 14 +- .../Mocks/AsyncStreamRangeReaderMock.cs | 5 - .../Mocks/AsyncStreamReaderMock.cs | 2 +- .../ContainerServiceBase.cs | 6 +- src/FrostFS.SDK.Tests/Mocks/ObjectMock.cs | 9 +- src/FrostFS.SDK.Tests/Mocks/SessionMock.cs | 2 +- src/FrostFS.SDK.Tests/NetworkTest.cs | 23 +- src/FrostFS.SDK.Tests/NetworkTestsBase.cs | 23 +- src/FrostFS.SDK.Tests/ObjectTest.cs | 24 +- src/FrostFS.SDK.Tests/ObjectTestsBase.cs | 4 +- src/FrostFS.SDK.Tests/PoolSmokeTests.cs | 71 ++-- src/FrostFS.SDK.Tests/SessionTests.cs | 24 +- src/FrostFS.SDK.Tests/SessionTestsBase.cs | 4 +- src/FrostFS.SDK.Tests/SmokeClientTests.cs | 266 ++++++-------- src/FrostFS.SDK.Tests/SmokeTestsBase.cs | 4 - src/FrostFS.SDK.Tests/{TestData => }/cat.jpg | Bin 57 files changed, 845 insertions(+), 1116 deletions(-) rename src/FrostFS.SDK.Tests/{TestData => }/cat.jpg (100%) diff --git a/.editorconfig b/.editorconfig index 7a29e50..482baac 100644 --- a/.editorconfig +++ b/.editorconfig @@ -214,7 +214,7 @@ dotnet_diagnostic.CA1700.severity = warning dotnet_diagnostic.CA1707.severity = warning # CA1708: Identifiers should differ by more than case -dotnet_diagnostic.CA1708.severity = warning +dotnet_diagnostic.CA1708.severity = none # CA1710: Identifiers should have correct suffix dotnet_diagnostic.CA1710.severity = warning @@ -232,7 +232,7 @@ dotnet_diagnostic.CA1713.severity = warning dotnet_diagnostic.CA1715.severity = warning # CA1716: Identifiers should not match keywords -dotnet_diagnostic.CA1716.severity = warning +dotnet_diagnostic.CA1716.severity = none # CA1720: Identifier contains type name dotnet_diagnostic.CA1720.severity = warning diff --git a/src/FrostFS.SDK.Client/CllientKey.cs b/src/FrostFS.SDK.Client/CllientKey.cs index cdafb3f..da02b02 100644 --- a/src/FrostFS.SDK.Client/CllientKey.cs +++ b/src/FrostFS.SDK.Client/CllientKey.cs @@ -11,4 +11,8 @@ public class ClientKey(ECDsa key) internal ECDsa ECDsaKey { get; } = key; internal ByteString PublicKeyProto { get; } = ByteString.CopyFrom(key.PublicKey()); + + internal string PublicKey { get; } = key.PublicKey().ToString(); + + internal FrostFsOwner Owner { get; } = new FrostFsOwner(key.PublicKey().PublicKeyToAddress()); } diff --git a/src/FrostFS.SDK.Client/FrostFS.SDK.Client.csproj b/src/FrostFS.SDK.Client/FrostFS.SDK.Client.csproj index e175d13..7f4df50 100644 --- a/src/FrostFS.SDK.Client/FrostFS.SDK.Client.csproj +++ b/src/FrostFS.SDK.Client/FrostFS.SDK.Client.csproj @@ -14,6 +14,10 @@ true + + + true + diff --git a/src/FrostFS.SDK.Client/FrostFSClient.cs b/src/FrostFS.SDK.Client/FrostFSClient.cs index 4112a5b..e73ba49 100644 --- a/src/FrostFS.SDK.Client/FrostFSClient.cs +++ b/src/FrostFS.SDK.Client/FrostFSClient.cs @@ -4,12 +4,7 @@ using System.Net.Http; using System.Threading.Tasks; using Frostfs.V2.Ape; -using Frostfs.V2.Apemanager; -using FrostFS.Accounting; -using FrostFS.Container; -using FrostFS.Netmap; -using FrostFS.Object; using FrostFS.SDK.Client.Interfaces; using FrostFS.SDK.Client.Services; using FrostFS.SDK.Cryptography; @@ -21,32 +16,40 @@ using Grpc.Net.Client; using Microsoft.Extensions.Options; +using static Frostfs.V2.Apemanager.APEManagerService; +using static FrostFS.Accounting.AccountingService; +using static FrostFS.Container.ContainerService; +using static FrostFS.Netmap.NetmapService; +using static FrostFS.Object.ObjectService; +using static FrostFS.Session.SessionService; + namespace FrostFS.SDK.Client; public class FrostFSClient : IFrostFSClient { private bool isDisposed; - internal ContainerService.ContainerServiceClient? ContainerServiceClient { get; set; } + internal ContainerServiceClient? ContainerServiceClient { get; set; } + internal ContainerServiceProvider? ContainerServiceProvider { get; set; } - internal NetmapService.NetmapServiceClient? NetmapServiceClient { get; set; } + internal NetmapServiceClient? NetmapServiceClient { get; set; } + internal NetmapServiceProvider? NetmapServiceProvider { get; set; } - internal APEManagerService.APEManagerServiceClient? ApeManagerServiceClient { get; set; } + internal APEManagerServiceClient? ApeManagerServiceClient { get; set; } + internal ApeManagerServiceProvider? ApeManagerServiceProvider { get; set; } - internal SessionService.SessionServiceClient? SessionServiceClient { get; set; } + internal SessionServiceClient? SessionServiceClient { get; set; } + internal SessionServiceProvider? SessionServiceProvider { get; set; } - internal ObjectService.ObjectServiceClient? ObjectServiceClient { get; set; } + internal ObjectServiceClient? ObjectServiceClient { get; set; } + internal ObjectServiceProvider? ObjectServiceProvider { get; set; } - internal AccountingService.AccountingServiceClient? AccountingServiceClient { get; set; } + internal AccountingServiceClient? AccountingServiceClient { get; set; } + internal AccountingServiceProvider? AccountingServiceProvider { get; set; } internal ClientContext ClientCtx { get; set; } - public static IFrostFSClient GetInstance(IOptions clientOptions, GrpcChannelOptions? channelOptions = null) - { - return new FrostFSClient(clientOptions, channelOptions); - } - - public static IFrostFSClient GetSingleOwnerInstance(IOptions clientOptions, GrpcChannelOptions? channelOptions = null) + public static IFrostFSClient GetSingleOwnerInstance(IOptions clientOptions, GrpcChannelOptions? channelOptions = null) { return new FrostFSClient(clientOptions, channelOptions); } @@ -62,28 +65,28 @@ public class FrostFSClient : IFrostFSClient /// Object.ObjectService.ObjectServiceClient implementation /// public static IFrostFSClient GetTestInstance( - IOptions clientOptions, + IOptions settings, GrpcChannelOptions? channelOptions, - NetmapService.NetmapServiceClient netmapService, - SessionService.SessionServiceClient sessionService, - ContainerService.ContainerServiceClient containerService, - ObjectService.ObjectServiceClient objectService) + NetmapServiceClient netmapService, + SessionServiceClient sessionService, + ContainerServiceClient containerService, + ObjectServiceClient objectService) { - if (clientOptions is null) + if (settings is null) { - throw new ArgumentNullException(nameof(clientOptions)); + throw new ArgumentNullException(nameof(settings)); } - return new FrostFSClient(clientOptions, channelOptions, containerService, netmapService, sessionService, objectService); + return new FrostFSClient(settings, channelOptions, containerService, netmapService, sessionService, objectService); } private FrostFSClient( - IOptions settings, + IOptions settings, GrpcChannelOptions? channelOptions, - ContainerService.ContainerServiceClient containerService, - NetmapService.NetmapServiceClient netmapService, - SessionService.SessionServiceClient sessionService, - ObjectService.ObjectServiceClient objectService) + ContainerServiceClient containerService, + NetmapServiceClient netmapService, + SessionServiceClient sessionService, + ObjectServiceClient objectService) { if (settings is null) { @@ -91,14 +94,18 @@ public class FrostFSClient : IFrostFSClient } var ecdsaKey = settings.Value.Key.LoadWif(); - FrostFsOwner.FromKey(ecdsaKey); ClientCtx = new ClientContext( client: this, - key: ecdsaKey, + key: new ClientKey(ecdsaKey), owner: FrostFsOwner.FromKey(ecdsaKey), channel: InitGrpcChannel(settings.Value.Host, channelOptions), - version: new FrostFsVersion(2, 13)); + version: new FrostFsVersion(2, 13)) + { + SessionCache = new SessionCache(0), + Callback = settings.Value.Callback, + Interceptors = settings.Value.Interceptors + }; ContainerServiceClient = containerService ?? throw new ArgumentNullException(nameof(containerService)); NetmapServiceClient = netmapService ?? throw new ArgumentNullException(nameof(netmapService)); @@ -106,28 +113,9 @@ public class FrostFSClient : IFrostFSClient ObjectServiceClient = objectService ?? throw new ArgumentNullException(nameof(objectService)); } - private FrostFSClient(IOptions options, GrpcChannelOptions? channelOptions) + private FrostFSClient(IOptions settings, GrpcChannelOptions? channelOptions) { - var clientSettings = (options?.Value) ?? throw new ArgumentNullException(nameof(options), "Options value must be initialized"); - - clientSettings.Validate(); - - var channel = InitGrpcChannel(clientSettings.Host, channelOptions); - - ClientCtx = new ClientContext( - this, - key: null, - owner: null, - channel: channel, - version: new FrostFsVersion(2, 13)); - - // TODO: define timeout logic - // CheckFrostFsVersionSupport(new Context { Timeout = TimeSpan.FromSeconds(20) }); - } - - private FrostFSClient(IOptions options, GrpcChannelOptions? channelOptions) - { - var clientSettings = (options?.Value) ?? throw new ArgumentNullException(nameof(options), "Options value must be initialized"); + var clientSettings = (settings?.Value) ?? throw new ArgumentNullException(nameof(settings), "Options value must be initialized"); clientSettings.Validate(); @@ -137,10 +125,15 @@ public class FrostFSClient : IFrostFSClient ClientCtx = new ClientContext( this, - key: ecdsaKey, + key: new ClientKey(ecdsaKey), owner: FrostFsOwner.FromKey(ecdsaKey), channel: channel, - version: new FrostFsVersion(2, 13)); + version: new FrostFsVersion(2, 13)) + { + SessionCache = new SessionCache(0), + Callback = settings.Value.Callback, + Interceptors = settings.Value.Interceptors + }; // TODO: define timeout logic // CheckFrostFsVersionSupport(new Context { Timeout = TimeSpan.FromSeconds(20) }); @@ -150,12 +143,14 @@ public class FrostFSClient : IFrostFSClient { ClientCtx = new ClientContext( client: this, - key: prm.Key, + key: new ClientKey(prm.Key), owner: FrostFsOwner.FromKey(prm.Key!), channel: InitGrpcChannel(prm.Address, null), //prm.GrpcChannelOptions), version: new FrostFsVersion(2, 13)) { - SessionCache = cache + SessionCache = cache, + Interceptors = prm.Interceptors, + Callback = prm.Callback }; } @@ -175,14 +170,14 @@ public class FrostFSClient : IFrostFSClient } #region ApeManagerImplementation - public Task AddChainAsync(PrmApeChainAdd args) + public Task> AddChainAsync(PrmApeChainAdd args) { if (args is null) { throw new ArgumentNullException(nameof(args)); } - var service = GetApeManagerService(args); + var service = GetApeManagerService(); return service.AddChainAsync(args); } @@ -193,7 +188,7 @@ public class FrostFSClient : IFrostFSClient throw new ArgumentNullException(nameof(args)); } - var service = GetApeManagerService(args); + var service = GetApeManagerService(); return service.RemoveChainAsync(args); } @@ -202,7 +197,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetApeManagerService(args); + var service = GetApeManagerService(); return service.ListChainAsync(args); } #endregion @@ -213,14 +208,14 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetContainerService(args); + var service = GetContainerService(); return service.GetContainerAsync(args); } public IAsyncEnumerable ListContainersAsync(PrmContainerGetAll? args = null) { args ??= new PrmContainerGetAll(); - var service = GetContainerService(args); + var service = GetContainerService(); return service.ListContainersAsync(args); } @@ -229,7 +224,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetContainerService(args); + var service = GetContainerService(); return service.CreateContainerAsync(args); } @@ -238,7 +233,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetContainerService(args); + var service = GetContainerService(); return service.DeleteContainerAsync(args); } #endregion @@ -247,21 +242,21 @@ public class FrostFSClient : IFrostFSClient public Task GetNetmapSnapshotAsync(PrmNetmapSnapshot? args) { args ??= new PrmNetmapSnapshot(); - var service = GetNetmapService(args); + var service = GetNetmapService(); return service.GetNetmapSnapshotAsync(args); } public Task GetNodeInfoAsync(PrmNodeInfo? args) { args ??= new PrmNodeInfo(); - var service = GetNetmapService(args); + var service = GetNetmapService(); return service.GetLocalNodeInfoAsync(args); } public Task GetNetworkSettingsAsync(PrmNetworkSettings? args) { args ??= new PrmNetworkSettings(); - var service = GetNetmapService(args); + var service = GetNetmapService(); return service.GetNetworkSettingsAsync(args.Context!); } #endregion @@ -272,7 +267,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetObjectService(args); + var service = GetObjectService(); return service.GetObjectHeadAsync(args); } @@ -281,7 +276,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetObjectService(args); + var service = GetObjectService(); return service.GetObjectAsync(args); } @@ -290,7 +285,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetObjectService(args); + var service = GetObjectService(); return service.GetRangeAsync(args); } @@ -299,17 +294,16 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetObjectService(args); + var service = GetObjectService(); return service.GetRangeHashAsync(args); } - public Task PutObjectAsync(PrmObjectPut args) { if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetObjectService(args); + var service = GetObjectService(); return service.PutObjectAsync(args); } @@ -318,7 +312,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetObjectService(args); + var service = GetObjectService(); return service.PutSingleObjectAsync(args); } @@ -329,7 +323,7 @@ public class FrostFSClient : IFrostFSClient throw new ArgumentNullException(nameof(args)); } - var service = GetObjectService(args); + var service = GetObjectService(); return service.PatchObjectAsync(args); } @@ -338,7 +332,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetObjectService(args); + var service = GetObjectService(); return service.DeleteObjectAsync(args); } @@ -347,7 +341,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetObjectService(args); + var service = GetObjectService(); return service.SearchObjectsAsync(args); } #endregion @@ -358,10 +352,9 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var session = await CreateSessionInternalAsync(args).ConfigureAwait(false); + var token = await CreateSessionInternalAsync(args).ConfigureAwait(false); - var token = session.Serialize(); - return new FrostFsSessionToken(token, session.Body.Id.ToUuid()); + return new FrostFsSessionToken(token); } internal Task CreateSessionInternalAsync(PrmSessionCreate args) @@ -369,7 +362,7 @@ public class FrostFSClient : IFrostFSClient if (args is null) throw new ArgumentNullException(nameof(args)); - var service = GetSessionService(args); + var service = GetSessionService(); return service.CreateSessionAsync(args); } #endregion @@ -379,18 +372,18 @@ public class FrostFSClient : IFrostFSClient { args ??= new PrmBalance(); - var service = GetAccouningService(args); + var service = GetAccouningService(); return await service.GetBallance(args).ConfigureAwait(false); } #endregion #region ToolsImplementation - public FrostFsObjectId CalculateObjectId(FrostFsObjectHeader header, CallContext ctx) + public FrostFsObjectId CalculateObjectId(FrostFsObjectHeader header) { if (header == null) throw new ArgumentNullException(nameof(header)); - return ObjectTools.CalculateObjectId(header, ctx); + return ObjectTools.CalculateObjectId(header, this.ClientCtx); } #endregion @@ -398,59 +391,34 @@ public class FrostFSClient : IFrostFSClient { var args = new PrmNodeInfo(ctx); - if (ctx?.Version == null) - throw new ArgumentNullException(nameof(ctx), "Version must be initialized"); - - var service = GetNetmapService(args); + var service = GetNetmapService(); var localNodeInfo = await service.GetLocalNodeInfoAsync(args).ConfigureAwait(false); - if (!localNodeInfo.Version.IsSupported(ctx.Version)) + if (!localNodeInfo.Version.IsSupported(ClientCtx.Version)) { var msg = $"FrostFS {localNodeInfo.Version} is not supported."; throw new FrostFsException(msg); } } - private CallInvoker? SetupClientContext(IContext ctx) + private CallInvoker? CreateInvoker() { if (isDisposed) throw new FrostFsInvalidObjectException("Client is disposed."); - if (ctx.Context!.Key == null) - { - if (ClientCtx.Key == null) - { - throw new ArgumentNullException(nameof(ctx), "Key is not initialized."); - } - - ctx.Context.Key = ClientCtx.Key.ECDsaKey; - } - - if (ctx.Context.OwnerId == null) - { - ctx.Context.OwnerId = ClientCtx.Owner ?? FrostFsOwner.FromKey(ctx.Context.Key); - } - - if (ctx.Context.Version == null) - { - if (ClientCtx.Version == null) - { - throw new ArgumentNullException(nameof(ctx), "Version is not initialized."); - } - - ctx.Context.Version = ClientCtx.Version; - } - CallInvoker? callInvoker = null; - foreach (var interceptor in ctx.Context.Interceptors) - callInvoker = AddInvoker(callInvoker, interceptor); + if (ClientCtx.Interceptors != null) + { + foreach (var interceptor in ClientCtx.Interceptors) + callInvoker = AddInvoker(callInvoker, interceptor); + } - if (ctx.Context.Callback != null) - callInvoker = AddInvoker(callInvoker, new MetricsInterceptor(ctx.Context.Callback)); + if (ClientCtx.Callback != null) + callInvoker = AddInvoker(callInvoker, new MetricsInterceptor(ClientCtx.Callback)); - if (ctx.Context.PoolErrorHandler != null) - callInvoker = AddInvoker(callInvoker, new ErrorInterceptor(ctx.Context.PoolErrorHandler)); + if (ClientCtx.PoolErrorHandler != null) + callInvoker = AddInvoker(callInvoker, new ErrorInterceptor(ClientCtx.PoolErrorHandler)); return callInvoker; @@ -465,75 +433,109 @@ public class FrostFSClient : IFrostFSClient } } - private NetmapServiceProvider GetNetmapService(IContext ctx) + private NetmapServiceProvider GetNetmapService() { - var callInvoker = SetupClientContext(ctx); - var client = NetmapServiceClient ?? (callInvoker != null - ? new NetmapService.NetmapServiceClient(callInvoker) - : new NetmapService.NetmapServiceClient(ClientCtx.Channel)); + if (NetmapServiceProvider == null) + { + var invoker = CreateInvoker(); - return new NetmapServiceProvider(client, ClientCtx); + NetmapServiceClient = NetmapServiceClient ?? ( + invoker != null + ? new NetmapServiceClient(invoker) + : new NetmapServiceClient(ClientCtx.Channel)); + + NetmapServiceProvider = new NetmapServiceProvider(NetmapServiceClient, ClientCtx); + } + + return NetmapServiceProvider; } - private SessionServiceProvider GetSessionService(IContext ctx) + private SessionServiceProvider GetSessionService() { - var callInvoker = SetupClientContext(ctx); - var client = SessionServiceClient ?? (callInvoker != null - ? new SessionService.SessionServiceClient(callInvoker) - : new SessionService.SessionServiceClient(ClientCtx.Channel)); + if (SessionServiceProvider == null) + { + var invoker = CreateInvoker(); + + SessionServiceClient = SessionServiceClient ?? ( + invoker != null + ? new SessionServiceClient(invoker) + : new SessionServiceClient(ClientCtx.Channel)); + + SessionServiceProvider = new SessionServiceProvider(SessionServiceClient, ClientCtx); + } + + return SessionServiceProvider; - return new SessionServiceProvider(client, ClientCtx); } - private ApeManagerServiceProvider GetApeManagerService(IContext ctx) + private ApeManagerServiceProvider GetApeManagerService() { - var callInvoker = SetupClientContext(ctx); - var client = ApeManagerServiceClient ?? (callInvoker != null - ? new APEManagerService.APEManagerServiceClient(callInvoker) - : new APEManagerService.APEManagerServiceClient(ClientCtx.Channel)); + if (ApeManagerServiceProvider == null) + { + var invoker = CreateInvoker(); - return new ApeManagerServiceProvider(client, ClientCtx); + ApeManagerServiceClient = ApeManagerServiceClient ?? ( + invoker != null + ? new APEManagerServiceClient(invoker) + : new APEManagerServiceClient(ClientCtx.Channel)); + + ApeManagerServiceProvider = new ApeManagerServiceProvider(ApeManagerServiceClient, ClientCtx); + } + + return ApeManagerServiceProvider; } - private AccountingServiceProvider GetAccouningService(IContext ctx) + private AccountingServiceProvider GetAccouningService() { - var callInvoker = SetupClientContext(ctx); - var client = AccountingServiceClient ?? (callInvoker != null - ? new AccountingService.AccountingServiceClient(callInvoker) - : new AccountingService.AccountingServiceClient(ClientCtx.Channel)); + if (this.AccountingServiceProvider == null) + { + var invoker = CreateInvoker(); - return new AccountingServiceProvider(client, ClientCtx); + AccountingServiceClient = AccountingServiceClient ?? ( + invoker != null + ? new AccountingServiceClient(invoker) + : new AccountingServiceClient(ClientCtx.Channel)); + + AccountingServiceProvider = new AccountingServiceProvider(AccountingServiceClient, ClientCtx); + } + + return AccountingServiceProvider; } - private ContainerServiceProvider GetContainerService(IContext ctx) + private ContainerServiceProvider GetContainerService() { - var callInvoker = SetupClientContext(ctx); - var client = ContainerServiceClient ?? (callInvoker != null - ? new ContainerService.ContainerServiceClient(callInvoker) - : new ContainerService.ContainerServiceClient(ClientCtx.Channel)); + if (this.ContainerServiceProvider == null) + { + var invoker = CreateInvoker(); - return new ContainerServiceProvider(client, ClientCtx); + ContainerServiceClient = ContainerServiceClient ?? ( + invoker != null + ? new ContainerServiceClient(invoker) + : new ContainerServiceClient(ClientCtx.Channel)); + + ContainerServiceProvider = new ContainerServiceProvider(ContainerServiceClient, ClientCtx); + } + + return ContainerServiceProvider; } - private ObjectServiceProvider GetObjectService(IContext ctx) + private ObjectServiceProvider GetObjectService() { - var callInvoker = SetupClientContext(ctx); - var client = ObjectServiceClient ?? (callInvoker != null - ? new ObjectService.ObjectServiceClient(callInvoker) - : new ObjectService.ObjectServiceClient(ClientCtx.Channel)); + if (this.ObjectServiceProvider == null) + { + var invoker = CreateInvoker(); - return new ObjectServiceProvider(client, ClientCtx); + ObjectServiceClient = ObjectServiceClient ?? ( + invoker != null + ? new ObjectServiceClient(invoker) + : new ObjectServiceClient(ClientCtx.Channel)); + + ObjectServiceProvider = new ObjectServiceProvider(ObjectServiceClient, ClientCtx); + } + + return ObjectServiceProvider; } - private AccountingServiceProvider GetAccountService(IContext ctx) - { - var callInvoker = SetupClientContext(ctx); - var client = AccountingServiceClient ?? (callInvoker != null - ? new AccountingService.AccountingServiceClient(callInvoker) - : new AccountingService.AccountingServiceClient(ClientCtx.Channel)); - - return new AccountingServiceProvider(client, ClientCtx); - } private static GrpcChannel InitGrpcChannel(string host, GrpcChannelOptions? channelOptions) { @@ -559,7 +561,7 @@ public class FrostFSClient : IFrostFSClient { var prm = new PrmBalance(ctx); - var service = GetAccouningService(prm); + var service = GetAccouningService(); _ = await service.GetBallance(prm).ConfigureAwait(false); return null; diff --git a/src/FrostFS.SDK.Client/Interfaces/IFrostFSClient.cs b/src/FrostFS.SDK.Client/Interfaces/IFrostFSClient.cs index 456962f..8520528 100644 --- a/src/FrostFS.SDK.Client/Interfaces/IFrostFSClient.cs +++ b/src/FrostFS.SDK.Client/Interfaces/IFrostFSClient.cs @@ -21,7 +21,7 @@ public interface IFrostFSClient : IDisposable #endregion #region ApeManager - Task AddChainAsync(PrmApeChainAdd args); + Task> AddChainAsync(PrmApeChainAdd args); Task RemoveChainAsync(PrmApeChainRemove args); @@ -63,7 +63,7 @@ public interface IFrostFSClient : IDisposable #endregion #region Tools - FrostFsObjectId CalculateObjectId(FrostFsObjectHeader header, CallContext ctx); + FrostFsObjectId CalculateObjectId(FrostFsObjectHeader header); #endregion public Task Dial(CallContext ctx); diff --git a/src/FrostFS.SDK.Client/Mappers/ContainerId.cs b/src/FrostFS.SDK.Client/Mappers/ContainerId.cs index fe8632f..df27320 100644 --- a/src/FrostFS.SDK.Client/Mappers/ContainerId.cs +++ b/src/FrostFS.SDK.Client/Mappers/ContainerId.cs @@ -43,6 +43,6 @@ public static class ContainerIdMapper throw new ArgumentNullException(nameof(message)); } - return new FrostFsContainerId(Base58.Encode(message.Value.ToByteArray())); + return new FrostFsContainerId(Base58.Encode(message.Value.Span)); } } \ No newline at end of file diff --git a/src/FrostFS.SDK.Client/Mappers/Netmap/NodeInfo.cs b/src/FrostFS.SDK.Client/Mappers/Netmap/NodeInfo.cs index 747468b..d7340eb 100644 --- a/src/FrostFS.SDK.Client/Mappers/Netmap/NodeInfo.cs +++ b/src/FrostFS.SDK.Client/Mappers/Netmap/NodeInfo.cs @@ -39,7 +39,7 @@ public static class NodeInfoMapper state: state, addresses: [.. nodeInfo.Addresses], attributes: nodeInfo.Attributes.ToDictionary(n => n.Key, n => n.Value), - publicKey: nodeInfo.PublicKey.ToByteArray() + publicKey: nodeInfo.PublicKey.Memory ); } } \ No newline at end of file diff --git a/src/FrostFS.SDK.Client/Mappers/Object/Object.cs b/src/FrostFS.SDK.Client/Mappers/Object/Object.cs index b9d1520..5f85fc5 100644 --- a/src/FrostFS.SDK.Client/Mappers/Object/Object.cs +++ b/src/FrostFS.SDK.Client/Mappers/Object/Object.cs @@ -6,7 +6,7 @@ internal static class ObjectMapper { return new FrostFsObject(obj.Header.ToModel()) { - ObjectId = FrostFsObjectId.FromHash(obj.ObjectId.Value.ToByteArray()) + ObjectId = FrostFsObjectId.FromHash(obj.ObjectId.Value.Span) }; } } diff --git a/src/FrostFS.SDK.Client/Mappers/Object/ObjectHeaderMapper.cs b/src/FrostFS.SDK.Client/Mappers/Object/ObjectHeaderMapper.cs index 1ee29a6..317b31c 100644 --- a/src/FrostFS.SDK.Client/Mappers/Object/ObjectHeaderMapper.cs +++ b/src/FrostFS.SDK.Client/Mappers/Object/ObjectHeaderMapper.cs @@ -40,7 +40,7 @@ public static class ObjectHeaderMapper } var model = new FrostFsObjectHeader( - new FrostFsContainerId(Base58.Encode(header.ContainerId.Value.ToByteArray())), + new FrostFsContainerId(Base58.Encode(header.ContainerId.Value.Span)), objTypeName, header.Attributes.Select(attribute => attribute.ToModel()).ToArray(), split, diff --git a/src/FrostFS.SDK.Client/Mappers/Object/ObjectId.cs b/src/FrostFS.SDK.Client/Mappers/Object/ObjectId.cs index 373edfb..343e3de 100644 --- a/src/FrostFS.SDK.Client/Mappers/Object/ObjectId.cs +++ b/src/FrostFS.SDK.Client/Mappers/Object/ObjectId.cs @@ -28,6 +28,6 @@ public static class ObjectIdMapper throw new ArgumentNullException(nameof(objectId)); } - return FrostFsObjectId.FromHash(objectId.Value.ToByteArray()); + return FrostFsObjectId.FromHash(objectId.Value.Span); } } \ No newline at end of file diff --git a/src/FrostFS.SDK.Client/Mappers/OwnerId.cs b/src/FrostFS.SDK.Client/Mappers/OwnerId.cs index 297178a..6739a0b 100644 --- a/src/FrostFS.SDK.Client/Mappers/OwnerId.cs +++ b/src/FrostFS.SDK.Client/Mappers/OwnerId.cs @@ -44,7 +44,7 @@ public static class OwnerIdMapper if (!Caches.Owners.TryGetValue(message, out FrostFsOwner? model)) { - model = new FrostFsOwner(Base58.Encode(message.Value.ToByteArray())); + model = new FrostFsOwner(Base58.Encode(message.Value.Span)); Caches.Owners.Set(message, model, _oneHourExpiration); } diff --git a/src/FrostFS.SDK.Client/Models/Client/ClientSettings.cs b/src/FrostFS.SDK.Client/Models/Client/ClientSettings.cs index 6ebec1f..4aa00f9 100644 --- a/src/FrostFS.SDK.Client/Models/Client/ClientSettings.cs +++ b/src/FrostFS.SDK.Client/Models/Client/ClientSettings.cs @@ -3,6 +3,8 @@ using System.Collections.ObjectModel; using System.Globalization; using System.Text; +using Grpc.Core.Interceptors; + namespace FrostFS.SDK; public class ClientSettings @@ -11,60 +13,25 @@ public class ClientSettings public string Host { get; set; } = string.Empty; - public virtual void Validate() - { - var errors = CheckFields(); - if (errors != null) - ThrowSettingsException(errors); - } - - protected Collection? CheckFields() - { - if (string.IsNullOrWhiteSpace(Host)) - { - var error = string.Format(CultureInfo.InvariantCulture, errorTemplate, nameof(Host)); - return new Collection([error]); - } - - return null; - } - - protected static void ThrowSettingsException(Collection errors) - { - if (errors is null) - { - throw new ArgumentNullException(nameof(errors)); - } - - StringBuilder messages = new(); - - foreach (var error in errors) - { - messages.AppendLine(error); - } - - throw new ArgumentException(messages.ToString()); - } -} - -public class SingleOwnerClientSettings : ClientSettings -{ public string Key { get; set; } = string.Empty; - public override void Validate() - { - var errors = CheckFields(); - if (errors != null) - ThrowSettingsException(errors); - } + public Action? Callback { get; set; } - protected new Collection? CheckFields() + public Collection Interceptors { get; } = []; + + public void Validate() { - Collection? errors = base.CheckFields(); + StringBuilder? errors = null; + + if (string.IsNullOrWhiteSpace(Host)) + (errors = new StringBuilder(128)).AppendLine(string.Format(CultureInfo.InvariantCulture, errorTemplate, nameof(Host))); if (string.IsNullOrWhiteSpace(Key)) - (errors ??= []).Add(string.Format(CultureInfo.InvariantCulture, errorTemplate, nameof(Key))); + (errors ??= new StringBuilder(128)).AppendLine(string.Format(CultureInfo.InvariantCulture, errorTemplate, nameof(Key))); - return errors; + if (errors != null) + { + throw new ArgumentException(errors.ToString()); + } } } \ No newline at end of file diff --git a/src/FrostFS.SDK.Client/Models/Containers/FrostFsContainerId.cs b/src/FrostFS.SDK.Client/Models/Containers/FrostFsContainerId.cs index f0b3fdf..9f081cd 100644 --- a/src/FrostFS.SDK.Client/Models/Containers/FrostFsContainerId.cs +++ b/src/FrostFS.SDK.Client/Models/Containers/FrostFsContainerId.cs @@ -27,7 +27,7 @@ public class FrostFsContainerId if (containerID != null) { - this.modelId = Base58.Encode(containerID.Value.ToByteArray()); + this.modelId = Base58.Encode(containerID.Value.Span); return this.modelId; } diff --git a/src/FrostFS.SDK.Client/Models/Containers/FrostFsContainerInfo.cs b/src/FrostFS.SDK.Client/Models/Containers/FrostFsContainerInfo.cs index 6223a7f..af4ffdc 100644 --- a/src/FrostFS.SDK.Client/Models/Containers/FrostFsContainerInfo.cs +++ b/src/FrostFS.SDK.Client/Models/Containers/FrostFsContainerInfo.cs @@ -96,7 +96,7 @@ public class FrostFsContainerInfo PlacementPolicy = PlacementPolicy.Value.GetPolicy(), Nonce = ByteString.CopyFrom(Nonce.ToBytes()), OwnerId = Owner?.OwnerID, - Version = Version?.Version + Version = Version?.VersionID }; var attribs = GetGrpsAttributes(); diff --git a/src/FrostFS.SDK.Client/Models/Netmap/FrostFsVersion.cs b/src/FrostFS.SDK.Client/Models/Netmap/FrostFsVersion.cs index 6fd0afc..81e642d 100644 --- a/src/FrostFS.SDK.Client/Models/Netmap/FrostFsVersion.cs +++ b/src/FrostFS.SDK.Client/Models/Netmap/FrostFsVersion.cs @@ -10,7 +10,7 @@ public class FrostFsVersion(int major, int minor) public int Major { get; set; } = major; public int Minor { get; set; } = minor; - internal Version Version + internal Version VersionID { get { diff --git a/src/FrostFS.SDK.Client/Models/Object/FrostFsObjectId.cs b/src/FrostFS.SDK.Client/Models/Object/FrostFsObjectId.cs index 49fde5a..57300a3 100644 --- a/src/FrostFS.SDK.Client/Models/Object/FrostFsObjectId.cs +++ b/src/FrostFS.SDK.Client/Models/Object/FrostFsObjectId.cs @@ -8,13 +8,8 @@ public class FrostFsObjectId(string id) { public string Value { get; } = id; - public static FrostFsObjectId FromHash(byte[] hash) + public static FrostFsObjectId FromHash(ReadOnlySpan hash) { - if (hash is null) - { - throw new ArgumentNullException(nameof(hash)); - } - if (hash.Length != Constants.Sha256HashLength) throw new FormatException("ObjectID must be a sha256 hash."); diff --git a/src/FrostFS.SDK.Client/Models/Object/FrostFsOwner.cs b/src/FrostFS.SDK.Client/Models/Object/FrostFsOwner.cs index 1bd6754..228edf2 100644 --- a/src/FrostFS.SDK.Client/Models/Object/FrostFsOwner.cs +++ b/src/FrostFS.SDK.Client/Models/Object/FrostFsOwner.cs @@ -4,6 +4,7 @@ using FrostFS.Refs; using FrostFS.SDK.Client.Mappers.GRPC; using FrostFS.SDK.Cryptography; + namespace FrostFS.SDK; public class FrostFsOwner(string id) diff --git a/src/FrostFS.SDK.Client/Models/Session/FrostFsSessionToken.cs b/src/FrostFS.SDK.Client/Models/Session/FrostFsSessionToken.cs index 672754d..54f0615 100644 --- a/src/FrostFS.SDK.Client/Models/Session/FrostFsSessionToken.cs +++ b/src/FrostFS.SDK.Client/Models/Session/FrostFsSessionToken.cs @@ -1,9 +1,117 @@ using System; +using FrostFS.Refs; + +using FrostFS.SDK.Client; +using FrostFS.SDK.Cryptography; +using FrostFS.Session; + +using Google.Protobuf; + namespace FrostFS.SDK; -public class FrostFsSessionToken(byte[] token, Guid id) +public class FrostFsSessionToken { - public Guid Id { get; private set; } = id; - public byte[] Token { get; private set; } = token; + private Guid _id; + private ReadOnlyMemory _sessionKey; + private readonly SessionToken.Types.Body _body; + + private FrostFsSessionToken() + { + ProtoId = ByteString.Empty; + ProtoSessionKey = ByteString.Empty; + _body = new SessionToken.Types.Body(); + } + + internal FrostFsSessionToken(SessionToken token) + { + ProtoId = token.Body.Id; + ProtoSessionKey = token.Body.SessionKey; + + _body = token.Body; + } + + public Guid Id + { + get + { + if (_id == Guid.Empty) + _id = ProtoId.ToUuid(); + + return _id; + } + } + + public ReadOnlyMemory SessionKey + { + get + { + if (_sessionKey.IsEmpty) + _sessionKey = ProtoSessionKey.Memory; + + return _sessionKey; + } + } + + internal ByteString ProtoId { get; } + + internal ByteString ProtoSessionKey { get; } + + public SessionToken CreateContainerToken(ContainerID? containerId, ContainerSessionContext.Types.Verb verb, ClientKey key) + { + if (key is null) + { + throw new ArgumentNullException(nameof(key)); + } + + SessionToken sessionToken = new() { Body = _body.Clone() }; + + sessionToken.Body.Container = new() { Verb = verb }; + + if (containerId != null) + sessionToken.Body.Container.ContainerId = containerId; + else + sessionToken.Body.Container.Wildcard = true; + + sessionToken.Body.SessionKey = key.PublicKeyProto; + + sessionToken.Signature = key.ECDsaKey.SignMessagePart(sessionToken.Body); + + return sessionToken; + } + + public SessionToken CreateObjectTokenContext(Address address, ObjectSessionContext.Types.Verb verb, ClientKey key) + { + if (address is null) + { + throw new ArgumentNullException(nameof(address)); + } + + if (key is null) + { + throw new ArgumentNullException(nameof(key)); + } + + SessionToken sessionToken = new() + { + Body = _body.Clone() + }; + + ObjectSessionContext.Types.Target target = new() { Container = address.ContainerId }; + + if (address.ObjectId != null) + target.Objects.Add(address.ObjectId); + + sessionToken.Body.Object = new() + { + Target = target, + Verb = verb + }; + + sessionToken.Body.SessionKey = key.PublicKeyProto; + + sessionToken.Signature = key.ECDsaKey.SignMessagePart(sessionToken.Body); + + return sessionToken; + } } diff --git a/src/FrostFS.SDK.Client/Parameters/CallContext.cs b/src/FrostFS.SDK.Client/Parameters/CallContext.cs index 323a990..05e061c 100644 --- a/src/FrostFS.SDK.Client/Parameters/CallContext.cs +++ b/src/FrostFS.SDK.Client/Parameters/CallContext.cs @@ -1,27 +1,16 @@ using System; -using System.Collections.ObjectModel; -using System.Security.Cryptography; using System.Threading; -using FrostFS.SDK.Cryptography; - -using Google.Protobuf; - -using Grpc.Core.Interceptors; - namespace FrostFS.SDK.Client; public class CallContext() { - private ByteString? publicKeyCache; - internal Action? PoolErrorHandler { get; set; } + // internal Action? PoolErrorHandler { get; set; } - public ECDsa? Key { get; set; } + // public FrostFsOwner? OwnerId { get; set; } - public FrostFsOwner? OwnerId { get; set; } - - public FrostFsVersion? Version { get; set; } + // public FrostFsVersion? Version { get; set; } public CancellationToken CancellationToken { get; set; } @@ -29,17 +18,7 @@ public class CallContext() public DateTime? Deadline => Timeout.Ticks > 0 ? DateTime.UtcNow.Add(Timeout) : null; - public Action? Callback { get; set; } + // public Action? Callback { get; set; } - public Collection Interceptors { get; } = []; - - public ByteString? GetPublicKeyCache() - { - if (publicKeyCache == null && Key != null) - { - publicKeyCache = ByteString.CopyFrom(Key.PublicKey()); - } - - return publicKeyCache; - } + // public Collection Interceptors { get; } = []; } diff --git a/src/FrostFS.SDK.Client/Pool/ClientWrapper.cs b/src/FrostFS.SDK.Client/Pool/ClientWrapper.cs index 7e50337..de984d6 100644 --- a/src/FrostFS.SDK.Client/Pool/ClientWrapper.cs +++ b/src/FrostFS.SDK.Client/Pool/ClientWrapper.cs @@ -9,10 +9,8 @@ namespace FrostFS.SDK.Client; public class ClientWrapper : ClientStatusMonitor { private readonly object _lock = new(); - private SessionCache sessionCache; - internal ClientWrapper(WrapperPrm wrapperPrm, Pool pool) : base(wrapperPrm.Logger, wrapperPrm.Address) { WrapperPrm = wrapperPrm; @@ -140,6 +138,7 @@ public class ClientWrapper : ClientStatusMonitor catch (FrostFsException) { SetUnhealthy(); + return wasHealthy; } diff --git a/src/FrostFS.SDK.Client/Pool/InitParameters.cs b/src/FrostFS.SDK.Client/Pool/InitParameters.cs index 480820e..3f4d14e 100644 --- a/src/FrostFS.SDK.Client/Pool/InitParameters.cs +++ b/src/FrostFS.SDK.Client/Pool/InitParameters.cs @@ -1,6 +1,8 @@ using System; +using System.Collections.ObjectModel; using System.Security.Cryptography; +using Grpc.Core.Interceptors; using Grpc.Net.Client; using Microsoft.Extensions.Logging; @@ -33,4 +35,8 @@ public class InitParameters public ulong GracefulCloseOnSwitchTimeout { get; set; } public ILogger? Logger { get; set; } + + public Action? Callback { get; set; } + + public Collection Interceptors { get; } = []; } diff --git a/src/FrostFS.SDK.Client/Pool/Pool.cs b/src/FrostFS.SDK.Client/Pool/Pool.cs index 105978d..acb60b9 100644 --- a/src/FrostFS.SDK.Client/Pool/Pool.cs +++ b/src/FrostFS.SDK.Client/Pool/Pool.cs @@ -35,18 +35,17 @@ public partial class Pool : IFrostFSClient private InnerPool[]? InnerPools { get; set; } - private ECDsa Key { get; set; } - - private string PublicKey { get; } + private ClientKey Key { get; set; } private OwnerID? _ownerId; + private FrostFsOwner? _owner; private FrostFsOwner Owner { get { - _owner ??= new FrostFsOwner(Key.PublicKey().PublicKeyToAddress()); + _owner ??= new FrostFsOwner(Key.ECDsaKey.PublicKey().PublicKeyToAddress()); return _owner; } } @@ -57,7 +56,7 @@ public partial class Pool : IFrostFSClient { if (_ownerId == null) { - _owner = new FrostFsOwner(Key.PublicKey().PublicKeyToAddress()); + _owner = Key.Owner; _ownerId = _owner.ToMessage(); } return _ownerId; @@ -101,8 +100,7 @@ public partial class Pool : IFrostFSClient FillDefaultInitParams(options, this); - Key = options.Key; - PublicKey = $"{Key.PublicKey()}"; + Key = new ClientKey(options.Key); SessionCache = cache; logger = options.Logger; @@ -115,16 +113,9 @@ public partial class Pool : IFrostFSClient options.SessionExpirationDuration); ClientBuilder = options.ClientBuilder!; - } - private void SetupContext(CallContext ctx) - { - if (ctx == null) - { - throw new ArgumentNullException(nameof(ctx)); - } + // ClientContext.PoolErrorHandler = client.HandleError; - ctx.Key ??= Key; } // Dial establishes a connection to the servers from the FrostFS network. @@ -137,8 +128,6 @@ public partial class Pool : IFrostFSClient // See also InitParameters.SetClientRebalanceInterval. public async Task Dial(CallContext ctx) { - SetupContext(ctx); - var inner = new InnerPool[RebalanceParams.NodesParams.Length]; bool atLeastOneHealthy = false; @@ -158,11 +147,11 @@ public partial class Pool : IFrostFSClient await client.Dial(ctx).ConfigureAwait(false); dialed = true; - var token = await InitSessionForDuration(ctx, client, RebalanceParams.SessionExpirationDuration, Key, false) + var token = await InitSessionForDuration(ctx, client, RebalanceParams.SessionExpirationDuration, Key.ECDsaKey, false) .ConfigureAwait(false); - var key = FormCacheKey(nodeParams.Addresses[j], Key.PrivateKey().ToString()); - _ = SessionCache.Cache[key] = token; + var key = FormCacheKey(nodeParams.Addresses[j], Key.PublicKey); + SessionCache.SetValue(key, token); atLeastOneHealthy = true; } @@ -291,12 +280,14 @@ public partial class Pool : IFrostFSClient var wrapperPrm = new WrapperPrm { Address = address, - Key = parameters.Key, + Key = parameters.Key!, Logger = parameters.Logger, DialTimeout = parameters.NodeDialTimeout, StreamTimeout = parameters.NodeStreamTimeout, ErrorThreshold = parameters.ErrorThreshold, - GracefulCloseOnSwitchTimeout = parameters.GracefulCloseOnSwitchTimeout + GracefulCloseOnSwitchTimeout = parameters.GracefulCloseOnSwitchTimeout, + Callback = parameters.Callback, + Interceptors = parameters.Interceptors }; return new ClientWrapper(wrapperPrm, pool); @@ -318,7 +309,7 @@ public partial class Pool : IFrostFSClient throw new FrostFsException("Cannot find alive client"); } - private static async Task InitSessionForDuration(CallContext ctx, ClientWrapper cw, ulong duration, ECDsa key, bool clientCut) + private static async Task InitSessionForDuration(CallContext ctx, ClientWrapper cw, ulong duration, ECDsa key, bool clientCut) { var client = cw.Client; var networkInfo = await client!.GetNetworkSettingsAsync(new PrmNetworkSettings(ctx)).ConfigureAwait(false); @@ -521,7 +512,6 @@ public partial class Pool : IFrostFSClient var client = Connection(); args ??= new(); - args.Context.PoolErrorHandler = client.HandleError; return await client.Client!.GetNetmapSnapshotAsync(args).ConfigureAwait(false); } @@ -529,278 +519,126 @@ public partial class Pool : IFrostFSClient public async Task GetNodeInfoAsync(PrmNodeInfo? args = null) { var client = Connection(); - - args ??= new(); - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.GetNodeInfoAsync(args).ConfigureAwait(false); } public async Task GetNetworkSettingsAsync(PrmNetworkSettings? args = null) { var client = Connection(); - - args ??= new(); - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.GetNetworkSettingsAsync(args).ConfigureAwait(false); } public async Task CreateSessionAsync(PrmSessionCreate args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.CreateSessionAsync(args).ConfigureAwait(false); } - public async Task AddChainAsync(PrmApeChainAdd args) + public async Task> AddChainAsync(PrmApeChainAdd args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.AddChainAsync(args).ConfigureAwait(false); } public async Task RemoveChainAsync(PrmApeChainRemove args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - await client.Client!.RemoveChainAsync(args).ConfigureAwait(false); } public async Task ListChainAsync(PrmApeChainList args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.ListChainAsync(args).ConfigureAwait(false); } public async Task GetContainerAsync(PrmContainerGet args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.GetContainerAsync(args).ConfigureAwait(false); } public IAsyncEnumerable ListContainersAsync(PrmContainerGetAll? args = null) { var client = Connection(); - - args ??= new(); - args.Context.PoolErrorHandler = client.HandleError; - return client.Client!.ListContainersAsync(args); } public async Task CreateContainerAsync(PrmContainerCreate args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.CreateContainerAsync(args).ConfigureAwait(false); } public async Task DeleteContainerAsync(PrmContainerDelete args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - await client.Client!.DeleteContainerAsync(args).ConfigureAwait(false); } public async Task GetObjectHeadAsync(PrmObjectHeadGet args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.GetObjectHeadAsync(args).ConfigureAwait(false); } public async Task GetObjectAsync(PrmObjectGet args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.GetObjectAsync(args).ConfigureAwait(false); } public async Task PutObjectAsync(PrmObjectPut args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.PutObjectAsync(args).ConfigureAwait(false); } public async Task PutSingleObjectAsync(PrmSingleObjectPut args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.PutSingleObjectAsync(args).ConfigureAwait(false); } public async Task PatchObjectAsync(PrmObjectPatch args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.PatchObjectAsync(args).ConfigureAwait(false); } public async Task GetRangeAsync(PrmRangeGet args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.GetRangeAsync(args).ConfigureAwait(false); } public async Task[]> GetRangeHashAsync(PrmRangeHashGet args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.GetRangeHashAsync(args).ConfigureAwait(false); } public async Task PatchAsync(PrmObjectPatch args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.PatchObjectAsync(args).ConfigureAwait(false); } public async Task DeleteObjectAsync(PrmObjectDelete args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - await client.Client!.DeleteObjectAsync(args).ConfigureAwait(false); } public IAsyncEnumerable SearchObjectsAsync(PrmObjectSearch args) { - if (args is null) - { - throw new ArgumentNullException(nameof(args)); - } - var client = Connection(); - - args.Context.PoolErrorHandler = client.HandleError; - return client.Client!.SearchObjectsAsync(args); } public async Task GetBalanceAsync(PrmBalance? args) { var client = Connection(); - - args ??= new(); - args.Context.PoolErrorHandler = client.HandleError; - return await client.Client!.GetBalanceAsync(args).ConfigureAwait(false); } @@ -824,7 +662,7 @@ public partial class Pool : IFrostFSClient GC.SuppressFinalize(this); } - public FrostFsObjectId CalculateObjectId(FrostFsObjectHeader header, CallContext ctx) + public FrostFsObjectId CalculateObjectId(FrostFsObjectHeader header) { throw new NotImplementedException(); } diff --git a/src/FrostFS.SDK.Client/Pool/SessionCache.cs b/src/FrostFS.SDK.Client/Pool/SessionCache.cs index 829b1c9..fae3de1 100644 --- a/src/FrostFS.SDK.Client/Pool/SessionCache.cs +++ b/src/FrostFS.SDK.Client/Pool/SessionCache.cs @@ -1,23 +1,49 @@ using System; -using System.Collections; +using System.Collections.Concurrent; namespace FrostFS.SDK.Client; -internal struct SessionCache(ulong sessionExpirationDuration) +internal sealed class SessionCache(ulong sessionExpirationDuration) { - internal Hashtable Cache { get; } = []; + private ConcurrentDictionary _cache { get; } = []; internal ulong CurrentEpoch { get; set; } internal ulong TokenDuration { get; set; } = sessionExpirationDuration; + internal bool Contains(string key) + { + return _cache.ContainsKey(key); + } + + internal bool TryGetValue(string? key, out FrostFsSessionToken? value) + { + if (key == null) + { + value = null; + return false; + } + + var ok = _cache.TryGetValue(key, out value); + + return ok && value != null; + } + + internal void SetValue(string? key, FrostFsSessionToken value) + { + if (key != null) + { + _cache[key] = value; + } + } + internal void DeleteByPrefix(string prefix) { - foreach (var key in Cache.Keys) + foreach (var key in _cache.Keys) { - if (((string)key).StartsWith(prefix, StringComparison.Ordinal)) + if (key.StartsWith(prefix, StringComparison.Ordinal)) { - Cache.Remove(key); + _cache.TryRemove(key, out var _); } } } diff --git a/src/FrostFS.SDK.Client/Pool/WrapperPrm.cs b/src/FrostFS.SDK.Client/Pool/WrapperPrm.cs index 3c23505..b68b6ee 100644 --- a/src/FrostFS.SDK.Client/Pool/WrapperPrm.cs +++ b/src/FrostFS.SDK.Client/Pool/WrapperPrm.cs @@ -1,13 +1,14 @@ using System; +using System.Collections.ObjectModel; using System.Security.Cryptography; +using Grpc.Core.Interceptors; using Grpc.Net.Client; using Microsoft.Extensions.Logging; namespace FrostFS.SDK.Client; -// wrapperPrm is params to create clientWrapper. [System.Diagnostics.CodeAnalysis.SuppressMessage("Performance", "CA1815:Override equals and operator equals on value types", Justification = "")] public struct WrapperPrm { @@ -15,7 +16,7 @@ public struct WrapperPrm internal string Address { get; set; } - internal ECDsa? Key { get; set; } + internal ECDsa Key { get; set; } internal ulong DialTimeout { get; set; } @@ -30,5 +31,9 @@ public struct WrapperPrm internal GrpcChannelOptions GrpcChannelOptions { get; set; } internal ulong GracefulCloseOnSwitchTimeout { get; set; } + + internal Action? Callback { get; set; } + + internal Collection? Interceptors { get; set; } } diff --git a/src/FrostFS.SDK.Client/Services/AccountingServiceProvider.cs b/src/FrostFS.SDK.Client/Services/AccountingServiceProvider.cs index ba8ba2a..bdb8e61 100644 --- a/src/FrostFS.SDK.Client/Services/AccountingServiceProvider.cs +++ b/src/FrostFS.SDK.Client/Services/AccountingServiceProvider.cs @@ -24,12 +24,12 @@ internal sealed class AccountingServiceProvider : ContextAccessor { Body = new() { - OwnerId = ctx.OwnerId!.OwnerID + OwnerId = ClientContext.Owner.OwnerID } }; request.AddMetaHeader(args.XHeaders); - request.Sign(ctx.Key!); + request.Sign(ClientContext.Key.ECDsaKey); var response = await _accountingServiceClient!.BalanceAsync(request, null, ctx.Deadline, ctx.CancellationToken); diff --git a/src/FrostFS.SDK.Client/Services/ApeManagerServiceProvider.cs b/src/FrostFS.SDK.Client/Services/ApeManagerServiceProvider.cs index 840223d..1c22cdc 100644 --- a/src/FrostFS.SDK.Client/Services/ApeManagerServiceProvider.cs +++ b/src/FrostFS.SDK.Client/Services/ApeManagerServiceProvider.cs @@ -16,13 +16,9 @@ internal sealed class ApeManagerServiceProvider : ContextAccessor _apeManagerServiceClient = apeManagerServiceClient; } - internal async Task AddChainAsync(PrmApeChainAdd args) + internal async Task> AddChainAsync(PrmApeChainAdd args) { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); AddChainRequest request = new() { @@ -34,22 +30,18 @@ internal sealed class ApeManagerServiceProvider : ContextAccessor }; request.AddMetaHeader(args.XHeaders); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await _apeManagerServiceClient!.AddChainAsync(request, null, ctx.Deadline, ctx.CancellationToken); Verifier.CheckResponse(response); - return response.Body.ChainId.ToByteArray(); + return response.Body.ChainId.Memory; } internal async Task RemoveChainAsync(PrmApeChainRemove args) { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); RemoveChainRequest request = new() { @@ -61,7 +53,7 @@ internal sealed class ApeManagerServiceProvider : ContextAccessor }; request.AddMetaHeader(args.XHeaders); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await _apeManagerServiceClient!.RemoveChainAsync(request, null, ctx.Deadline, ctx.CancellationToken); @@ -71,10 +63,6 @@ internal sealed class ApeManagerServiceProvider : ContextAccessor internal async Task ListChainAsync(PrmApeChainList args) { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); ListChainsRequest request = new() { @@ -85,7 +73,7 @@ internal sealed class ApeManagerServiceProvider : ContextAccessor }; request.AddMetaHeader(args.XHeaders); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await _apeManagerServiceClient!.ListChainsAsync(request, null, ctx.Deadline, ctx.CancellationToken); diff --git a/src/FrostFS.SDK.Client/Services/ContainerServiceProvider.cs b/src/FrostFS.SDK.Client/Services/ContainerServiceProvider.cs index 3a0379e..fcdc0e5 100644 --- a/src/FrostFS.SDK.Client/Services/ContainerServiceProvider.cs +++ b/src/FrostFS.SDK.Client/Services/ContainerServiceProvider.cs @@ -1,6 +1,7 @@ using System; using System.Collections.Generic; using System.Collections.Specialized; +using System.Security.Cryptography; using System.Threading.Tasks; using FrostFS.Container; @@ -12,26 +13,34 @@ using FrostFS.Session; namespace FrostFS.SDK.Client; -internal sealed class ContainerServiceProvider(ContainerService.ContainerServiceClient service, ClientContext clientCtx) : ContextAccessor(clientCtx), ISessionProvider +internal sealed class ContainerServiceProvider(ContainerService.ContainerServiceClient service, ClientContext clientCtx) : ContextAccessor(clientCtx) { private SessionProvider? sessions; - public async ValueTask GetOrCreateSession(ISessionToken args, CallContext ctx) + public async ValueTask GetDefaultSession(ISessionToken args, CallContext ctx) { sessions ??= new(ClientContext); - if (ClientContext.SessionCache.Cache != null && - ClientContext.SessionCache.Cache.ContainsKey(ClientContext.SessionCacheKey)) + if (!ClientContext.SessionCache!.TryGetValue(ClientContext.SessionCacheKey, out var token)) { - return (SessionToken)ClientContext.SessionCache.Cache[ClientContext.SessionCacheKey]; + var protoToken = await sessions.GetDefaultSession(args, ctx).ConfigureAwait(false); + + token = new FrostFsSessionToken(protoToken); + + ClientContext.SessionCache.SetValue(ClientContext.SessionCacheKey, token); } - return await sessions.GetOrCreateSession(args, ctx).ConfigureAwait(false); + if (token == null) + { + throw new FrostFsException("Cannot create session"); + } + + return token; } internal async Task GetContainerAsync(PrmContainerGet args) { - GetRequest request = GetContainerRequest(args.Container.ContainerID, args.XHeaders, args.Context); + GetRequest request = GetContainerRequest(args.Container.ContainerID, args.XHeaders, ClientContext.Key.ECDsaKey); var response = await service.GetAsync(request, null, args.Context.Deadline, args.Context.CancellationToken); @@ -43,24 +52,17 @@ internal sealed class ContainerServiceProvider(ContainerService.ContainerService internal async IAsyncEnumerable ListContainersAsync(PrmContainerGetAll args) { var ctx = args.Context!; - ctx.OwnerId ??= ClientContext.Owner; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); - if (ctx.OwnerId == null) - throw new ArgumentException(nameof(ctx.OwnerId)); var request = new ListRequest { Body = new() { - OwnerId = ctx.OwnerId.ToMessage() + OwnerId = ClientContext.Owner.OwnerID } }; request.AddMetaHeader(args.XHeaders); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await service.ListAsync(request, null, ctx.Deadline, ctx.CancellationToken); @@ -68,7 +70,7 @@ internal sealed class ContainerServiceProvider(ContainerService.ContainerService foreach (var cid in response.Body.ContainerIds) { - yield return new FrostFsContainerId(Base58.Encode(cid.Value.ToByteArray())); + yield return new FrostFsContainerId(Base58.Encode(cid.Value.Span)); } } @@ -78,36 +80,28 @@ internal sealed class ContainerServiceProvider(ContainerService.ContainerService var grpcContainer = args.Container.GetContainer(); - grpcContainer.OwnerId ??= ctx.OwnerId?.ToMessage(); - grpcContainer.Version ??= ctx.Version?.ToMessage(); - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); - if (grpcContainer.OwnerId == null) - throw new ArgumentException(nameof(grpcContainer.OwnerId)); - if (grpcContainer.Version == null) - throw new ArgumentException(nameof(grpcContainer.Version)); + grpcContainer.OwnerId ??= ClientContext.Owner.OwnerID; + grpcContainer.Version ??= ClientContext.Version.VersionID; var request = new PutRequest { Body = new PutRequest.Types.Body { Container = grpcContainer, - Signature = ctx.Key.SignRFC6979(grpcContainer) + Signature = ClientContext.Key.ECDsaKey.SignRFC6979(grpcContainer) } }; - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = (args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false)) ?? throw new FrostFsException("Cannot create session token"); - sessionToken.CreateContainerTokenContext( + var protoToken = sessionToken.CreateContainerToken( null, ContainerSessionContext.Types.Verb.Put, - ctx.Key, - ctx.GetPublicKeyCache()!); + ClientContext.Key); - request.AddMetaHeader(args.XHeaders, sessionToken); + request.AddMetaHeader(args.XHeaders, protoToken); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await service.PutAsync(request, null, ctx.Deadline, ctx.CancellationToken); @@ -121,29 +115,26 @@ internal sealed class ContainerServiceProvider(ContainerService.ContainerService internal async Task DeleteContainerAsync(PrmContainerDelete args) { var ctx = args.Context!; - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); var request = new DeleteRequest { Body = new DeleteRequest.Types.Body { ContainerId = args.ContainerId.ToMessage(), - Signature = ctx.Key.SignRFC6979(args.ContainerId.ToMessage().Value) + Signature = ClientContext.Key.ECDsaKey.SignRFC6979(args.ContainerId.ToMessage().Value) } }; - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false); - sessionToken.CreateContainerTokenContext( + var protoToken = sessionToken.CreateContainerToken( request.Body.ContainerId, ContainerSessionContext.Types.Verb.Delete, - ctx.Key, - ctx.GetPublicKeyCache()!); + ClientContext.Key); - request.AddMetaHeader(args.XHeaders, sessionToken); + request.AddMetaHeader(args.XHeaders, protoToken); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await service.DeleteAsync(request, null, ctx.Deadline, ctx.CancellationToken); @@ -155,11 +146,8 @@ internal sealed class ContainerServiceProvider(ContainerService.ContainerService Verifier.CheckResponse(response); } - private static GetRequest GetContainerRequest(ContainerID id, NameValueCollection? xHeaders, CallContext ctx) + private static GetRequest GetContainerRequest(ContainerID id, NameValueCollection? xHeaders, ECDsa key) { - if (ctx.Key == null) - throw new ArgumentNullException(nameof(ctx), "Key is null"); - var request = new GetRequest { Body = new GetRequest.Types.Body @@ -169,7 +157,7 @@ internal sealed class ContainerServiceProvider(ContainerService.ContainerService }; request.AddMetaHeader(xHeaders); - request.Sign(ctx.Key); + request.Sign(key); return request; } @@ -182,7 +170,7 @@ internal sealed class ContainerServiceProvider(ContainerService.ContainerService private async Task WaitForContainer(WaitExpects expect, ContainerID id, PrmWait? waitParams, CallContext ctx) { - var request = GetContainerRequest(id, null, ctx); + var request = GetContainerRequest(id, null, ClientContext.Key.ECDsaKey); async Task action() { diff --git a/src/FrostFS.SDK.Client/Services/NetmapServiceProvider.cs b/src/FrostFS.SDK.Client/Services/NetmapServiceProvider.cs index 18649d8..570bd24 100644 --- a/src/FrostFS.SDK.Client/Services/NetmapServiceProvider.cs +++ b/src/FrostFS.SDK.Client/Services/NetmapServiceProvider.cs @@ -1,6 +1,4 @@ using System; -using System.Linq; -using System.Text; using System.Threading.Tasks; using FrostFS.Netmap; @@ -47,10 +45,6 @@ internal sealed class NetmapServiceProvider : ContextAccessor internal async Task GetLocalNodeInfoAsync(PrmNodeInfo args) { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); var request = new LocalNodeInfoRequest { @@ -58,7 +52,7 @@ internal sealed class NetmapServiceProvider : ContextAccessor }; request.AddMetaHeader(args.XHeaders); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await netmapServiceClient.LocalNodeInfoAsync(request, null, ctx.Deadline, ctx.CancellationToken); @@ -69,15 +63,10 @@ internal sealed class NetmapServiceProvider : ContextAccessor internal async Task GetNetworkInfoAsync(CallContext ctx) { - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(ctx), "Key is null"); - var request = new NetworkInfoRequest(); request.AddMetaHeader(null); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await netmapServiceClient.NetworkInfoAsync(request, null, ctx.Deadline, ctx.CancellationToken) .ConfigureAwait(false); @@ -90,15 +79,11 @@ internal sealed class NetmapServiceProvider : ContextAccessor internal async Task GetNetmapSnapshotAsync(PrmNetmapSnapshot args) { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); var request = new NetmapSnapshotRequest(); request.AddMetaHeader(args.XHeaders); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await netmapServiceClient.NetmapSnapshotAsync(request, null, ctx.Deadline, ctx.CancellationToken); @@ -107,12 +92,16 @@ internal sealed class NetmapServiceProvider : ContextAccessor return response.ToModel(); } - private static bool GetBoolValue(byte[] bytes) + private static bool GetBoolValue(ReadOnlySpan bytes) { - return bytes.Any(b => b != 0); + for (int i = bytes.Length - 1; i >= 0; i--) + if (bytes[i] != 0) + return true; + + return false; } - private static ulong GetLongValue(byte[] bytes) + private static ulong GetLongValue(ReadOnlySpan bytes) { ulong val = 0; for (var i = bytes.Length - 1; i >= 0; i--) @@ -123,24 +112,50 @@ internal sealed class NetmapServiceProvider : ContextAccessor private static void SetNetworksParam(Parameter param, NetworkSettings settings) { - var key = Encoding.UTF8.GetString(param.Key.ToByteArray()); + var key = param.Key.ToStringUtf8(); - var valueBytes = param.Value.ToByteArray(); + var valueBytes = param.Value.Span; switch (key) { - case "AuditFee": settings.AuditFee = GetLongValue(valueBytes); break; - case "BasicIncomeRate": settings.BasicIncomeRate = GetLongValue(valueBytes); break; - case "ContainerFee": settings.ContainerFee = GetLongValue(valueBytes); break; - case "ContainerAliasFee": settings.ContainerAliasFee = GetLongValue(valueBytes); break; - case "EpochDuration": settings.EpochDuration = GetLongValue(valueBytes); break; - case "InnerRingCandidateFee": settings.InnerRingCandidateFee = GetLongValue(valueBytes); break; - case "MaxECDataCount": settings.MaxECDataCount = GetLongValue(valueBytes); break; - case "MaxECParityCount": settings.MaxECParityCount = GetLongValue(valueBytes); break; - case "MaxObjectSize": settings.MaxObjectSize = GetLongValue(valueBytes); break; - case "WithdrawFee": settings.WithdrawFee = GetLongValue(valueBytes); break; - case "HomomorphicHashingDisabled": settings.HomomorphicHashingDisabled = GetBoolValue(valueBytes); break; - case "MaintenanceModeAllowed": settings.MaintenanceModeAllowed = GetBoolValue(valueBytes); break; - default: settings.UnnamedSettings.Add(key, valueBytes); break; + case "AuditFee": + settings.AuditFee = GetLongValue(valueBytes); + break; + case "BasicIncomeRate": + settings.BasicIncomeRate = GetLongValue(valueBytes); + break; + case "ContainerFee": + settings.ContainerFee = GetLongValue(valueBytes); + break; + case "ContainerAliasFee": + settings.ContainerAliasFee = GetLongValue(valueBytes); + break; + case "EpochDuration": + settings.EpochDuration = GetLongValue(valueBytes); + break; + case "InnerRingCandidateFee": + settings.InnerRingCandidateFee = GetLongValue(valueBytes); + break; + case "MaxECDataCount": + settings.MaxECDataCount = GetLongValue(valueBytes); + break; + case "MaxECParityCount": + settings.MaxECParityCount = GetLongValue(valueBytes); + break; + case "MaxObjectSize": + settings.MaxObjectSize = GetLongValue(valueBytes); + break; + case "WithdrawFee": + settings.WithdrawFee = GetLongValue(valueBytes); + break; + case "HomomorphicHashingDisabled": + settings.HomomorphicHashingDisabled = GetBoolValue(valueBytes); + break; + case "MaintenanceModeAllowed": + settings.MaintenanceModeAllowed = GetBoolValue(valueBytes); + break; + default: + settings.UnnamedSettings.Add(key, valueBytes.ToArray()); + break; } } } diff --git a/src/FrostFS.SDK.Client/Services/ObjectServiceProvider.cs b/src/FrostFS.SDK.Client/Services/ObjectServiceProvider.cs index 7737c62..d3c4347 100644 --- a/src/FrostFS.SDK.Client/Services/ObjectServiceProvider.cs +++ b/src/FrostFS.SDK.Client/Services/ObjectServiceProvider.cs @@ -17,31 +17,35 @@ using Google.Protobuf; namespace FrostFS.SDK.Client; internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient client, ClientContext clientCtx) - : ContextAccessor(clientCtx), ISessionProvider + : ContextAccessor(clientCtx) { private SessionProvider? sessions; private readonly ObjectService.ObjectServiceClient client = client; - public async ValueTask GetOrCreateSession(ISessionToken args, CallContext ctx) + public async ValueTask GetDefaultSession(ISessionToken args, CallContext ctx) { sessions ??= new(ClientContext); - if (ClientContext.SessionCache.Cache != null && - ClientContext.SessionCache.Cache.ContainsKey(ClientContext.SessionCacheKey)) + if (!ClientContext.SessionCache!.TryGetValue(ClientContext.SessionCacheKey, out var token)) { - return (SessionToken)ClientContext.SessionCache.Cache[ClientContext.SessionCacheKey]; + var protoToken = await sessions.GetDefaultSession(args, ctx).ConfigureAwait(false); + + token = new FrostFsSessionToken(protoToken); + + ClientContext.SessionCache.SetValue(ClientContext.SessionCacheKey, token); } - return await sessions.GetOrCreateSession(args, ctx).ConfigureAwait(false); + if (token == null) + { + throw new FrostFsException("Cannot create session"); + } + + return token; } internal async Task GetObjectHeadAsync(PrmObjectHeadGet args) { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); var request = new HeadRequest { @@ -55,16 +59,16 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl } }; - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false); - sessionToken.CreateObjectTokenContext( + var protoToken = sessionToken.CreateObjectTokenContext( request.Body.Address, ObjectSessionContext.Types.Verb.Head, - ctx.Key); + ClientContext.Key); - request.AddMetaHeader(args.XHeaders, sessionToken); + request.AddMetaHeader(args.XHeaders, protoToken); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await client!.HeadAsync(request, null, ctx.Deadline, ctx.CancellationToken).ConfigureAwait(false); @@ -77,11 +81,6 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); - var request = new GetRequest { Body = new GetRequest.Types.Body @@ -94,16 +93,16 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl } }; - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false); - sessionToken.CreateObjectTokenContext( + var protoToken = sessionToken.CreateObjectTokenContext( request.Body.Address, ObjectSessionContext.Types.Verb.Get, - ctx.Key); + ClientContext.Key); - request.AddMetaHeader(args.XHeaders, sessionToken); + request.AddMetaHeader(args.XHeaders, protoToken); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); return await GetObject(request, ctx).ConfigureAwait(false); } @@ -112,11 +111,6 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); - var request = new GetRangeRequest { Body = new GetRangeRequest.Types.Body @@ -135,16 +129,16 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl } }; - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false); - sessionToken.CreateObjectTokenContext( + var protoToken = sessionToken.CreateObjectTokenContext( request.Body.Address, ObjectSessionContext.Types.Verb.Range, - ctx.Key); + ClientContext.Key); - request.AddMetaHeader(args.XHeaders, sessionToken); + request.AddMetaHeader(args.XHeaders, protoToken); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var call = client.GetRange(request, null, ctx.Deadline, ctx.CancellationToken); return new RangeReader(call); @@ -154,11 +148,6 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); - var request = new GetRangeHashRequest { Body = new GetRangeHashRequest.Types.Body @@ -182,16 +171,16 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl }); } - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false); - sessionToken.CreateObjectTokenContext( + var protoToken = sessionToken.CreateObjectTokenContext( request.Body.Address, ObjectSessionContext.Types.Verb.Rangehash, - ctx.Key); + ClientContext.Key); - request.AddMetaHeader(args.XHeaders, sessionToken); + request.AddMetaHeader(args.XHeaders, protoToken); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await client.GetRangeHashAsync(request, null, ctx.Deadline, ctx.CancellationToken); @@ -206,10 +195,6 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl internal async Task DeleteObjectAsync(PrmObjectDelete args) { var ctx = args.Context!; - ctx.Key ??= ClientContext.Key?.ECDsaKey; - - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); var request = new DeleteRequest { @@ -223,15 +208,15 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl } }; - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false); - sessionToken.CreateObjectTokenContext( + var protoToken = sessionToken.CreateObjectTokenContext( request.Body.Address, ObjectSessionContext.Types.Verb.Delete, - ctx.Key); + ClientContext.Key); - request.AddMetaHeader(args.XHeaders, sessionToken); - request.Sign(ctx.Key); + request.AddMetaHeader(args.XHeaders, protoToken); + request.Sign(ClientContext.Key.ECDsaKey); var response = await client.DeleteAsync(request, null, ctx.Deadline, ctx.CancellationToken); @@ -242,9 +227,6 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl { var ctx = args.Context!; - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); - var request = new SearchRequest { Body = new SearchRequest.Types.Body @@ -256,22 +238,30 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl request.Body.Filters.AddRange(args.Filters.Select(f => f.ToMessage())); - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false); - sessionToken.CreateObjectTokenContext( + var protoToken = sessionToken.CreateObjectTokenContext( new Address { ContainerId = request.Body.ContainerId }, ObjectSessionContext.Types.Verb.Search, - ctx.Key); + ClientContext.Key); - request.AddMetaHeader(args.XHeaders, sessionToken); + request.AddMetaHeader(args.XHeaders, protoToken); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); - var objectsIds = SearchObjects(request, ctx); + using var stream = GetSearchReader(request, ctx); - await foreach (var oid in objectsIds) + while (true) { - yield return FrostFsObjectId.FromHash(oid.Value.ToByteArray()); + var ids = await stream.Read(ctx.CancellationToken).ConfigureAwait(false); + + if (ids == null) + yield break; + + foreach (var oid in ids) + { + yield return FrostFsObjectId.FromHash(oid.Value.Span); + } } } @@ -296,6 +286,8 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl args.FullLength = args.Header.PayloadLength; else if (args.Payload.CanSeek) args.FullLength = (ulong)args.Payload.Length; + else + throw new ArgumentException("The stream does not have a length and payload length is not defined"); var response = await PutStreamObject(args).ConfigureAwait(false); @@ -307,39 +299,34 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl { var ctx = args.Context!; - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); - - var grpcObject = ObjectTools.CreateObject(args.FrostFsObject, ctx); + var grpcObject = ObjectTools.CreateObject(args.FrostFsObject, ClientContext); var request = new PutSingleRequest { Body = new() { Object = grpcObject } }; - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false); - sessionToken.CreateObjectTokenContext( + var protoToken = sessionToken.CreateObjectTokenContext( new Address { ContainerId = grpcObject.Header.ContainerId, ObjectId = grpcObject.ObjectId }, ObjectSessionContext.Types.Verb.Put, - ctx.Key); + ClientContext.Key); - request.AddMetaHeader(args.XHeaders, sessionToken); + request.AddMetaHeader(args.XHeaders, protoToken); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); var response = await client.PutSingleAsync(request, null, ctx.Deadline, ctx.CancellationToken).ConfigureAwait(false); Verifier.CheckResponse(response); - return FrostFsObjectId.FromHash(grpcObject.ObjectId.Value.ToByteArray()); + return FrostFsObjectId.FromHash(grpcObject.ObjectId.Value.Span); } internal async Task PatchObjectAsync(PrmObjectPatch args) { var ctx = args.Context!; - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); var chunkSize = args.MaxPayloadPatchChunkLength; Stream payload = args.Payload ?? throw new ArgumentNullException(nameof(args), "Stream parameter is null"); @@ -350,7 +337,7 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl try { // common - chunkBuffer = ClientContext.GetArrayPool(Constants.ObjectChunkSize).Rent(chunkSize); + chunkBuffer = ArrayPool.Shared.Rent(chunkSize); var address = new Address { @@ -358,13 +345,12 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl ContainerId = args.Address.ContainerId }; - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false); - sessionToken.CreateObjectTokenContext( + var protoToken = sessionToken.CreateObjectTokenContext( address, ObjectSessionContext.Types.Verb.Patch, - ctx.Key - ); + ClientContext.Key); var request = new PatchRequest() { @@ -403,9 +389,9 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl currentPos += (ulong)bytesCount; - request.AddMetaHeader(args.XHeaders, sessionToken); + request.AddMetaHeader(args.XHeaders, protoToken); - request.Sign(ctx.Key); + request.Sign(ClientContext.Key.ECDsaKey); await call.RequestStream.WriteAsync(request).ConfigureAwait(false); @@ -433,20 +419,17 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl { var ctx = args.Context!; - var tokenRaw = await GetOrCreateSession(args, ctx).ConfigureAwait(false); - var token = new FrostFsSessionToken(tokenRaw.Serialize(), tokenRaw.Body.Id.ToUuid()); - - args.SessionToken = token; + args.SessionToken ??= await GetDefaultSession(args, ctx).ConfigureAwait(false); var payloadStream = args.Payload!; var header = args.Header!; - var fullLength = header.PayloadLength; - - if (payloadStream.CanSeek && fullLength == 0) - fullLength = (ulong)payloadStream.Length; - - args.FullLength = fullLength; + if (header.PayloadLength > 0) + args.FullLength = header.PayloadLength; + else if (payloadStream.CanSeek) + args.FullLength = (ulong)payloadStream.Length; + else + throw new ArgumentException("The stream does not have a length and payload length is not defined"); if (args.MaxObjectSizeCache == 0) { @@ -456,12 +439,12 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl args.MaxObjectSizeCache = (int)networkSettings.MaxObjectSize; } - var restBytes = fullLength - args.CurrentStreamPosition; + var restBytes = args.FullLength - args.CurrentStreamPosition; var objectSize = restBytes > 0 ? (int)Math.Min((ulong)args.MaxObjectSizeCache, restBytes) : args.MaxObjectSizeCache; //define collection capacity var restPart = (restBytes % (ulong)objectSize) > 0 ? 1 : 0; - var objectsCount = fullLength > 0 ? (int)(restBytes / (ulong)objectSize) + restPart : 0; + var objectsCount = args.FullLength > 0 ? (int)(restBytes / (ulong)objectSize) + restPart : 0; List sentObjectIds = new(objectsCount); @@ -491,7 +474,7 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl { var largeObjectHeader = new FrostFsObjectHeader(header.ContainerId, FrostFsObjectType.Regular, [.. attributes]) { - PayloadLength = fullLength, + PayloadLength = args.FullLength, }; args.Header.Split!.ParentHeader = largeObjectHeader; @@ -526,8 +509,6 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl private async Task PutStreamObject(PrmObjectPut args) { var ctx = args.Context!; - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); var payload = args.Payload!; @@ -542,21 +523,26 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl try { + // 0 means no limit from client, so server side cut is performed + var objectLimitSize = args.ClientCut ? args.MaxObjectSizeCache : 0; + if (args.CustomBuffer != null) { + if (args.CustomBuffer.Length < chunkSize) + { + throw new ArgumentException($"Buffer size is too small. At least {chunkSize} required"); + } + chunkBuffer = args.CustomBuffer; } else { - chunkBuffer = ClientContext.GetArrayPool(Constants.ObjectChunkSize).Rent(chunkSize); + chunkBuffer = ArrayPool.Shared.Rent(chunkSize); isRentBuffer = true; } var sentBytes = 0; - // 0 means no limit from client, so server side cut is performed - var objectLimitSize = args.ClientCut ? args.MaxObjectSizeCache : 0; - using var stream = await GetUploadStream(args, ctx).ConfigureAwait(false); while (objectLimitSize == 0 || sentBytes < objectLimitSize) @@ -581,7 +567,7 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl } }; - chunkRequest.Sign(ctx.Key); + chunkRequest.Sign(ClientContext.Key.ECDsaKey); await stream.Write(chunkRequest).ConfigureAwait(false); } @@ -589,7 +575,7 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl var response = await stream.Close().ConfigureAwait(false); Verifier.CheckResponse(response); - return new PutObjectResult(FrostFsObjectId.FromHash(response.Body.ObjectId.Value.ToByteArray()), sentBytes); + return new PutObjectResult(FrostFsObjectId.FromHash(response.Body.ObjectId.Value.Span), sentBytes); } finally { @@ -604,17 +590,14 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl { var header = args.Header!; - if (ctx.Key == null) - throw new ArgumentNullException(nameof(args), "Key is null"); - - header.OwnerId ??= ctx.OwnerId; - header.Version ??= ctx.Version; + header.OwnerId ??= ClientContext.Owner; + header.Version ??= ClientContext.Version; var grpcHeader = header.GetHeader(); if (header.Split != null) { - ObjectTools.SetSplitValues(grpcHeader, header.Split, ctx); + ObjectTools.SetSplitValues(grpcHeader, header.Split, ClientContext); } var oid = new ObjectID { Value = grpcHeader.Sha256() }; @@ -630,17 +613,16 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl } }; - var sessionToken = await GetOrCreateSession(args, ctx).ConfigureAwait(false); + var sessionToken = (args.SessionToken ?? await GetDefaultSession(args, ctx).ConfigureAwait(false)); - sessionToken.CreateObjectTokenContext( + var protoToken = sessionToken.CreateObjectTokenContext( new Address { ContainerId = grpcHeader.ContainerId, ObjectId = oid }, ObjectSessionContext.Types.Verb.Put, - ctx.Key - ); + ClientContext.Key); - initRequest.AddMetaHeader(args.XHeaders, sessionToken); + initRequest.AddMetaHeader(args.XHeaders, protoToken); - initRequest.Sign(ctx.Key); + initRequest.Sign(ClientContext.Key.ECDsaKey); return await PutObjectInit(initRequest, ctx).ConfigureAwait(false); } @@ -681,24 +663,6 @@ internal sealed class ObjectServiceProvider(ObjectService.ObjectServiceClient cl return new ObjectReader(call); } - private async IAsyncEnumerable SearchObjects(SearchRequest request, CallContext ctx) - { - using var stream = GetSearchReader(request, ctx); - - while (true) - { - var ids = await stream.Read(ctx.CancellationToken).ConfigureAwait(false); - - if (ids == null) - break; - - foreach (var oid in ids) - { - yield return oid; - } - } - } - private SearchReader GetSearchReader(SearchRequest initRequest, CallContext ctx) { if (initRequest is null) diff --git a/src/FrostFS.SDK.Client/Services/SessionServiceProvider.cs b/src/FrostFS.SDK.Client/Services/SessionServiceProvider.cs index e9b5963..859d836 100644 --- a/src/FrostFS.SDK.Client/Services/SessionServiceProvider.cs +++ b/src/FrostFS.SDK.Client/Services/SessionServiceProvider.cs @@ -1,7 +1,5 @@ using System.Threading.Tasks; -using FrostFS.SDK.Client; -using FrostFS.SDK.Client.Mappers.GRPC; using FrostFS.Session; namespace FrostFS.SDK.Client; @@ -20,19 +18,17 @@ internal sealed class SessionServiceProvider : ContextAccessor { var ctx = args.Context!; - ctx.OwnerId ??= ClientContext.Owner; - var request = new CreateRequest { Body = new CreateRequest.Types.Body { - OwnerId = ctx.OwnerId!.ToMessage(), + OwnerId = ClientContext.Owner.OwnerID, Expiration = args.Expiration } }; request.AddMetaHeader(args.XHeaders); - request.Sign(ctx.Key!); + request.Sign(ClientContext.Key.ECDsaKey); return await CreateSession(request, args.Context!).ConfigureAwait(false); } diff --git a/src/FrostFS.SDK.Client/Services/Shared/SessionProvider.cs b/src/FrostFS.SDK.Client/Services/Shared/SessionProvider.cs index a8d80ac..e8ff793 100644 --- a/src/FrostFS.SDK.Client/Services/Shared/SessionProvider.cs +++ b/src/FrostFS.SDK.Client/Services/Shared/SessionProvider.cs @@ -9,14 +9,15 @@ internal interface ISessionProvider internal sealed class SessionProvider(ClientContext envCtx) { - public async ValueTask GetOrCreateSession(ISessionToken args, CallContext ctx) + public async Task CreateSession(ISessionToken args, CallContext ctx) { - if (args.SessionToken is null) - { - return await envCtx.Client.CreateSessionInternalAsync(new PrmSessionCreate(uint.MaxValue, ctx)) - .ConfigureAwait(false); - } + var token = await GetDefaultSession(args, ctx).ConfigureAwait(false); - return new Session.SessionToken().Deserialize(args.SessionToken.Token); + return new FrostFsSessionToken(token); + } + + internal async Task GetDefaultSession(ISessionToken args, CallContext ctx) + { + return await envCtx.Client.CreateSessionInternalAsync(new PrmSessionCreate(uint.MaxValue, ctx)).ConfigureAwait(false); } } \ No newline at end of file diff --git a/src/FrostFS.SDK.Client/Tools/ClientContext.cs b/src/FrostFS.SDK.Client/Tools/ClientContext.cs index 5f7c72e..133ad3f 100644 --- a/src/FrostFS.SDK.Client/Tools/ClientContext.cs +++ b/src/FrostFS.SDK.Client/Tools/ClientContext.cs @@ -1,19 +1,16 @@ using System; -using System.Buffers; -using System.Security.Cryptography; - -using FrostFS.SDK.Cryptography; +using System.Collections.ObjectModel; +using Grpc.Core.Interceptors; using Grpc.Net.Client; namespace FrostFS.SDK.Client; -public class ClientContext(FrostFSClient client, ECDsa? key, FrostFsOwner? owner, GrpcChannel channel, FrostFsVersion version) : IDisposable +public class ClientContext(FrostFSClient client, ClientKey key, FrostFsOwner owner, GrpcChannel channel, FrostFsVersion version) : IDisposable { - private ArrayPool? _arrayPool; private string? sessionKey; - internal FrostFsOwner? Owner { get; } = owner; + internal FrostFsOwner Owner { get; } = owner; internal string? Address { get; } = channel.Target; @@ -25,9 +22,16 @@ public class ClientContext(FrostFSClient client, ECDsa? key, FrostFsOwner? owner internal FrostFSClient Client { get; } = client; - internal ClientKey? Key { get; } = key != null ? new ClientKey(key) : null; + internal ClientKey Key { get; } = key; + + internal SessionCache? SessionCache { get; set; } + + internal Action? Callback { get; set; } + + internal Collection? Interceptors { get; set; } + + internal Action? PoolErrorHandler { get; set; } - internal SessionCache SessionCache { get; set; } internal string? SessionCacheKey { @@ -35,23 +39,13 @@ public class ClientContext(FrostFSClient client, ECDsa? key, FrostFsOwner? owner { if (sessionKey == null && Key != null && Address != null) { - sessionKey = Pool.FormCacheKey(Address, Key.ECDsaKey.PrivateKey().ToString()); + sessionKey = Pool.FormCacheKey(Address, Key.PublicKey); } return sessionKey; } } - /// - /// Custom pool is used for predefined sizes of buffers like grpc chunk - /// - internal ArrayPool GetArrayPool(int size) - { - _arrayPool ??= ArrayPool.Create(size, 256); - - return _arrayPool; - } - public void Dispose() { Dispose(true); diff --git a/src/FrostFS.SDK.Client/Tools/ObjectTools.cs b/src/FrostFS.SDK.Client/Tools/ObjectTools.cs index f20cacd..eeb06c0 100644 --- a/src/FrostFS.SDK.Client/Tools/ObjectTools.cs +++ b/src/FrostFS.SDK.Client/Tools/ObjectTools.cs @@ -11,7 +11,7 @@ namespace FrostFS.SDK.Client; internal static class ObjectTools { - internal static FrostFsObjectId CalculateObjectId(FrostFsObjectHeader header, CallContext ctx) + internal static FrostFsObjectId CalculateObjectId(FrostFsObjectHeader header, ClientContext ctx) { var grpcHeader = CreateHeader(header, [], ctx); @@ -21,9 +21,9 @@ internal static class ObjectTools return new ObjectID { Value = grpcHeader.Sha256() }.ToModel(); } - internal static Object.Object CreateObject(FrostFsObject @object, CallContext ctx) + internal static Object.Object CreateObject(FrostFsObject @object, ClientContext ctx) { - @object.Header.OwnerId ??= ctx.OwnerId; + @object.Header.OwnerId ??= ctx.Owner; @object.Header.Version ??= ctx.Version; var grpcHeader = @object.Header.GetHeader(); @@ -46,14 +46,14 @@ internal static class ObjectTools obj.Signature = new Signature { - Key = ctx.GetPublicKeyCache(), - Sign = ByteString.CopyFrom(ctx.Key!.SignData(obj.ObjectId.ToByteArray())), + Key = ctx.Key.PublicKeyProto, + Sign = ctx.Key.ECDsaKey.SignData(obj.ObjectId.ToByteArray()), }; return obj; } - internal static void SetSplitValues(Header grpcHeader, FrostFsSplit split, CallContext ctx) + internal static void SetSplitValues(Header grpcHeader, FrostFsSplit split, ClientContext ctx) { if (split == null) return; @@ -77,17 +77,17 @@ internal static class ObjectTools grpcHeader.Split.ParentHeader = grpcParentHeader; grpcHeader.Split.ParentSignature = new Signature { - Key = ctx.GetPublicKeyCache(), - Sign = ByteString.CopyFrom(ctx.Key.SignData(grpcHeader.Split.Parent.ToByteArray())), + Key = ctx.Key.PublicKeyProto, + Sign = ctx.Key.ECDsaKey.SignData(grpcHeader.Split.Parent.ToByteArray()), }; } grpcHeader.Split.Previous = split.Previous?.ToMessage(); } - internal static Header CreateHeader(FrostFsObjectHeader header, byte[]? payload, CallContext ctx) + internal static Header CreateHeader(FrostFsObjectHeader header, byte[]? payload, ClientContext ctx) { - header.OwnerId ??= ctx.OwnerId; + header.OwnerId ??= ctx.Owner; header.Version ??= ctx.Version; var grpcHeader = header.GetHeader(); diff --git a/src/FrostFS.SDK.Client/Tools/RequestConstructor.cs b/src/FrostFS.SDK.Client/Tools/RequestConstructor.cs index 65d9e35..64aecc6 100644 --- a/src/FrostFS.SDK.Client/Tools/RequestConstructor.cs +++ b/src/FrostFS.SDK.Client/Tools/RequestConstructor.cs @@ -1,11 +1,8 @@ using System; using System.Collections.Specialized; using System.Linq; -using System.Security.Cryptography; -using FrostFS.Refs; using FrostFS.SDK.Client.Mappers.GRPC; -using FrostFS.SDK.Cryptography; using FrostFS.SDK.Proto.Interfaces; using FrostFS.Session; @@ -33,59 +30,4 @@ public static class RequestConstructor xHeaders.Cast().SelectMany(key => xHeaders.GetValues(key), (k, v) => new XHeader { Key = k, Value = v })); } - - public static void CreateObjectTokenContext(this SessionToken sessionToken, - Address address, - ObjectSessionContext.Types.Verb verb, - ECDsa key) - { - if (sessionToken is null) - { - throw new ArgumentNullException(nameof(sessionToken)); - } - - if (address is null) - { - throw new ArgumentNullException(nameof(address)); - } - - if (sessionToken.Body.Object?.Target != null) - return; - - ObjectSessionContext.Types.Target target = new() { Container = address.ContainerId }; - - if (address.ObjectId != null) - target.Objects.Add(address.ObjectId); - - sessionToken.Body.Object = new() - { - Target = target, - Verb = verb - }; - - sessionToken.Body.SessionKey = Google.Protobuf.ByteString.CopyFrom(key.PublicKey()); - - sessionToken.Signature = key.SignMessagePart(sessionToken.Body); - } - - internal static void CreateContainerTokenContext(this SessionToken sessionToken, - ContainerID? containerId, - ContainerSessionContext.Types.Verb verb, - ECDsa key, - Google.Protobuf.ByteString publicKey) - { - if (sessionToken.Body.Container?.ContainerId != null) - return; - - sessionToken.Body.Container = new() { Verb = verb }; - - if (containerId != null) - sessionToken.Body.Container.ContainerId = containerId; - else - sessionToken.Body.Container.Wildcard = true; - - sessionToken.Body.SessionKey = publicKey; - - sessionToken.Signature = key.SignMessagePart(sessionToken.Body); - } } diff --git a/src/FrostFS.SDK.Client/Tools/RequestSigner.cs b/src/FrostFS.SDK.Client/Tools/RequestSigner.cs index a45c569..f747b3f 100644 --- a/src/FrostFS.SDK.Client/Tools/RequestSigner.cs +++ b/src/FrostFS.SDK.Client/Tools/RequestSigner.cs @@ -14,24 +14,21 @@ using Org.BouncyCastle.Crypto.Parameters; using Org.BouncyCastle.Crypto.Signers; using Org.BouncyCastle.Math; +using Signature = FrostFS.Refs.Signature; + namespace FrostFS.SDK.Client; public static class RequestSigner { internal const int RFC6979SignatureSize = 64; - internal static byte[] SignRFC6979(this ECDsa key, byte[] data) + internal static ByteString SignRFC6979(this ECDsa key, byte[] data) { if (key is null) { throw new ArgumentNullException(nameof(key)); } - if (data is null) - { - throw new ArgumentNullException(nameof(data)); - } - var digest = new Sha256Digest(); var secp256R1 = SecNamedCurves.GetByName("secp256r1"); var ecParameters = new ECDomainParameters(secp256R1.Curve, secp256R1.G, secp256R1.N); @@ -44,24 +41,27 @@ public static class RequestSigner signer.Init(true, privateKey); var rs = signer.GenerateSignature(hash); - var signature = new byte[RFC6979SignatureSize]; + + Span signature = stackalloc byte[RFC6979SignatureSize]; + var rbytes = rs[0].ToByteArrayUnsigned(); var sbytes = rs[1].ToByteArrayUnsigned(); var index = RFC6979SignatureSize / 2 - rbytes.Length; - rbytes.CopyTo(signature, index); + rbytes.AsSpan().CopyTo(signature[index..]); index = RFC6979SignatureSize - sbytes.Length; - sbytes.CopyTo(signature, index); + sbytes.AsSpan().CopyTo(signature[index..]); - return signature; + return ByteString.CopyFrom(signature); } + internal static SignatureRFC6979 SignRFC6979(this ECDsa key, IMessage message) { return new SignatureRFC6979 { Key = ByteString.CopyFrom(key.PublicKey()), - Sign = ByteString.CopyFrom(key.SignRFC6979(message.ToByteArray())), + Sign = key.SignRFC6979(message.ToByteArray()), }; } @@ -70,23 +70,26 @@ public static class RequestSigner return new SignatureRFC6979 { Key = ByteString.CopyFrom(key.PublicKey()), - Sign = ByteString.CopyFrom(key.SignRFC6979(data.ToByteArray())), + Sign = key.SignRFC6979(data.ToByteArray()), }; } - public static byte[] SignData(this ECDsa key, byte[] data) + public static ByteString SignData(this ECDsa key, byte[] data) { if (key is null) { throw new ArgumentNullException(nameof(key)); } - var hash = new byte[65]; - hash[0] = 0x04; + Span result = stackalloc byte[65]; + result[0] = 0x04; - key.SignHash(data.Sha512()).CopyTo(hash, 1); + //var hash = new byte[65]; + //hash[0] = 0x04; - return hash; + key.SignHash(data.Sha512()).AsSpan().CopyTo(result[1..]); + + return ByteString.CopyFrom(result); } internal static Signature SignMessagePart(this ECDsa key, IMessage? data) @@ -95,7 +98,7 @@ public static class RequestSigner var sig = new Signature { Key = ByteString.CopyFrom(key.PublicKey()), - Sign = ByteString.CopyFrom(key.SignData(data2Sign)), + Sign = key.SignData(data2Sign), }; return sig; diff --git a/src/FrostFS.SDK.Client/Tools/SearchReader.cs b/src/FrostFS.SDK.Client/Tools/SearchReader.cs index c2e7607..ba9b1d1 100644 --- a/src/FrostFS.SDK.Client/Tools/SearchReader.cs +++ b/src/FrostFS.SDK.Client/Tools/SearchReader.cs @@ -1,21 +1,21 @@ using System; -using System.Collections.Generic; -using System.Linq; using System.Threading; using System.Threading.Tasks; using FrostFS.Object; using FrostFS.Refs; +using Google.Protobuf.Collections; + using Grpc.Core; namespace FrostFS.SDK.Client; internal sealed class SearchReader(AsyncServerStreamingCall call) : IDisposable { - public AsyncServerStreamingCall Call { get; private set; } = call; + internal AsyncServerStreamingCall Call { get; private set; } = call; - public async Task?> Read(CancellationToken cancellationToken) + internal async Task?> Read(CancellationToken cancellationToken) { if (!await Call.ResponseStream.MoveNext(cancellationToken).ConfigureAwait(false)) return null; @@ -24,7 +24,7 @@ internal sealed class SearchReader(AsyncServerStreamingCall call Verifier.CheckResponse(response); - return response.Body?.IdList.ToList(); + return response.Body?.IdList; } public void Dispose() diff --git a/src/FrostFS.SDK.Client/Tools/Verifier.cs b/src/FrostFS.SDK.Client/Tools/Verifier.cs index bc8dd31..ce215ba 100644 --- a/src/FrostFS.SDK.Client/Tools/Verifier.cs +++ b/src/FrostFS.SDK.Client/Tools/Verifier.cs @@ -88,6 +88,7 @@ public static class Verifier return key.VerifyData(data2Verify, sig.Sign.ToByteArray()); } + internal static bool VerifyMatryoskaLevel(IMessage body, IMetaHeader meta, IVerificationHeader verification) { if (!verification.MetaSignature.VerifyMessagePart(meta)) diff --git a/src/FrostFS.SDK.Cryptography/FrostFS.SDK.Cryptography.csproj b/src/FrostFS.SDK.Cryptography/FrostFS.SDK.Cryptography.csproj index 148ae78..604fe7c 100644 --- a/src/FrostFS.SDK.Cryptography/FrostFS.SDK.Cryptography.csproj +++ b/src/FrostFS.SDK.Cryptography/FrostFS.SDK.Cryptography.csproj @@ -7,12 +7,12 @@ - true + true - + - true - + true + diff --git a/src/FrostFS.SDK.Cryptography/UUID.cs b/src/FrostFS.SDK.Cryptography/UUID.cs index c92a874..94fba3f 100644 --- a/src/FrostFS.SDK.Cryptography/UUID.cs +++ b/src/FrostFS.SDK.Cryptography/UUID.cs @@ -11,9 +11,7 @@ public static class UUIDExtension if (id == null) throw new ArgumentNullException(nameof(id)); - var bytes = id.ToByteArray(); - - var orderedBytes = GetGuidBytesDirectOrder(bytes); + var orderedBytes = GetGuidBytesDirectOrder(id.Span); return new Guid(orderedBytes); } @@ -32,7 +30,7 @@ public static class UUIDExtension return orderedBytes; } - private static byte[] GetGuidBytesDirectOrder(byte[] source) + private static byte[] GetGuidBytesDirectOrder(ReadOnlySpan source) { if (source.Length != 16) throw new ArgumentException("Wrong uuid binary format"); diff --git a/src/FrostFS.SDK.Protos/FrostFS.SDK.Protos.csproj b/src/FrostFS.SDK.Protos/FrostFS.SDK.Protos.csproj index 0bf48d5..8f69fe0 100644 --- a/src/FrostFS.SDK.Protos/FrostFS.SDK.Protos.csproj +++ b/src/FrostFS.SDK.Protos/FrostFS.SDK.Protos.csproj @@ -7,11 +7,11 @@ - true + true - + - true + true diff --git a/src/FrostFS.SDK.Tests/ContainerTestsBase.cs b/src/FrostFS.SDK.Tests/ContainerTestsBase.cs index 55f548d..3a0d396 100644 --- a/src/FrostFS.SDK.Tests/ContainerTestsBase.cs +++ b/src/FrostFS.SDK.Tests/ContainerTestsBase.cs @@ -8,12 +8,12 @@ public abstract class ContainerTestsBase { internal readonly string key = "KwHDAJ66o8FoLBjVbjP2sWBmgBMGjt7Vv4boA7xQrBoAYBE397Aq"; - protected IOptions Settings { get; set; } + protected IOptions Settings { get; set; } protected ContainerMocker Mocker { get; set; } protected ContainerTestsBase() { - Settings = Options.Create(new SingleOwnerClientSettings + Settings = Options.Create(new ClientSettings { Key = key, Host = "http://localhost:8080" diff --git a/src/FrostFS.SDK.Tests/FrostFS.SDK.Tests.csproj b/src/FrostFS.SDK.Tests/FrostFS.SDK.Tests.csproj index d5cb78f..30c0564 100644 --- a/src/FrostFS.SDK.Tests/FrostFS.SDK.Tests.csproj +++ b/src/FrostFS.SDK.Tests/FrostFS.SDK.Tests.csproj @@ -1,4 +1,4 @@ - + net8.0 @@ -10,15 +10,15 @@ - true + true - + - true + true - + - + @@ -38,7 +38,7 @@ - + PreserveNewest diff --git a/src/FrostFS.SDK.Tests/Mocks/AsyncStreamRangeReaderMock.cs b/src/FrostFS.SDK.Tests/Mocks/AsyncStreamRangeReaderMock.cs index d1366c7..bd9d858 100644 --- a/src/FrostFS.SDK.Tests/Mocks/AsyncStreamRangeReaderMock.cs +++ b/src/FrostFS.SDK.Tests/Mocks/AsyncStreamRangeReaderMock.cs @@ -1,9 +1,4 @@ -using System.Security.Cryptography; - using FrostFS.Object; -using FrostFS.SDK.Client; -using FrostFS.SDK.Client.Mappers.GRPC; -using FrostFS.SDK.Cryptography; using FrostFS.Session; using Google.Protobuf; diff --git a/src/FrostFS.SDK.Tests/Mocks/AsyncStreamReaderMock.cs b/src/FrostFS.SDK.Tests/Mocks/AsyncStreamReaderMock.cs index c09df83..9e9a18d 100644 --- a/src/FrostFS.SDK.Tests/Mocks/AsyncStreamReaderMock.cs +++ b/src/FrostFS.SDK.Tests/Mocks/AsyncStreamReaderMock.cs @@ -43,7 +43,7 @@ public class AsyncStreamReaderMock(string key, FrostFsObjectHeader objectHeader) Signature = new Refs.Signature { Key = ByteString.CopyFrom(Key.PublicKey()), - Sign = ByteString.CopyFrom(Key.SignData(header.ToByteArray())), + Sign = Key.SignData(header.ToByteArray()), } } }, diff --git a/src/FrostFS.SDK.Tests/Mocks/ContainerServiceMocks/ContainerServiceBase.cs b/src/FrostFS.SDK.Tests/Mocks/ContainerServiceMocks/ContainerServiceBase.cs index ca1c670..8d4230c 100644 --- a/src/FrostFS.SDK.Tests/Mocks/ContainerServiceMocks/ContainerServiceBase.cs +++ b/src/FrostFS.SDK.Tests/Mocks/ContainerServiceMocks/ContainerServiceBase.cs @@ -46,19 +46,19 @@ public abstract class ServiceBase(string key) { Key = ByteString.CopyFrom(Key.PublicKey()), Scheme = Refs.SignatureScheme.EcdsaRfc6979Sha256, - Sign = ByteString.CopyFrom(Key.SignData(response.MetaHeader.ToByteArray())) + Sign = Key.SignData(response.MetaHeader.ToByteArray()) }, BodySignature = new Refs.Signature { Key = ByteString.CopyFrom(Key.PublicKey()), Scheme = Refs.SignatureScheme.EcdsaRfc6979Sha256, - Sign = ByteString.CopyFrom(Key.SignData(response.GetBody().ToByteArray())) + Sign = Key.SignData(response.GetBody().ToByteArray()) }, OriginSignature = new Refs.Signature { Key = ByteString.CopyFrom(Key.PublicKey()), Scheme = Refs.SignatureScheme.EcdsaRfc6979Sha256, - Sign = ByteString.CopyFrom(Key.SignData([])) + Sign = Key.SignData([]) } }; diff --git a/src/FrostFS.SDK.Tests/Mocks/ObjectMock.cs b/src/FrostFS.SDK.Tests/Mocks/ObjectMock.cs index 4b2b5c1..4f07e41 100644 --- a/src/FrostFS.SDK.Tests/Mocks/ObjectMock.cs +++ b/src/FrostFS.SDK.Tests/Mocks/ObjectMock.cs @@ -5,7 +5,6 @@ using FrostFS.Object; using FrostFS.SDK.Client; using FrostFS.SDK.Client.Mappers.GRPC; using FrostFS.SDK.Cryptography; -using FrostFS.Session; using Google.Protobuf; @@ -41,7 +40,7 @@ public class ObjectMocker(string key) : ObjectServiceBase(key) public GetRangeHashRequest? GetRangeHashRequest { get; set; } - public Collection RangeHashResponses { get; } = []; + public Collection RangeHashResponses { get; } = []; public override Mock GetMock() { @@ -94,7 +93,7 @@ public class ObjectMocker(string key) : ObjectServiceBase(key) headResponse.Body.Header.Signature = new Refs.Signature { Key = ByteString.CopyFrom(Key.PublicKey()), - Sign = ByteString.CopyFrom(Key.SignData(headResponse.Body.Header.ToByteArray())), + Sign = Key.SignData(headResponse.Body.Header.ToByteArray()), }; headResponse.VerifyHeader = GetResponseVerificationHeader(headResponse); @@ -259,7 +258,7 @@ public class ObjectMocker(string key) : ObjectServiceBase(key) response.Body.HashList.Add(hash); } } - + response.VerifyHeader = GetResponseVerificationHeader(response); return new AsyncUnaryCall( @@ -281,7 +280,7 @@ public class ObjectMocker(string key) : ObjectServiceBase(key) { Body = new PatchResponse.Types.Body { - ObjectId = new Refs.ObjectID { Value = ByteString.CopyFrom(SHA256.HashData([1,2,3])) }, + ObjectId = new Refs.ObjectID { Value = ByteString.CopyFrom(SHA256.HashData([1, 2, 3])) }, }, MetaHeader = ResponseMetaHeader }; diff --git a/src/FrostFS.SDK.Tests/Mocks/SessionMock.cs b/src/FrostFS.SDK.Tests/Mocks/SessionMock.cs index 88ed60d..ebea821 100644 --- a/src/FrostFS.SDK.Tests/Mocks/SessionMock.cs +++ b/src/FrostFS.SDK.Tests/Mocks/SessionMock.cs @@ -31,7 +31,7 @@ public class SessionMocker(string key) : ServiceBase(key) if (SessionKey == null) { - SessionKey = new byte[32]; + SessionKey = new byte[33]; rand.NextBytes(SessionKey); } diff --git a/src/FrostFS.SDK.Tests/NetworkTest.cs b/src/FrostFS.SDK.Tests/NetworkTest.cs index 2123460..9e83bb0 100644 --- a/src/FrostFS.SDK.Tests/NetworkTest.cs +++ b/src/FrostFS.SDK.Tests/NetworkTest.cs @@ -32,16 +32,13 @@ public class NetworkTest : NetworkTestsBase new PrmNetworkSettings(new CallContext { CancellationToken = Mocker.CancellationTokenSource.Token, - Timeout = TimeSpan.FromSeconds(20), - OwnerId = OwnerId, - Key = ECDsaKey, - Version = Version + Timeout = TimeSpan.FromSeconds(20) }) : new PrmNetworkSettings(); var validTimeoutFrom = DateTime.UtcNow.AddSeconds(20); - var result = await GetClient().GetNetworkSettingsAsync(param); + var result = await GetClient(DefaultSettings).GetNetworkSettingsAsync(param); var validTimeoutTo = DateTime.UtcNow.AddSeconds(20); @@ -113,17 +110,14 @@ public class NetworkTest : NetworkTestsBase Mocker.NetmapSnapshotResponse = new NetmapSnapshotResponse { Body = body }; - PrmNetmapSnapshot param; + PrmNetmapSnapshot param = new(); if (useContext) { var ctx = new CallContext { CancellationToken = Mocker.CancellationTokenSource.Token, - Timeout = TimeSpan.FromSeconds(20), - OwnerId = OwnerId, - Key = ECDsaKey, - Version = Version + Timeout = TimeSpan.FromSeconds(20) }; param = new(ctx); @@ -137,7 +131,7 @@ public class NetworkTest : NetworkTestsBase var validTimeoutFrom = DateTime.UtcNow.AddSeconds(20); - var result = await GetClient().GetNetmapSnapshotAsync(param); + var result = await GetClient(DefaultSettings).GetNetmapSnapshotAsync(param); var validTimeoutTo = DateTime.UtcNow.AddSeconds(20); @@ -219,10 +213,7 @@ public class NetworkTest : NetworkTestsBase var ctx = new CallContext { CancellationToken = Mocker.CancellationTokenSource.Token, - Timeout = TimeSpan.FromSeconds(20), - OwnerId = OwnerId, - Key = ECDsaKey, - Version = Version + Timeout = TimeSpan.FromSeconds(20) }; param = new(ctx); @@ -236,7 +227,7 @@ public class NetworkTest : NetworkTestsBase var validTimeoutFrom = DateTime.UtcNow.AddSeconds(20); - var result = await GetClient().GetNodeInfoAsync(param); + var result = await GetClient(DefaultSettings).GetNodeInfoAsync(param); var validTimeoutTo = DateTime.UtcNow.AddSeconds(20); diff --git a/src/FrostFS.SDK.Tests/NetworkTestsBase.cs b/src/FrostFS.SDK.Tests/NetworkTestsBase.cs index 8090ecd..0cabab5 100644 --- a/src/FrostFS.SDK.Tests/NetworkTestsBase.cs +++ b/src/FrostFS.SDK.Tests/NetworkTestsBase.cs @@ -1,8 +1,6 @@ using System.Diagnostics.CodeAnalysis; -using System.Security.Cryptography; using FrostFS.SDK.Client.Interfaces; -using FrostFS.SDK.Cryptography; using Microsoft.Extensions.Options; @@ -13,32 +11,27 @@ public abstract class NetworkTestsBase { internal readonly string key = "KwHDAJ66o8FoLBjVbjP2sWBmgBMGjt7Vv4boA7xQrBoAYBE397Aq"; - protected IOptions Settings { get; set; } + // protected FrostFsVersion Version { get; set; } = new FrostFsVersion(2, 13); - protected FrostFsVersion Version { get; set; } = new FrostFsVersion(2, 13); - - protected ECDsa ECDsaKey { get; set; } - protected FrostFsOwner OwnerId { get; set; } protected NetworkMocker Mocker { get; set; } + protected ClientSettings DefaultSettings { get; } + protected NetworkTestsBase() { - Settings = Options.Create(new SingleOwnerClientSettings + DefaultSettings = new ClientSettings { Key = key, - Host = "http://localhost:8080" - }); - - ECDsaKey = key.LoadWif(); - OwnerId = FrostFsOwner.FromKey(ECDsaKey); + Host = "http://localhost:8080", + }; Mocker = new NetworkMocker(this.key); } - protected IFrostFSClient GetClient() + protected IFrostFSClient GetClient(ClientSettings settings) { return Client.FrostFSClient.GetTestInstance( - Settings, + Options.Create(settings), null, Mocker.GetMock().Object, new SessionMocker(this.key).GetMock().Object, diff --git a/src/FrostFS.SDK.Tests/ObjectTest.cs b/src/FrostFS.SDK.Tests/ObjectTest.cs index dabbbc8..5914605 100644 --- a/src/FrostFS.SDK.Tests/ObjectTest.cs +++ b/src/FrostFS.SDK.Tests/ObjectTest.cs @@ -6,12 +6,9 @@ using System.Text; using FrostFS.Refs; using FrostFS.SDK.Client; using FrostFS.SDK.Client.Mappers.GRPC; -using FrostFS.SDK.Cryptography; using Google.Protobuf; -using static FrostFS.Object.ECInfo.Types; - namespace FrostFS.SDK.Tests; [SuppressMessage("Reliability", "CA2007:Consider calling ConfigureAwait on the awaited task", Justification = "Default Value is correct for tests")] @@ -23,18 +20,9 @@ public class ObjectTest : ObjectTestsBase { var client = GetClient(); - var ecdsaKey = key.LoadWif(); + var objectId = client.CalculateObjectId(Mocker.ObjectHeader!); - var ctx = new CallContext - { - Key = ecdsaKey, - OwnerId = FrostFsOwner.FromKey(ecdsaKey), - Version = new FrostFsVersion(2, 13) - }; - - var objectId = client.CalculateObjectId(Mocker.ObjectHeader!, ctx); - - var result = await client.GetObjectAsync(new PrmObjectGet(ContainerId, objectId, ctx)); + var result = await client.GetObjectAsync(new PrmObjectGet(ContainerId, objectId)); Assert.NotNull(result); @@ -90,7 +78,7 @@ public class ObjectTest : ObjectTestsBase NetworkMocker.Parameters.Add("MaxObjectSize", [0x0, 0xa]); var blockSize = 2560; - byte[] bytes = File.ReadAllBytes(@".\..\..\..\TestData\cat.jpg"); + byte[] bytes = File.ReadAllBytes(@".\..\..\..\cat.jpg"); var fileLength = bytes.Length; var param = new PrmObjectPut @@ -295,7 +283,7 @@ public class ObjectTest : ObjectTestsBase Assert.Equal(SHA256.HashData(hash), SHA256.HashData(result.First().ToArray())); } - + [Fact] public async void PatchTest() { @@ -337,9 +325,9 @@ public class ObjectTest : ObjectTestsBase Assert.Equal(address.ContainerId, body.Address.ContainerId); Assert.Equal(address.ObjectId, body.Address.ObjectId); - + Assert.Equal(32, body.Patch.Chunk.Length); - + Assert.Equal(SHA256.HashData(patch), SHA256.HashData(body.Patch.Chunk.ToArray())); } } diff --git a/src/FrostFS.SDK.Tests/ObjectTestsBase.cs b/src/FrostFS.SDK.Tests/ObjectTestsBase.cs index ba68e6f..21289e4 100644 --- a/src/FrostFS.SDK.Tests/ObjectTestsBase.cs +++ b/src/FrostFS.SDK.Tests/ObjectTestsBase.cs @@ -10,7 +10,7 @@ public abstract class ObjectTestsBase { protected static readonly string key = "KwHDAJ66o8FoLBjVbjP2sWBmgBMGjt7Vv4boA7xQrBoAYBE397Aq"; - protected IOptions Settings { get; set; } + protected IOptions Settings { get; set; } protected FrostFsContainerId ContainerId { get; set; } protected NetworkMocker NetworkMocker { get; set; } = new NetworkMocker(key); @@ -22,7 +22,7 @@ public abstract class ObjectTestsBase { var ecdsaKey = key.LoadWif(); - Settings = Options.Create(new SingleOwnerClientSettings + Settings = Options.Create(new ClientSettings { Key = key, Host = "http://localhost:8080" diff --git a/src/FrostFS.SDK.Tests/PoolSmokeTests.cs b/src/FrostFS.SDK.Tests/PoolSmokeTests.cs index efc4802..0a6f2b3 100644 --- a/src/FrostFS.SDK.Tests/PoolSmokeTests.cs +++ b/src/FrostFS.SDK.Tests/PoolSmokeTests.cs @@ -6,8 +6,6 @@ using FrostFS.SDK.Cryptography; using Microsoft.Extensions.Options; -using static FrostFS.Session.SessionToken.Types.Body; - namespace FrostFS.SDK.SmokeTests; [SuppressMessage("Reliability", "CA2007:Consider calling ConfigureAwait on the awaited task", Justification = "Default Value is correct for tests")] @@ -78,6 +76,8 @@ public class PoolSmokeTests : SmokeTestsBase [Fact] public async void NodeInfoStatisticsTwoNodesTest() { + var callbackText = string.Empty; + var options = new InitParameters { Key = keyString.LoadWif(), @@ -87,16 +87,14 @@ public class PoolSmokeTests : SmokeTestsBase ], ClientBuilder = null, GracefulCloseOnSwitchTimeout = 30_000_000, - Logger = null + Logger = null, + Callback = (cs) => callbackText = $"{cs.MethodName} took {cs.ElapsedMicroSeconds} microseconds" }; using var pool = new Pool(options); - var callbackText = string.Empty; - var ctx = new CallContext { - Callback = (cs) => callbackText = $"{cs.MethodName} took {cs.ElapsedMicroSeconds} microseconds" }; var error = await pool.Dial(ctx).ConfigureAwait(true); @@ -118,14 +116,12 @@ public class PoolSmokeTests : SmokeTestsBase { var options = GetDefaultParams(); + var callbackText = string.Empty; + options.Callback = (cs) => callbackText = $"{cs.MethodName} took {cs.ElapsedMicroSeconds} microseconds"; + using var pool = new Pool(options); - var callbackText = string.Empty; - - var ctx = new CallContext - { - Callback = (cs) => callbackText = $"{cs.MethodName} took {cs.ElapsedMicroSeconds} microseconds" - }; + var ctx = new CallContext(); var error = await pool.Dial(ctx).ConfigureAwait(true); @@ -154,18 +150,11 @@ public class PoolSmokeTests : SmokeTestsBase var token = await pool.CreateSessionAsync(prm).ConfigureAwait(true); - var session = new Session.SessionToken().Deserialize(token.Token); - var ownerHash = Base58.Decode(OwnerId!.Value); - Assert.NotNull(session); - Assert.Null(session.Body.Container); - Assert.Null(session.Body.Object); - Assert.Equal(16, session.Body.Id.Length); - Assert.Equal(100ul, session.Body.Lifetime.Exp); - Assert.Equal(ownerHash, session.Body.OwnerId.Value); - Assert.Equal(33, session.Body.SessionKey.Length); - Assert.Equal(ContextOneofCase.None, session.Body.ContextCase); + Assert.NotNull(token); + Assert.NotEqual(Guid.Empty, token.Id); + Assert.Equal(33, token.SessionKey.Length); } [Fact] @@ -314,8 +303,16 @@ public class PoolSmokeTests : SmokeTestsBase [InlineData(6 * 1024 * 1024 + 100)] public async void SimpleScenarioTest(int objectSize) { + bool callbackInvoked = false; + var options = GetDefaultParams(); + options.Callback = new((CallStatistics cs) => + { + callbackInvoked = true; + Assert.True(cs.ElapsedMicroSeconds > 0); + }); + using var pool = new Pool(options); var error = await pool.Dial(new CallContext()).ConfigureAwait(true); @@ -324,16 +321,7 @@ public class PoolSmokeTests : SmokeTestsBase await Cleanup(pool); - bool callbackInvoked = false; - var ctx = new CallContext - { - // Timeout = TimeSpan.FromSeconds(20), - Callback = new((CallStatistics cs) => - { - callbackInvoked = true; - Assert.True(cs.ElapsedMicroSeconds > 0); - }) - }; + var ctx = new CallContext { }; var createContainerParam = new PrmContainerCreate( new FrostFsContainerInfo(new FrostFsPlacementPolicy(true, new FrostFsReplica(1)), [new("testKey", "testValue")]), ctx); @@ -346,12 +334,7 @@ public class PoolSmokeTests : SmokeTestsBase var bytes = GetRandomBytes(objectSize); - var ctx1 = new CallContext - { - Callback = new((CallStatistics cs) => Assert.True(cs.ElapsedMicroSeconds > 0)) - }; - - var param = new PrmObjectPut(ctx1) + var param = new PrmObjectPut() { Header = new FrostFsObjectHeader( containerId: createdContainer, @@ -410,6 +393,7 @@ public class PoolSmokeTests : SmokeTestsBase var options = GetDefaultParams(); using var pool = new Pool(options); + options.Callback = new((CallStatistics cs) => Assert.True(cs.ElapsedMicroSeconds > 0)); var error = await pool.Dial(new CallContext()).ConfigureAwait(true); @@ -421,8 +405,7 @@ public class PoolSmokeTests : SmokeTestsBase var ctx = new CallContext { - Timeout = TimeSpan.FromSeconds(20), - Callback = new((CallStatistics cs) => Assert.True(cs.ElapsedMicroSeconds > 0)) + Timeout = TimeSpan.FromSeconds(20) }; var createContainerParam = new PrmContainerCreate( @@ -435,7 +418,7 @@ public class PoolSmokeTests : SmokeTestsBase var bytes = GetRandomBytes(objectSize); - var param = new PrmObjectPut(new CallContext { Callback = new((CallStatistics cs) => Assert.True(cs.ElapsedMicroSeconds > 0)) }) + var param = new PrmObjectPut(new CallContext()) { Header = new FrostFsObjectHeader( containerId: container, @@ -517,7 +500,7 @@ public class PoolSmokeTests : SmokeTestsBase Timeout = TimeSpan.FromSeconds(10), }; - ctx.Interceptors.Add(new CallbackInterceptor()); + // ctx.Interceptors.Add(new CallbackInterceptor()); var container = await pool.GetContainerAsync(new PrmContainerGet(containerId, ctx)); @@ -585,9 +568,9 @@ public class PoolSmokeTests : SmokeTestsBase return bytes; } - private static IOptions GetSingleOwnerOptions(string key, string url) + private static IOptions GetSingleOwnerOptions(string key, string url) { - return Options.Create(new SingleOwnerClientSettings + return Options.Create(new ClientSettings { Key = key, Host = url diff --git a/src/FrostFS.SDK.Tests/SessionTests.cs b/src/FrostFS.SDK.Tests/SessionTests.cs index 4cf03ba..9e79b29 100644 --- a/src/FrostFS.SDK.Tests/SessionTests.cs +++ b/src/FrostFS.SDK.Tests/SessionTests.cs @@ -2,6 +2,7 @@ using System.Diagnostics.CodeAnalysis; using FrostFS.SDK.Client; using FrostFS.SDK.Client.Mappers.GRPC; +using FrostFS.SDK.Cryptography; namespace FrostFS.SDK.Tests; @@ -21,10 +22,7 @@ public class SessionTest : SessionTestsBase var ctx = new CallContext { CancellationToken = Mocker.CancellationTokenSource.Token, - Timeout = TimeSpan.FromSeconds(20), - OwnerId = OwnerId, - Key = ECDsaKey, - Version = Mocker.Version + Timeout = TimeSpan.FromSeconds(20) }; param = new PrmSessionCreate(exp, ctx); @@ -43,18 +41,16 @@ public class SessionTest : SessionTestsBase var validTimeoutTo = DateTime.UtcNow.AddSeconds(20); Assert.NotNull(result); - Assert.NotNull(result.Token); + Assert.NotEqual(Guid.Empty, result.Id); - var session = new Session.SessionToken().Deserialize(result.Token); + Assert.Equal(Mocker.SessionId, result.Id.ToBytes()); + Assert.Equal(Mocker.SessionKey, result.SessionKey.ToArray()); - Assert.Equal(Mocker.SessionId, session.Body.Id); - Assert.Equal(Mocker.SessionKey, session.Body.SessionKey); - - Assert.Equal(OwnerId.ToMessage(), session.Body.OwnerId); - Assert.Equal(exp, session.Body.Lifetime.Exp); - Assert.Equal(exp, session.Body.Lifetime.Iat); - Assert.Equal(exp, session.Body.Lifetime.Nbf); - Assert.Null(session.Body.Container); + //Assert.Equal(OwnerId.ToMessage(), result.Token.Body.OwnerId); + //Assert.Equal(exp, result.Token.Body.Lifetime.Exp); + //Assert.Equal(exp, result.Token.Body.Lifetime.Iat); + //Assert.Equal(exp, result.Token.Body.Lifetime.Nbf); + //Assert.Null(result.Token.Body.Container); Assert.NotNull(Mocker.CreateSessionRequest); diff --git a/src/FrostFS.SDK.Tests/SessionTestsBase.cs b/src/FrostFS.SDK.Tests/SessionTestsBase.cs index 93bc140..bd958fc 100644 --- a/src/FrostFS.SDK.Tests/SessionTestsBase.cs +++ b/src/FrostFS.SDK.Tests/SessionTestsBase.cs @@ -11,7 +11,7 @@ public abstract class SessionTestsBase { internal readonly string key = "KwHDAJ66o8FoLBjVbjP2sWBmgBMGjt7Vv4boA7xQrBoAYBE397Aq"; - protected IOptions Settings { get; set; } + protected IOptions Settings { get; set; } protected ECDsa ECDsaKey { get; set; } protected FrostFsOwner OwnerId { get; set; } @@ -19,7 +19,7 @@ public abstract class SessionTestsBase protected SessionTestsBase() { - Settings = Options.Create(new SingleOwnerClientSettings + Settings = Options.Create(new ClientSettings { Key = key, Host = "http://localhost:8080" diff --git a/src/FrostFS.SDK.Tests/SmokeClientTests.cs b/src/FrostFS.SDK.Tests/SmokeClientTests.cs index ba822c7..aaf1869 100644 --- a/src/FrostFS.SDK.Tests/SmokeClientTests.cs +++ b/src/FrostFS.SDK.Tests/SmokeClientTests.cs @@ -1,4 +1,4 @@ -using System.Diagnostics.CodeAnalysis; +using System.Diagnostics.CodeAnalysis; using System.Security.Cryptography; using FrostFS.SDK.Client; @@ -7,8 +7,6 @@ using FrostFS.SDK.Cryptography; using Microsoft.Extensions.Options; -using static FrostFS.Session.SessionToken.Types.Body; - namespace FrostFS.SDK.SmokeTests; [SuppressMessage("Reliability", "CA2007:Consider calling ConfigureAwait on the awaited task", Justification = "Default Value is correct for tests")] @@ -17,29 +15,23 @@ public class SmokeClientTests : SmokeTestsBase { private static readonly PrmWait lightWait = new(100, 1); - [Theory] - [InlineData(false)] - [InlineData(true)] - public async void AccountTest(bool isSingleOnwerClient) + [Fact] + public async void AccountTest() { - using var client = isSingleOnwerClient - ? FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)) - : FrostFSClient.GetInstance(GetOptions(this.url)); + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); - PrmBalance? prm = isSingleOnwerClient ? default : new(Ctx); + var result = await client.GetBalanceAsync(); - var result = await client.GetBalanceAsync(prm); + Assert.NotNull(result); + Assert.True(result.Value == 0); } - [Theory] - [InlineData(false)] - [InlineData(true)] - public async void NetworkMapTest(bool isSingleOnwerClient) + [Fact] + public async void NetworkMapTest() { - using var client = isSingleOnwerClient ? FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)) : FrostFSClient.GetInstance(GetOptions(this.url)); + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); - PrmNetmapSnapshot? prm = isSingleOnwerClient ? default : new(Ctx); - var result = await client.GetNetmapSnapshotAsync(prm); + var result = await client.GetNetmapSnapshotAsync(); Assert.True(result.Epoch > 0); Assert.Single(result.NodeInfoCollection); @@ -54,18 +46,12 @@ public class SmokeClientTests : SmokeTestsBase } - [Theory] - [InlineData(false)] - [InlineData(true)] - public async void NodeInfoTest(bool isSingleOnwerClient) + [Fact] + public async void NodeInfoTest() { - using var client = isSingleOnwerClient - ? FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)) - : FrostFSClient.GetInstance(GetOptions(this.url)); + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); - PrmNodeInfo? prm = isSingleOnwerClient ? default : new(Ctx); - - var result = await client.GetNodeInfoAsync(prm); + var result = await client.GetNodeInfoAsync(); Assert.Equal(2, result.Version.Major); Assert.Equal(13, result.Version.Minor); @@ -78,45 +64,38 @@ public class SmokeClientTests : SmokeTestsBase [Fact] public async void NodeInfoStatisticsTest() { - var ctx = new CallContext - { - Callback = (cs) => Console.WriteLine($"{cs.MethodName} took {cs.ElapsedMicroSeconds} microseconds") - }; + var options = GetClientOptions(this.keyString, this.url); - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); + var callbackContent = string.Empty; + options.Value.Callback = (cs) => callbackContent = $"{cs.MethodName} took {cs.ElapsedMicroSeconds} microseconds"; + + using var client = FrostFSClient.GetSingleOwnerInstance(options); var result = await client.GetNodeInfoAsync(); + + Assert.NotEmpty(callbackContent); } - [Theory] - [InlineData(false)] - [InlineData(true)] - public async void GetSessionTest(bool isSingleOnwerClient) + [Fact] + public async void GetSessionTest() { - using var client = isSingleOnwerClient ? FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)) : FrostFSClient.GetInstance(GetOptions(this.url)); + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); - PrmSessionCreate? prm = isSingleOnwerClient ? new PrmSessionCreate(100) : new PrmSessionCreate(100, Ctx); + PrmSessionCreate? prm = new(100); var token = await client.CreateSessionAsync(prm); - var session = new Session.SessionToken().Deserialize(token.Token); - var ownerHash = Base58.Decode(OwnerId!.Value); - Assert.NotNull(session); - Assert.Null(session.Body.Container); - Assert.Null(session.Body.Object); - Assert.Equal(16, session.Body.Id.Length); - Assert.Equal(100ul, session.Body.Lifetime.Exp); - Assert.Equal(ownerHash, session.Body.OwnerId.Value); - Assert.Equal(33, session.Body.SessionKey.Length); - Assert.Equal(ContextOneofCase.None, session.Body.ContextCase); + Assert.NotNull(token); + Assert.NotEqual(Guid.Empty, token.Id); + Assert.Equal(33, token.SessionKey.Length); } [Fact] public async void CreateObjectWithSessionToken() { - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); await Cleanup(client); @@ -164,7 +143,7 @@ public class SmokeClientTests : SmokeTestsBase [Fact] public async void FilterTest() { - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); await Cleanup(client); @@ -248,89 +227,87 @@ public class SmokeClientTests : SmokeTestsBase [InlineData(6 * 1024 * 1024 + 100)] public async void SimpleScenarioTest(int objectSize) { - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); + bool callbackInvoked = false; - await Cleanup(client); + var options = GetClientOptions(this.keyString, this.url); - bool callbackInvoked = false; - var ctx = new CallContext - { - // Timeout = TimeSpan.FromSeconds(20), - Callback = new((CallStatistics cs) => - { - callbackInvoked = true; - Assert.True(cs.ElapsedMicroSeconds > 0); - }) - }; + options.Value.Callback = new((CallStatistics cs) => + { + callbackInvoked = true; + Assert.True(cs.ElapsedMicroSeconds > 0); + }); - var createContainerParam = new PrmContainerCreate( - new FrostFsContainerInfo(new FrostFsPlacementPolicy(true, new FrostFsReplica(1)), [new("testKey", "testValue")]), ctx); + using var client = FrostFSClient.GetSingleOwnerInstance(options); - var createdContainer = await client.CreateContainerAsync(createContainerParam); + await Cleanup(client); - var container = await client.GetContainerAsync(new PrmContainerGet(createdContainer, ctx)); - Assert.NotNull(container); - Assert.True(callbackInvoked); + var ctx = new CallContext { Timeout = TimeSpan.FromSeconds(20) }; - var bytes = GetRandomBytes(objectSize); + var createContainerParam = new PrmContainerCreate( + new FrostFsContainerInfo(new FrostFsPlacementPolicy(true, new FrostFsReplica(1)), [new("testKey", "testValue")]), ctx); - var param = new PrmObjectPut(new CallContext - { - Callback = new((CallStatistics cs) => Assert.True(cs.ElapsedMicroSeconds > 0)) - }) - { - Header = new FrostFsObjectHeader( - containerId: createdContainer, - type: FrostFsObjectType.Regular, - [new FrostFsAttributePair("fileName", "test")]), - Payload = new MemoryStream(bytes), - ClientCut = false - }; + var createdContainer = await client.CreateContainerAsync(createContainerParam); - var objectId = await client.PutObjectAsync(param); + var container = await client.GetContainerAsync(new PrmContainerGet(createdContainer)); + Assert.NotNull(container); + Assert.True(callbackInvoked); - var filter = new FilterByAttributePair(FrostFsMatchType.Equals, "fileName", "test"); + var bytes = GetRandomBytes(objectSize); - bool hasObject = false; - await foreach (var objId in client.SearchObjectsAsync(new PrmObjectSearch(createdContainer) { Filters = [filter] })) - { - hasObject = true; + var param = new PrmObjectPut + { + Header = new FrostFsObjectHeader( + containerId: createdContainer, + type: FrostFsObjectType.Regular, + [new FrostFsAttributePair("fileName", "test")]), + Payload = new MemoryStream(bytes), + ClientCut = false + }; - var objHeader = await client.GetObjectHeadAsync(new PrmObjectHeadGet(createdContainer, objectId)); - Assert.Equal((ulong)bytes.Length, objHeader.PayloadLength); - Assert.NotNull(objHeader.Attributes); - Assert.Single(objHeader.Attributes); - Assert.Equal("fileName", objHeader.Attributes.First().Key); - Assert.Equal("test", objHeader.Attributes.First().Value); - } + var objectId = await client.PutObjectAsync(param); - Assert.True(hasObject); + var filter = new FilterByAttributePair(FrostFsMatchType.Equals, "fileName", "test"); - var @object = await client.GetObjectAsync(new PrmObjectGet(createdContainer, objectId)); + bool hasObject = false; + await foreach (var objId in client.SearchObjectsAsync(new PrmObjectSearch(createdContainer) { Filters = [filter] })) + { + hasObject = true; - var downloadedBytes = new byte[@object.Header.PayloadLength]; - MemoryStream ms = new(downloadedBytes); + var objHeader = await client.GetObjectHeadAsync(new PrmObjectHeadGet(createdContainer, objectId)); + Assert.Equal((ulong)bytes.Length, objHeader.PayloadLength); + Assert.NotNull(objHeader.Attributes); + Assert.Single(objHeader.Attributes); + Assert.Equal("fileName", objHeader.Attributes.First().Key); + Assert.Equal("test", objHeader.Attributes.First().Value); + } - ReadOnlyMemory? chunk = null; - while ((chunk = await @object.ObjectReader!.ReadChunk()) != null) - { - ms.Write(chunk.Value.Span); - } + Assert.True(hasObject); - Assert.Equal(SHA256.HashData(bytes), SHA256.HashData(downloadedBytes)); + var @object = await client.GetObjectAsync(new PrmObjectGet(createdContainer, objectId)); - await Cleanup(client); + var downloadedBytes = new byte[@object.Header.PayloadLength]; + MemoryStream ms = new(downloadedBytes); - await foreach (var _ in client.ListContainersAsync()) - { - Assert.Fail("Containers exist"); - } + ReadOnlyMemory? chunk = null; + while ((chunk = await @object.ObjectReader!.ReadChunk()) != null) + { + ms.Write(chunk.Value.Span); + } + + Assert.Equal(SHA256.HashData(bytes), SHA256.HashData(downloadedBytes)); + + await Cleanup(client); + + await foreach (var _ in client.ListContainersAsync()) + { + Assert.Fail("Containers exist"); + } } [Fact] public async void PatchTest() { - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); await Cleanup(client); @@ -388,7 +365,7 @@ public class SmokeClientTests : SmokeTestsBase ms.Write(chunk.Value.Span); } - for(int i = 0; i < (int)range.Offset; i++) + for (int i = 0; i < (int)range.Offset; i++) Assert.Equal(downloadedBytes[i], bytes[i]); var rangeEnd = range.Offset + range.Length; @@ -410,7 +387,7 @@ public class SmokeClientTests : SmokeTestsBase [Fact] public async void RangeTest() { - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); await Cleanup(client); @@ -465,7 +442,7 @@ public class SmokeClientTests : SmokeTestsBase [Fact] public async void RangeHashTest() { - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); await Cleanup(client); @@ -494,15 +471,15 @@ public class SmokeClientTests : SmokeTestsBase var objectId = await client.PutObjectAsync(param); - var rangeParam = new PrmRangeHashGet(createdContainer, objectId, [ new FrostFsRange(100, 64)], bytes); + var rangeParam = new PrmRangeHashGet(createdContainer, objectId, [new FrostFsRange(100, 64)], bytes); var hashes = await client.GetRangeHashAsync(rangeParam); foreach (var hash in hashes) { - var x = hash.Slice(0, 32).ToArray(); + var x = hash[..32].ToArray(); } - + await Cleanup(client); await foreach (var _ in client.ListContainersAsync()) @@ -517,8 +494,8 @@ public class SmokeClientTests : SmokeTestsBase [InlineData(6 * 1024 * 1024 + 100)] public async void SimpleScenarioWithSessionTest(int objectSize) { - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); - + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); + //Callback = new((CallStatistics cs) => Assert.True(cs.ElapsedMicroSeconds > 0)) var token = await client.CreateSessionAsync(new PrmSessionCreate(int.MaxValue)); await Cleanup(client); @@ -526,7 +503,7 @@ public class SmokeClientTests : SmokeTestsBase var ctx = new CallContext { Timeout = TimeSpan.FromSeconds(20), - Callback = new((CallStatistics cs) => Assert.True(cs.ElapsedMicroSeconds > 0)) + }; var createContainerParam = new PrmContainerCreate( @@ -539,10 +516,7 @@ public class SmokeClientTests : SmokeTestsBase var bytes = GetRandomBytes(objectSize); - var param = new PrmObjectPut(new CallContext - { - Callback = new((CallStatistics cs) => Assert.True(cs.ElapsedMicroSeconds > 0)) - }) + var param = new PrmObjectPut { Header = new FrostFsObjectHeader( containerId: container, @@ -602,7 +576,11 @@ public class SmokeClientTests : SmokeTestsBase [InlineData(200)] public async void ClientCutScenarioTest(int objectSize) { - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); + + var options = GetClientOptions(this.keyString, this.url); + options.Value.Interceptors.Add(new CallbackInterceptor()); + + using var client = FrostFSClient.GetSingleOwnerInstance(GetClientOptions(this.keyString, this.url)); await Cleanup(client); @@ -613,12 +591,7 @@ public class SmokeClientTests : SmokeTestsBase var containerId = await client.CreateContainerAsync(createContainerParam); - var ctx = new CallContext - { - Timeout = TimeSpan.FromSeconds(10) - }; - - ctx.Interceptors.Add(new CallbackInterceptor()); + var ctx = new CallContext { Timeout = TimeSpan.FromSeconds(10) }; var container = await client.GetContainerAsync(new PrmContainerGet(containerId, ctx)); @@ -692,23 +665,21 @@ public class SmokeClientTests : SmokeTestsBase [Fact] public async void NodeInfoCallbackAndInterceptorTest() { - using var client = FrostFSClient.GetSingleOwnerInstance(GetSingleOwnerOptions(this.keyString, this.url)); - bool callbackInvoked = false; bool intercepterInvoked = false; - var ctx = new CallContext + var options = GetClientOptions(this.keyString, this.url); + options.Value.Callback = (CallStatistics cs) => { - Callback = new((CallStatistics cs) => - { - callbackInvoked = true; - Assert.True(cs.ElapsedMicroSeconds > 0); - }) + callbackInvoked = true; + Assert.True(cs.ElapsedMicroSeconds > 0); }; - ctx.Interceptors.Add(new CallbackInterceptor(s => intercepterInvoked = true)); + options.Value.Interceptors.Add(new CallbackInterceptor(s => intercepterInvoked = true)); - var result = await client.GetNodeInfoAsync(new PrmNodeInfo(ctx)); + using var client = FrostFSClient.GetSingleOwnerInstance(options); + + var result = await client.GetNodeInfoAsync(new PrmNodeInfo()); Assert.True(callbackInvoked); Assert.True(intercepterInvoked); @@ -729,22 +700,15 @@ public class SmokeClientTests : SmokeTestsBase return bytes; } - private static IOptions GetSingleOwnerOptions(string key, string url) + private static IOptions GetClientOptions(string key, string url) { - return Options.Create(new SingleOwnerClientSettings + return Options.Create(new ClientSettings { Key = key, Host = url }); } - private static IOptions GetOptions(string url) - { - return Options.Create(new ClientSettings - { - Host = url - }); - } static async Task Cleanup(IFrostFSClient client) { diff --git a/src/FrostFS.SDK.Tests/SmokeTestsBase.cs b/src/FrostFS.SDK.Tests/SmokeTestsBase.cs index 5d67029..b96997c 100644 --- a/src/FrostFS.SDK.Tests/SmokeTestsBase.cs +++ b/src/FrostFS.SDK.Tests/SmokeTestsBase.cs @@ -7,8 +7,6 @@ namespace FrostFS.SDK.SmokeTests; public abstract class SmokeTestsBase { - // internal readonly string keyString = "KzPXA6669m2pf18XmUdoR8MnP1pi1PMmefiFujStVFnv7WR5SRmK"; - internal readonly string keyString = "KzPXA6669m2pf18XmUdoR8MnP1pi1PMmefiFujStVFnv7WR5SRmK"; internal readonly string url = "http://172.23.32.4:8080"; @@ -26,7 +24,5 @@ public abstract class SmokeTestsBase Key = keyString.LoadWif(); OwnerId = FrostFsOwner.FromKey(Key); Version = new FrostFsVersion(2, 13); - - Ctx = new CallContext { Key = Key, OwnerId = OwnerId, Version = Version }; } } diff --git a/src/FrostFS.SDK.Tests/TestData/cat.jpg b/src/FrostFS.SDK.Tests/cat.jpg similarity index 100% rename from src/FrostFS.SDK.Tests/TestData/cat.jpg rename to src/FrostFS.SDK.Tests/cat.jpg