mirror of
https://github.com/Shchoholiev/shopping-assistant-api.git
synced 2025-04-04 16:49:36 +00:00
Merge branch 'develop' into bug/SA-143-Fix-data-retrieval-to-return-not-deleted-records
This commit is contained in:
commit
4bd13648b6
@ -3,6 +3,8 @@ using Microsoft.AspNetCore.Mvc;
|
||||
using Newtonsoft.Json;
|
||||
using ShoppingAssistantApi.Application.IServices;
|
||||
using ShoppingAssistantApi.Application.Models.CreateDtos;
|
||||
using ShoppingAssistantApi.Application.Models.Dtos;
|
||||
using ShoppingAssistantApi.Domain.Enums;
|
||||
|
||||
namespace ShoppingAssistantApi.Api.Controllers;
|
||||
|
||||
@ -11,14 +13,24 @@ public class ProductsSearchController : BaseController
|
||||
{
|
||||
private readonly IProductService _productService;
|
||||
|
||||
public ProductsSearchController(IProductService productService)
|
||||
private readonly IWishlistsService _wishlistsService;
|
||||
|
||||
public ProductsSearchController(IProductService productService, IWishlistsService wishlistsService)
|
||||
{
|
||||
_productService = productService;
|
||||
_wishlistsService = wishlistsService;
|
||||
}
|
||||
|
||||
[HttpPost("search/{wishlistId}")]
|
||||
public async Task StreamDataToClient(string wishlistId, [FromBody]MessageCreateDto message, CancellationToken cancellationToken)
|
||||
{
|
||||
var dto = new MessageDto()
|
||||
{
|
||||
Text = message.Text,
|
||||
Role = MessageRoles.User.ToString(),
|
||||
};
|
||||
await _wishlistsService.AddMessageToPersonalWishlistAsync(wishlistId, dto, cancellationToken);
|
||||
|
||||
Response.Headers.Add("Content-Type", "text/event-stream");
|
||||
Response.Headers.Add("Cache-Control", "no-cache");
|
||||
Response.Headers.Add("Connection", "keep-alive");
|
||||
|
@ -15,7 +15,7 @@ public class WishlistsMutation
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.GenerateNameForPersonalWishlistAsync(wishlistId, cancellationToken);
|
||||
|
||||
public Task<MessageDto> AddMessageToPersonalWishlistAsync(string wishlistId, MessageCreateDto dto, CancellationToken cancellationToken,
|
||||
public Task<MessageDto> AddMessageToPersonalWishlistAsync(string wishlistId, MessageDto dto, CancellationToken cancellationToken,
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.AddMessageToPersonalWishlistAsync(wishlistId, dto, cancellationToken);
|
||||
|
||||
|
@ -10,7 +10,7 @@ public interface IWishlistsService
|
||||
|
||||
Task<WishlistDto> GenerateNameForPersonalWishlistAsync(string wishlistId, CancellationToken cancellationToken);
|
||||
|
||||
Task<MessageDto> AddMessageToPersonalWishlistAsync(string wishlistId, MessageCreateDto dto, CancellationToken cancellationToken);
|
||||
Task<MessageDto> AddMessageToPersonalWishlistAsync(string wishlistId, MessageDto dto, CancellationToken cancellationToken);
|
||||
|
||||
Task<PagedList<WishlistDto>> GetPersonalWishlistsPageAsync(int pageNumber, int pageSize, CancellationToken cancellationToken);
|
||||
|
||||
|
@ -9,6 +9,8 @@ public class ProductCreateDto
|
||||
public required string Description { get; set; }
|
||||
|
||||
public required double Rating { get; set; }
|
||||
|
||||
public required double Price { get; set; }
|
||||
|
||||
public required string[] ImagesUrls { get; set; }
|
||||
|
||||
|
@ -2,11 +2,11 @@ namespace ShoppingAssistantApi.Application.Models.Dtos;
|
||||
|
||||
public class MessageDto
|
||||
{
|
||||
public required string Id { get; set; }
|
||||
public string Id { get; set; }
|
||||
|
||||
public required string Text { get; set; }
|
||||
public string Text { get; set; }
|
||||
|
||||
public required string Role { get; set; }
|
||||
public string Role { get; set; }
|
||||
|
||||
public required string CreatedById { get; set; }
|
||||
public string CreatedById { get; set; }
|
||||
}
|
||||
|
@ -12,6 +12,8 @@ public class ProductDto
|
||||
|
||||
public required double Rating { get; set; }
|
||||
|
||||
public required double Price { get; set; }
|
||||
|
||||
public required string[] ImagesUrls { get; set; }
|
||||
|
||||
public required bool WasOpened { get; set; }
|
||||
|
@ -2,7 +2,7 @@ namespace ShoppingAssistantApi.Application.Models.OpenAi;
|
||||
|
||||
public class ChatCompletionRequest
|
||||
{
|
||||
public string Model { get; set; } = "gpt-4";
|
||||
public string Model { get; set; } = "gpt-4-1106-preview";
|
||||
|
||||
public List<OpenAiMessage> Messages { get; set; }
|
||||
|
||||
|
@ -17,6 +17,8 @@ public class Product : EntityBase
|
||||
public string[] ImagesUrls { get; set; }
|
||||
|
||||
public bool WasOpened { get; set; }
|
||||
|
||||
public double Price { get; set; }
|
||||
|
||||
public ObjectId WishlistId { get; set; }
|
||||
}
|
||||
|
@ -19,6 +19,10 @@ public class ProductService : IProductService
|
||||
private readonly IOpenAiService _openAiService;
|
||||
|
||||
private readonly IMessagesRepository _messagesRepository;
|
||||
|
||||
private bool mqchecker = false;
|
||||
|
||||
private SearchEventType currentDataType = SearchEventType.Wishlist;
|
||||
|
||||
public ProductService(IOpenAiService openAiService, IWishlistsService wishlistsService, IMessagesRepository messagesRepository)
|
||||
{
|
||||
@ -60,12 +64,24 @@ public class ProductService : IProductService
|
||||
|
||||
foreach (var item in previousMessages.Items)
|
||||
{
|
||||
messagesForOpenAI
|
||||
.Add(new OpenAiMessage()
|
||||
{
|
||||
Role = item.Role.ToLower(),
|
||||
Content = item.Text
|
||||
});
|
||||
if (item.Role == "Application")
|
||||
{
|
||||
messagesForOpenAI
|
||||
.Add(new OpenAiMessage()
|
||||
{
|
||||
Role = OpenAiRole.Assistant.RequestConvert(),
|
||||
Content = item.Text
|
||||
});
|
||||
}
|
||||
else
|
||||
{
|
||||
messagesForOpenAI
|
||||
.Add(new OpenAiMessage()
|
||||
{
|
||||
Role = item.Role.ToLower(),
|
||||
Content = item.Text
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
messagesForOpenAI.Add(new OpenAiMessage()
|
||||
@ -79,20 +95,27 @@ public class ProductService : IProductService
|
||||
var suggestionBuffer = new Suggestion();
|
||||
var messageBuffer = new MessagePart();
|
||||
var productBuffer = new ProductName();
|
||||
var currentDataType = SearchEventType.Wishlist;
|
||||
var dataTypeHolder = string.Empty;
|
||||
var counter = 0;
|
||||
|
||||
await foreach (var data in _openAiService.GetChatCompletionStream(chatRequest, cancellationToken))
|
||||
{
|
||||
if (data.Contains("["))
|
||||
counter++;
|
||||
if (mqchecker && currentDataType == SearchEventType.Message && messageBuffer != null)
|
||||
{
|
||||
if (dataTypeHolder=="[Message]" && messageBuffer.Text!=null)
|
||||
if (data == "[")
|
||||
{
|
||||
_wishlistsService.AddMessageToPersonalWishlistAsync(wishlistId, new MessageCreateDto()
|
||||
_wishlistsService.AddMessageToPersonalWishlistAsync(wishlistId, new MessageDto()
|
||||
{
|
||||
Text = messageBuffer.Text,
|
||||
Role = MessageRoles.Application.ToString(),
|
||||
}, cancellationToken);
|
||||
mqchecker = false;
|
||||
}
|
||||
}
|
||||
|
||||
if (data.Contains("["))
|
||||
{
|
||||
dataTypeHolder = string.Empty;
|
||||
dataTypeHolder += data;
|
||||
}
|
||||
@ -101,13 +124,17 @@ public class ProductService : IProductService
|
||||
{
|
||||
dataTypeHolder += data;
|
||||
currentDataType = DetermineDataType(dataTypeHolder);
|
||||
if (currentDataType == SearchEventType.Message)
|
||||
{
|
||||
mqchecker = true;
|
||||
}
|
||||
}
|
||||
|
||||
else if (dataTypeHolder=="[" && !data.Contains("["))
|
||||
{
|
||||
dataTypeHolder += data;
|
||||
}
|
||||
|
||||
|
||||
else
|
||||
{
|
||||
switch (currentDataType)
|
||||
@ -118,6 +145,7 @@ public class ProductService : IProductService
|
||||
Event = SearchEventType.Message,
|
||||
Data = data
|
||||
};
|
||||
currentDataType = SearchEventType.Message;
|
||||
messageBuffer.Text += data;
|
||||
break;
|
||||
|
||||
@ -152,6 +180,7 @@ public class ProductService : IProductService
|
||||
Name = productBuffer.Name,
|
||||
Rating = 0,
|
||||
Description = "",
|
||||
Price = 0,
|
||||
ImagesUrls = new []{"", ""},
|
||||
WasOpened = false
|
||||
}, cancellationToken);
|
||||
@ -162,6 +191,15 @@ public class ProductService : IProductService
|
||||
}
|
||||
}
|
||||
}
|
||||
if (currentDataType == SearchEventType.Message)
|
||||
{
|
||||
_wishlistsService.AddMessageToPersonalWishlistAsync(wishlistId, new MessageDto()
|
||||
{
|
||||
Text = messageBuffer.Text,
|
||||
Role = MessageRoles.Application.ToString(),
|
||||
}, cancellationToken);
|
||||
mqchecker = false;
|
||||
}
|
||||
}
|
||||
|
||||
private SearchEventType DetermineDataType(string dataTypeHolder)
|
||||
|
@ -45,7 +45,7 @@ public class WishlistsService : IWishlistsService
|
||||
throw new InvalidDataException("Provided type is invalid.");
|
||||
}
|
||||
|
||||
newWishlist.CreatedById = (ObjectId) GlobalUser.Id;
|
||||
newWishlist.CreatedById = GlobalUser.Id.Value;
|
||||
newWishlist.CreatedDateUtc = DateTime.UtcNow;
|
||||
newWishlist.Name = $"{newWishlist.Type} Search";
|
||||
|
||||
@ -54,8 +54,8 @@ public class WishlistsService : IWishlistsService
|
||||
var newMessage = new Message
|
||||
{
|
||||
Text = dto.FirstMessageText,
|
||||
Role = MessageRoles.User.ToString(),
|
||||
CreatedById = (ObjectId) GlobalUser.Id,
|
||||
Role = MessageRoles.Application.ToString(),
|
||||
CreatedById = GlobalUser.Id.Value,
|
||||
CreatedDateUtc = DateTime.UtcNow,
|
||||
WishlistId = createdWishlist.Id
|
||||
};
|
||||
@ -96,12 +96,12 @@ public class WishlistsService : IWishlistsService
|
||||
var openAiMessage = await _openAiService.GetChatCompletion(chatCompletionRequest, cancellationToken);
|
||||
|
||||
wishlist = await _wishlistsRepository.UpdateWishlistNameAsync(wishlist.Id,
|
||||
openAiMessage.Content, (ObjectId) GlobalUser.Id, cancellationToken);
|
||||
openAiMessage.Content, GlobalUser.Id.Value, cancellationToken);
|
||||
|
||||
return _mapper.Map<WishlistDto>(wishlist);
|
||||
}
|
||||
|
||||
public async Task<MessageDto> AddMessageToPersonalWishlistAsync(string wishlistId, MessageCreateDto dto, CancellationToken cancellationToken)
|
||||
public async Task<MessageDto> AddMessageToPersonalWishlistAsync(string wishlistId, MessageDto dto, CancellationToken cancellationToken)
|
||||
{
|
||||
var newMessage = _mapper.Map<Message>(dto);
|
||||
|
||||
@ -110,8 +110,7 @@ public class WishlistsService : IWishlistsService
|
||||
throw new InvalidDataException("Provided id is invalid.");
|
||||
}
|
||||
|
||||
newMessage.Role = MessageRoles.User.ToString();
|
||||
newMessage.CreatedById = (ObjectId) GlobalUser.Id;
|
||||
newMessage.CreatedById = GlobalUser.Id.Value;
|
||||
newMessage.CreatedDateUtc = DateTime.UtcNow;
|
||||
newMessage.WishlistId = wishlistObjectId;
|
||||
|
||||
@ -169,7 +168,7 @@ public class WishlistsService : IWishlistsService
|
||||
|
||||
await TryGetPersonalWishlist(wishlistObjectId, cancellationToken);
|
||||
|
||||
newProduct.CreatedById = (ObjectId) GlobalUser.Id;
|
||||
newProduct.CreatedById = GlobalUser.Id.Value;
|
||||
newProduct.CreatedDateUtc = DateTime.UtcNow;
|
||||
newProduct.WishlistId = wishlistObjectId;
|
||||
|
||||
|
@ -212,6 +212,7 @@ public class DbInitialaizer
|
||||
Name = "Thermaltake Glacier 360 Liquid-Cooled PC",
|
||||
Description = "Cool PC for any task!",
|
||||
Rating = 4.3,
|
||||
Price = 855,
|
||||
Url = "https://www.amazon.com/Thermaltake-Liquid-Cooled-ToughRAM-Computer-S3WT-B550-G36-LCS/dp" +
|
||||
"/B09FYNM2GW/ref=sr_1_1?crid=391KAS4JFJSFF&keywords=gaming%2Bpc&qid=1697132083&sprefix=gaming%2Bpc%2Caps%2C209&sr=8-1&th=1",
|
||||
ImagesUrls = new string[]
|
||||
@ -229,6 +230,7 @@ public class DbInitialaizer
|
||||
Name = "Apple MagSafe Battery Pack",
|
||||
Description = "Portable Charger with Fast Charging Capability, Power Bank Compatible with iPhone",
|
||||
Rating = 4.3,
|
||||
Price = 35.99,
|
||||
Url = "https://www.amazon.com/Apple-MJWY3AM-A-MagSafe-Battery/dp/" +
|
||||
"B099BWY7WT/ref=sr_1_1?keywords=apple+power+bank&qid=1697375350&sr=8-1",
|
||||
ImagesUrls = new string[]
|
||||
@ -247,6 +249,7 @@ public class DbInitialaizer
|
||||
Name = "Logitech K400 Plus Wireless Touch With Easy Media Control and Built-in Touchpad",
|
||||
Description = "Reliable membrane keyboard with touchpad!",
|
||||
Rating = 4.5,
|
||||
Price = 99,
|
||||
Url = "https://www.amazon.com/Logitech-Wireless-Keyboard-Touchpad-PC-connected/dp/B014EUQOGK/" +
|
||||
"ref=sr_1_11?crid=BU2PHZKHKD65&keywords=keyboard+wireless&qid=1697375559&sprefix=keyboard+wir%2Caps%2C195&sr=8-11",
|
||||
ImagesUrls = new string[]
|
||||
@ -266,6 +269,7 @@ public class DbInitialaizer
|
||||
Description = "Cross computer control: Game changing capacity to navigate seamlessly on three computers," +
|
||||
" and copy paste text, images, and files from one to the other using Logitech Flow",
|
||||
Rating = 4.6,
|
||||
Price = 50.99,
|
||||
Url = "https://www.amazon.com/Logitech-Hyper-Fast-Scrolling-Rechargeable-Computers/dp/B08P2JFPQC/ref=sr_1_8?" +
|
||||
"crid=2BL6Z14W2TPP3&keywords=mouse%2Bwireless&qid=1697375784&sprefix=mousewireless%2Caps%2C197&sr=8-8&th=1",
|
||||
ImagesUrls = new string[]
|
||||
|
@ -257,6 +257,7 @@ public class DbInitializer
|
||||
Name = "AMD Ryzen 5 5600G 6-Core 12-Thread Unlocked Desktop Processor with Radeon Graphics",
|
||||
Description = "Features best-in-class graphics performance in a desktop processor for smooth 1080p gaming, no graphics card required",
|
||||
Rating = 4.8,
|
||||
Price = 120,
|
||||
Url = "https://a.co/d/5ceuIrq",
|
||||
ImagesUrls = new string[]
|
||||
{
|
||||
@ -273,6 +274,7 @@ public class DbInitializer
|
||||
Name = "Samsung 970 EVO Plus SSD 2TB NVMe M.2 Internal Solid State Hard Drive, V-NAND Technology, Storage and Memory Expansion for Gaming, Graphics w/ Heat Control, Max Speed, MZ-V7S2T0B/AM ",
|
||||
Description = "7 Year Limited Warranty: The 970 EVO Plus provides up to 1200 TBW (Terabytes Written) with 5-years of protection for exceptional endurance powered by the latest V-NAND technology and Samsung's reputation for quality ",
|
||||
Rating = 4.8,
|
||||
Price = 153,
|
||||
Url = "https://a.co/d/gxnuqs1",
|
||||
ImagesUrls = new string[]
|
||||
{
|
||||
|
@ -3,6 +3,7 @@ using ShoppingAssistantApi.Domain.Enums;
|
||||
using ShoppingAssistantApi.Application.Models.Dtos;
|
||||
using ShoppingAssistantApi.Application.Paging;
|
||||
using Newtonsoft.Json.Linq;
|
||||
using MongoDB.Bson;
|
||||
|
||||
namespace ShoppingAssistantApi.Tests.Tests;
|
||||
|
||||
@ -88,6 +89,30 @@ public class WishlistsTests : TestsBase
|
||||
|
||||
Assert.NotNull(startWishlistResponse);
|
||||
|
||||
const string MessageText = "I want laptop";
|
||||
var mutation = new
|
||||
{
|
||||
query = @"
|
||||
mutation addMessageToPersonalWishlist($wishlistId: String!, $dto: MessageDtoInput!) {
|
||||
addMessageToPersonalWishlist(wishlistId: $wishlistId, dto: $dto) {
|
||||
role, text, createdById
|
||||
}
|
||||
}",
|
||||
variables = new
|
||||
{
|
||||
wishlistId = startWishlistResponse.Id,
|
||||
dto = new
|
||||
{
|
||||
id = ObjectId.Empty,
|
||||
text = MessageText,
|
||||
role = MessageRoles.User.ToString(),
|
||||
createdById = ObjectId.Empty,
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
await SendGraphQlRequestAsync(mutation);
|
||||
|
||||
var generateWishlistNameMutation = new
|
||||
{
|
||||
query = @"
|
||||
@ -174,7 +199,7 @@ public class WishlistsTests : TestsBase
|
||||
var mutation = new
|
||||
{
|
||||
query = @"
|
||||
mutation addMessageToPersonalWishlist($wishlistId: String!, $dto: MessageCreateDtoInput!) {
|
||||
mutation addMessageToPersonalWishlist($wishlistId: String!, $dto: MessageDtoInput!) {
|
||||
addMessageToPersonalWishlist(wishlistId: $wishlistId, dto: $dto) {
|
||||
role, text, createdById
|
||||
}
|
||||
@ -184,7 +209,10 @@ public class WishlistsTests : TestsBase
|
||||
wishlistId = TestingValidWishlistId,
|
||||
dto = new
|
||||
{
|
||||
text = MessageText
|
||||
id = ObjectId.Empty,
|
||||
text = MessageText,
|
||||
role = MessageRoles.User.ToString(),
|
||||
createdById = ObjectId.Empty,
|
||||
}
|
||||
}
|
||||
};
|
||||
@ -242,7 +270,7 @@ public class WishlistsTests : TestsBase
|
||||
query = @"
|
||||
mutation addProductToPersonalWishlist($wishlistId: String!, $dto: ProductCreateDtoInput!) {
|
||||
addProductToPersonalWishlist (wishlistId: $wishlistId, dto: $dto) {
|
||||
url, name, description, rating, imagesUrls, wasOpened
|
||||
url, name, price, description, rating, imagesUrls, wasOpened
|
||||
}
|
||||
}",
|
||||
variables = new
|
||||
@ -252,6 +280,7 @@ public class WishlistsTests : TestsBase
|
||||
{
|
||||
url = "https://www.amazon.com/url",
|
||||
name = "Generic name",
|
||||
price = 1,
|
||||
description = "Generic description",
|
||||
rating = 4.8,
|
||||
imagesUrls = new string[]
|
||||
@ -272,6 +301,7 @@ public class WishlistsTests : TestsBase
|
||||
Assert.Equal("Generic name", product.Name);
|
||||
Assert.Equal("Generic description", product.Description);
|
||||
Assert.Equal(4.8, product.Rating);
|
||||
Assert.Equal(1, product.Price);
|
||||
Assert.Equal("https://www.amazon.com/image-url-1", product.ImagesUrls[0]);
|
||||
}
|
||||
|
||||
|
@ -46,10 +46,10 @@ public class ProductTests
|
||||
{
|
||||
"[", "Message", "]", " What", " u", " want", " ?", "[", "Options", "]", " USB-C", " ;", " Keyboard", " ultra",
|
||||
" ;", "[", "Options", "]", " USB", "-C", " ;", "[", "Products", "]", " GTX", " 3090", " ;", " GTX",
|
||||
" 3070TI", " ;", " GTX", " 4070TI", " ;", " ?"
|
||||
" 3070TI", " ;", " GTX", " 4070TI", " ;", " ?", "[", "Message", "]", " What", " u", " want", " ?"
|
||||
};
|
||||
|
||||
var expectedMessages = new List<string> { " What", " u", " want", " ?" };
|
||||
var expectedMessages = new List<string> { " What", " u", " want", " ?", " What", " u", " want", " ?" };
|
||||
var expectedSuggestion = new List<string> { " USB-C", " Keyboard ultra", " USB-C" };
|
||||
|
||||
// Mock the GetChatCompletionStream method to provide the expected SSE data
|
||||
@ -59,7 +59,7 @@ public class ProductTests
|
||||
_messagesRepositoryMock.Setup(m => m.GetCountAsync(It.IsAny<Expression<Func<Message, bool>>>(), It.IsAny<CancellationToken>()))
|
||||
.ReturnsAsync(1);
|
||||
|
||||
_wishListServiceMock.Setup(w => w.AddMessageToPersonalWishlistAsync(wishlistId, It.IsAny<MessageCreateDto>(), cancellationToken))
|
||||
_wishListServiceMock.Setup(w => w.AddMessageToPersonalWishlistAsync(wishlistId, It.IsAny<MessageDto>(), cancellationToken))
|
||||
.Verifiable();
|
||||
|
||||
_wishListServiceMock
|
||||
@ -135,7 +135,7 @@ public class ProductTests
|
||||
_messagesRepositoryMock.Setup(m => m.GetCountAsync(It.IsAny<Expression<Func<Message, bool>>>(), It.IsAny<CancellationToken>()))
|
||||
.ReturnsAsync(3);
|
||||
|
||||
_wishListServiceMock.Setup(w => w.AddMessageToPersonalWishlistAsync(wishlistId, It.IsAny<MessageCreateDto>(), cancellationToken))
|
||||
_wishListServiceMock.Setup(w => w.AddMessageToPersonalWishlistAsync(wishlistId, It.IsAny<MessageDto>(), cancellationToken))
|
||||
.Verifiable();
|
||||
|
||||
_wishListServiceMock
|
||||
@ -186,7 +186,7 @@ public class ProductTests
|
||||
Assert.NotNull(actualSseEvents);
|
||||
Assert.Equal(expectedMessages, receivedMessages);
|
||||
Assert.Equal(expectedSuggestions, receivedSuggestions);
|
||||
_wishListServiceMock.Verify(w => w.AddMessageToPersonalWishlistAsync(wishlistId, It.IsAny<MessageCreateDto>(), cancellationToken), Times.Once);
|
||||
_wishListServiceMock.Verify(w => w.AddMessageToPersonalWishlistAsync(wishlistId, It.IsAny<MessageDto>(), cancellationToken), Times.Once);
|
||||
}
|
||||
|
||||
|
||||
@ -276,6 +276,6 @@ public class ProductTests
|
||||
_wishListServiceMock.Verify(w => w.AddProductToPersonalWishlistAsync(
|
||||
It.IsAny<string>(), It.IsAny<ProductCreateDto>(), It.IsAny<CancellationToken>()), Times.Exactly(3));
|
||||
_wishListServiceMock.Verify(w => w.AddMessageToPersonalWishlistAsync(
|
||||
wishlistId, It.IsAny<MessageCreateDto>(), cancellationToken), Times.Once);
|
||||
wishlistId, It.IsAny<MessageDto>(), cancellationToken), Times.Once);
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user