mirror of
https://github.com/Shchoholiev/shopping-assistant-api.git
synced 2025-04-02 23:59:35 +00:00
Merge branch 'develop' into feature/SA-14-guest-authorization
This commit is contained in:
commit
5fada6c3ec
3
.vscode/settings.json
vendored
3
.vscode/settings.json
vendored
@ -1,5 +1,6 @@
|
||||
{
|
||||
"files.exclude": {
|
||||
"**/bin": true
|
||||
}
|
||||
},
|
||||
"editor.formatOnType": true
|
||||
}
|
@ -7,11 +7,19 @@ namespace ShoppingAssistantApi.Api.Mutations;
|
||||
[ExtendObjectType(OperationTypeNames.Mutation)]
|
||||
public class WishlistsMutation
|
||||
{
|
||||
public Task<WishlistDto> StartPersonalWishlist(WishlistCreateDto dto, CancellationToken cancellationToken,
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
public Task<WishlistDto> StartPersonalWishlistAsync(WishlistCreateDto dto, CancellationToken cancellationToken,
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.StartPersonalWishlistAsync(dto, cancellationToken);
|
||||
|
||||
public Task<MessageDto> AddMessageToPersonalWishlist(string wishlistId, MessageCreateDto dto, CancellationToken cancellationToken,
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
public Task<MessageDto> AddMessageToPersonalWishlistAsync(string wishlistId, MessageCreateDto dto, CancellationToken cancellationToken,
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.AddMessageToPersonalWishlistAsync(wishlistId, dto, cancellationToken);
|
||||
|
||||
public Task<ProductDto> AddProductToPersonalWishlistAsync(string wishlistId, ProductCreateDto dto, CancellationToken cancellationToken,
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.AddProductToPersonalWishlistAsync(wishlistId, dto, cancellationToken);
|
||||
|
||||
public Task<WishlistDto> DeletePersonalWishlistAsync(string wishlistId, CancellationToken cancellationToken,
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.DeletePersonalWishlistAsync(wishlistId, cancellationToken);
|
||||
}
|
||||
|
@ -9,12 +9,22 @@ namespace ShoppingAssistantApi.Api.Queries;
|
||||
public class WishlistsQuery
|
||||
{
|
||||
[Authorize]
|
||||
public Task<PagedList<WishlistDto>> GetPersonalWishlistsPageAsync(int pageNumber, int pageSize, CancellationToken cancellationToken,
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.GetPersonalWishlistsPageAsync(pageNumber, pageSize, cancellationToken);
|
||||
public Task<PagedList<WishlistDto>> GetPersonalWishlistsPageAsync(int pageNumber, int pageSize,
|
||||
CancellationToken cancellationToken, [Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.GetPersonalWishlistsPageAsync(pageNumber, pageSize, cancellationToken);
|
||||
|
||||
[Authorize]
|
||||
public Task<WishlistDto> GetPersonalWishlistAsync(string wishlistId, CancellationToken cancellationToken,
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.GetPersonalWishlistAsync(wishlistId, cancellationToken);
|
||||
[Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.GetPersonalWishlistAsync(wishlistId, cancellationToken);
|
||||
|
||||
[Authorize]
|
||||
public Task<PagedList<MessageDto>> GetMessagesPageFromPersonalWishlistAsync(string wishlistId, int pageNumber, int pageSize,
|
||||
CancellationToken cancellationToken, [Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.GetMessagesPageFromPersonalWishlistAsync(wishlistId, pageNumber, pageSize, cancellationToken);
|
||||
|
||||
[Authorize]
|
||||
public Task<PagedList<ProductDto>> GetProductsPageFromPersonalWishlistAsync(string wishlistId, int pageNumber, int pageSize,
|
||||
CancellationToken cancellationToken, [Service] IWishlistsService wishlistsService)
|
||||
=> wishlistsService.GetProductsPageFromPersonalWishlistAsync(wishlistId, pageNumber, pageSize, cancellationToken);
|
||||
}
|
||||
|
@ -1,5 +1,9 @@
|
||||
using System.Linq.Expressions;
|
||||
using ShoppingAssistantApi.Domain.Entities;
|
||||
|
||||
namespace ShoppingAssistantApi.Application.IRepositories;
|
||||
|
||||
public interface IMessagesRepository : IBaseRepository<Message> { }
|
||||
public interface IMessagesRepository : IBaseRepository<Message>
|
||||
{
|
||||
Task<List<Message>> GetPageStartingFromEndAsync(int pageNumber, int pageSize, Expression<Func<Message, bool>> predicate, CancellationToken cancellationToken);
|
||||
}
|
||||
|
@ -0,0 +1,5 @@
|
||||
using ShoppingAssistantApi.Domain.Entities;
|
||||
|
||||
namespace ShoppingAssistantApi.Application.IRepositories;
|
||||
|
||||
public interface IProductsRepository : IBaseRepository<Product> { }
|
13
ShoppingAssistantApi.Application/IServices/IOpenAiService.cs
Normal file
13
ShoppingAssistantApi.Application/IServices/IOpenAiService.cs
Normal file
@ -0,0 +1,13 @@
|
||||
using ShoppingAssistantApi.Application.Models.OpenAi;
|
||||
|
||||
namespace ShoppingAssistantApi.Application.IServices;
|
||||
|
||||
public interface IOpenAiService
|
||||
{
|
||||
Task<OpenAiMessage> GetChatCompletion(ChatCompletionRequest chat, CancellationToken cancellationToken);
|
||||
|
||||
/// <summary>
|
||||
/// Retrieves a stream of tokens (pieces of words) based on provided chat.
|
||||
/// </summary>
|
||||
IAsyncEnumerable<string> GetChatCompletionStream(ChatCompletionRequest chat, CancellationToken cancellationToken);
|
||||
}
|
@ -13,4 +13,12 @@ public interface IWishlistsService
|
||||
Task<PagedList<WishlistDto>> GetPersonalWishlistsPageAsync(int pageNumber, int pageSize, CancellationToken cancellationToken);
|
||||
|
||||
Task<WishlistDto> GetPersonalWishlistAsync(string wishlistId, CancellationToken cancellationToken);
|
||||
|
||||
Task<PagedList<MessageDto>> GetMessagesPageFromPersonalWishlistAsync(string wishlistId, int pageNumber, int pageSize, CancellationToken cancellationToken);
|
||||
|
||||
Task<ProductDto> AddProductToPersonalWishlistAsync(string wishlistId, ProductCreateDto dto, CancellationToken cancellationToken);
|
||||
|
||||
Task<PagedList<ProductDto>> GetProductsPageFromPersonalWishlistAsync(string wishlistId, int pageNumber, int pageSize, CancellationToken cancellationToken);
|
||||
|
||||
Task<WishlistDto> DeletePersonalWishlistAsync(string wishlistId, CancellationToken cancellationToken);
|
||||
}
|
||||
|
@ -0,0 +1,15 @@
|
||||
using AutoMapper;
|
||||
using ShoppingAssistantApi.Application.Models.CreateDtos;
|
||||
using ShoppingAssistantApi.Application.Models.Dtos;
|
||||
using ShoppingAssistantApi.Domain.Entities;
|
||||
|
||||
namespace ShoppingAssistantApi.Application.MappingProfiles;
|
||||
public class ProductProfile : Profile
|
||||
{
|
||||
public ProductProfile()
|
||||
{
|
||||
CreateMap<Product, ProductDto>().ReverseMap();
|
||||
|
||||
CreateMap<ProductCreateDto, Product>().ReverseMap();
|
||||
}
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
namespace ShoppingAssistantApi.Application.Models.CreateDtos;
|
||||
|
||||
public class ProductCreateDto
|
||||
{
|
||||
public required string Url { get; set; }
|
||||
|
||||
public required string Name { get; set; }
|
||||
|
||||
public required string Description { get; set; }
|
||||
|
||||
public required double Rating { get; set; }
|
||||
|
||||
public required string[] ImagesUrls { get; set; }
|
||||
|
||||
public required bool WasOpened { get; set; }
|
||||
}
|
@ -8,5 +8,5 @@ public class MessageDto
|
||||
|
||||
public required string Role { get; set; }
|
||||
|
||||
public string? CreatedById { get; set; } = null;
|
||||
public required string CreatedById { get; set; }
|
||||
}
|
||||
|
20
ShoppingAssistantApi.Application/Models/Dtos/ProductDto.cs
Normal file
20
ShoppingAssistantApi.Application/Models/Dtos/ProductDto.cs
Normal file
@ -0,0 +1,20 @@
|
||||
namespace ShoppingAssistantApi.Application.Models.Dtos;
|
||||
|
||||
public class ProductDto
|
||||
{
|
||||
public required string Id { get; set; }
|
||||
|
||||
public required string Url { get; set; }
|
||||
|
||||
public required string Name { get; set; }
|
||||
|
||||
public required string Description { get; set; }
|
||||
|
||||
public required double Rating { get; set; }
|
||||
|
||||
public required string[] ImagesUrls { get; set; }
|
||||
|
||||
public required bool WasOpened { get; set; }
|
||||
|
||||
public required string WishlistId { get; set; }
|
||||
}
|
@ -8,5 +8,5 @@ public class WishlistDto
|
||||
|
||||
public required string Type { get; set; }
|
||||
|
||||
public string CreatedById { get; set; } = null!;
|
||||
public required string CreatedById { get; set; }
|
||||
}
|
||||
|
@ -0,0 +1,14 @@
|
||||
namespace ShoppingAssistantApi.Application.Models.OpenAi;
|
||||
|
||||
public class ChatCompletionRequest
|
||||
{
|
||||
public string Model { get; set; } = "gpt-3.5-turbo";
|
||||
|
||||
public List<OpenAiMessage> Messages { get; set; }
|
||||
|
||||
public double Temperature { get; set; } = 0.7;
|
||||
|
||||
public int MaxTokens { get; set; } = 256;
|
||||
|
||||
public bool Stream { get; set; } = false;
|
||||
}
|
@ -0,0 +1,10 @@
|
||||
using ShoppingAssistantApi.Domain.Enums;
|
||||
|
||||
namespace ShoppingAssistantApi.Application.Models.OpenAi;
|
||||
|
||||
public class OpenAiMessage
|
||||
{
|
||||
public OpenAiRole Role { get; set; }
|
||||
|
||||
public string Content { get; set; }
|
||||
}
|
@ -5,9 +5,9 @@ namespace ShoppingAssistantApi.Domain.Entities;
|
||||
|
||||
public class Message : EntityBase
|
||||
{
|
||||
public required string Text { get; set; }
|
||||
public string Text { get; set; }
|
||||
|
||||
public required string Role { get; set; }
|
||||
public string Role { get; set; }
|
||||
|
||||
public ObjectId WishlistId { get; set; }
|
||||
}
|
||||
|
22
ShoppingAssistantApi.Domain/Entities/Product.cs
Normal file
22
ShoppingAssistantApi.Domain/Entities/Product.cs
Normal file
@ -0,0 +1,22 @@
|
||||
using MongoDB.Bson;
|
||||
using ShoppingAssistantApi.Domain.Common;
|
||||
|
||||
namespace ShoppingAssistantApi.Domain.Entities;
|
||||
|
||||
public class Product : EntityBase
|
||||
{
|
||||
|
||||
public string Url { get; set; }
|
||||
|
||||
public string Name { get; set; }
|
||||
|
||||
public string Description { get; set; }
|
||||
|
||||
public double Rating { get; set; }
|
||||
|
||||
public string[] ImagesUrls { get; set; }
|
||||
|
||||
public bool WasOpened { get; set; }
|
||||
|
||||
public ObjectId WishlistId { get; set; }
|
||||
}
|
@ -4,9 +4,9 @@ namespace ShoppingAssistantApi.Domain.Entities;
|
||||
|
||||
public class Wishlist : EntityBase
|
||||
{
|
||||
public required string Name { get; set; }
|
||||
public string Name { get; set; }
|
||||
|
||||
public required string Type { get; set; }
|
||||
public string Type { get; set; }
|
||||
|
||||
public ICollection<Message>? Messages { get; set; }
|
||||
}
|
||||
|
8
ShoppingAssistantApi.Domain/Enums/OpenAiRole.cs
Normal file
8
ShoppingAssistantApi.Domain/Enums/OpenAiRole.cs
Normal file
@ -0,0 +1,8 @@
|
||||
namespace ShoppingAssistantApi.Domain.Enums;
|
||||
|
||||
public enum OpenAiRole
|
||||
{
|
||||
System,
|
||||
User,
|
||||
Assistant
|
||||
}
|
@ -0,0 +1,24 @@
|
||||
using ShoppingAssistantApi.Application.IServices;
|
||||
using ShoppingAssistantApi.Application.Models.OpenAi;
|
||||
|
||||
namespace ShoppingAssistantApi.Infrastructure.Services;
|
||||
|
||||
public class OpenAiService : IOpenAiService
|
||||
{
|
||||
private readonly HttpClient _httpClient;
|
||||
|
||||
public OpenAiService(HttpClient client)
|
||||
{
|
||||
_httpClient = client;
|
||||
}
|
||||
|
||||
public Task<OpenAiMessage> GetChatCompletion(ChatCompletionRequest chat, CancellationToken cancellationToken)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
|
||||
public IAsyncEnumerable<string> GetChatCompletionStream(ChatCompletionRequest chat, CancellationToken cancellationToken)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
}
|
@ -18,12 +18,15 @@ public class WishlistsService : IWishlistsService
|
||||
|
||||
private readonly IMessagesRepository _messagesRepository;
|
||||
|
||||
private readonly IProductsRepository _productsRepository;
|
||||
|
||||
private readonly IMapper _mapper;
|
||||
|
||||
public WishlistsService(IWishlistsRepository wishlistRepository, IMessagesRepository messageRepository, IMapper mapper)
|
||||
public WishlistsService(IWishlistsRepository wishlistRepository, IMessagesRepository messageRepository, IProductsRepository productRepository, IMapper mapper)
|
||||
{
|
||||
_wishlistsRepository = wishlistRepository;
|
||||
_messagesRepository = messageRepository;
|
||||
_productsRepository = productRepository;
|
||||
_mapper = mapper;
|
||||
}
|
||||
|
||||
@ -47,6 +50,8 @@ public class WishlistsService : IWishlistsService
|
||||
{
|
||||
Text = dto.FirstMessageText,
|
||||
Role = MessageRoles.User.ToString(),
|
||||
CreatedById = (ObjectId) GlobalUser.Id,
|
||||
CreatedDateUtc = DateTime.UtcNow,
|
||||
WishlistId = createdWishlist.Id
|
||||
};
|
||||
var createdMessage = await _messagesRepository.AddAsync(newMessage, cancellationToken);
|
||||
@ -62,17 +67,13 @@ public class WishlistsService : IWishlistsService
|
||||
{
|
||||
throw new InvalidDataException("Provided id is invalid.");
|
||||
}
|
||||
newMessage.WishlistId = wishlistObjectId;
|
||||
|
||||
newMessage.Role = MessageRoles.User.ToString();
|
||||
newMessage.CreatedById = (ObjectId) GlobalUser.Id;
|
||||
newMessage.CreatedDateUtc = DateTime.UtcNow;
|
||||
newMessage.WishlistId = wishlistObjectId;
|
||||
|
||||
var relatedWishlist = await _wishlistsRepository.GetWishlistAsync(x => x.Id == wishlistObjectId && x.CreatedById == GlobalUser.Id, cancellationToken);
|
||||
|
||||
if (relatedWishlist == null)
|
||||
{
|
||||
throw new UnAuthorizedException<Wishlist>();
|
||||
}
|
||||
await TryGetPersonalWishlist(wishlistObjectId, cancellationToken);
|
||||
|
||||
var createdMessage = await _messagesRepository.AddAsync(newMessage, cancellationToken);
|
||||
|
||||
@ -81,7 +82,7 @@ public class WishlistsService : IWishlistsService
|
||||
|
||||
public async Task<PagedList<WishlistDto>> GetPersonalWishlistsPageAsync(int pageNumber, int pageSize, CancellationToken cancellationToken)
|
||||
{
|
||||
var entities = await _wishlistsRepository.GetPageAsync(pageNumber, pageSize, cancellationToken);
|
||||
var entities = await _wishlistsRepository.GetPageAsync(pageNumber, pageSize, x => x.CreatedById == GlobalUser.Id, cancellationToken);
|
||||
var dtos = _mapper.Map<List<WishlistDto>>(entities);
|
||||
var count = await _wishlistsRepository.GetTotalCountAsync();
|
||||
return new PagedList<WishlistDto>(dtos, pageNumber, pageSize, count);
|
||||
@ -93,15 +94,95 @@ public class WishlistsService : IWishlistsService
|
||||
{
|
||||
throw new InvalidDataException("Provided id is invalid.");
|
||||
}
|
||||
var entity = await _wishlistsRepository.GetWishlistAsync(x => x.Id == wishlistObjectId && x.CreatedById == GlobalUser.Id, cancellationToken);
|
||||
|
||||
Console.WriteLine(" WISHLIST: " + entity.CreatedById + " " + GlobalUser.Id);
|
||||
|
||||
if (entity == null)
|
||||
{
|
||||
throw new UnAuthorizedException<Wishlist>();
|
||||
}
|
||||
var entity = await TryGetPersonalWishlist(wishlistObjectId, cancellationToken);
|
||||
|
||||
return _mapper.Map<WishlistDto>(entity);
|
||||
}
|
||||
|
||||
public async Task<PagedList<MessageDto>> GetMessagesPageFromPersonalWishlistAsync(string wishlistId, int pageNumber, int pageSize, CancellationToken cancellationToken)
|
||||
{
|
||||
if (!ObjectId.TryParse(wishlistId, out var wishlistObjectId))
|
||||
{
|
||||
throw new InvalidDataException("Provided id is invalid.");
|
||||
}
|
||||
|
||||
await TryGetPersonalWishlist(wishlistObjectId, cancellationToken);
|
||||
|
||||
var entities = await _messagesRepository.GetPageStartingFromEndAsync(pageNumber, pageSize, x => x.WishlistId == wishlistObjectId, cancellationToken);
|
||||
|
||||
var dtos = _mapper.Map<List<MessageDto>>(entities);
|
||||
var count = await _messagesRepository.GetCountAsync(x => x.WishlistId == wishlistObjectId, cancellationToken);
|
||||
return new PagedList<MessageDto>(dtos, pageNumber, pageSize, count);
|
||||
}
|
||||
|
||||
public async Task<ProductDto> AddProductToPersonalWishlistAsync(string wishlistId, ProductCreateDto dto, CancellationToken cancellationToken)
|
||||
{
|
||||
var newProduct = _mapper.Map<Product>(dto);
|
||||
|
||||
if (!ObjectId.TryParse(wishlistId, out var wishlistObjectId))
|
||||
{
|
||||
throw new InvalidDataException("Provided id is invalid.");
|
||||
}
|
||||
|
||||
await TryGetPersonalWishlist(wishlistObjectId, cancellationToken);
|
||||
|
||||
newProduct.CreatedById = (ObjectId) GlobalUser.Id;
|
||||
newProduct.CreatedDateUtc = DateTime.UtcNow;
|
||||
newProduct.WishlistId = wishlistObjectId;
|
||||
|
||||
var createdProduct = await _productsRepository.AddAsync(newProduct, cancellationToken);
|
||||
|
||||
return _mapper.Map<ProductDto>(createdProduct);
|
||||
}
|
||||
|
||||
public async Task<PagedList<ProductDto>> GetProductsPageFromPersonalWishlistAsync(string wishlistId, int pageNumber, int pageSize, CancellationToken cancellationToken)
|
||||
{
|
||||
if (!ObjectId.TryParse(wishlistId, out var wishlistObjectId))
|
||||
{
|
||||
throw new InvalidDataException("Provided id is invalid.");
|
||||
}
|
||||
|
||||
await TryGetPersonalWishlist(wishlistObjectId, cancellationToken);
|
||||
|
||||
var entities = await _productsRepository.GetPageAsync(pageNumber, pageSize, x => x.WishlistId == wishlistObjectId, cancellationToken);
|
||||
|
||||
var dtos = _mapper.Map<List<ProductDto>>(entities);
|
||||
var count = await _productsRepository.GetCountAsync(x => x.WishlistId == wishlistObjectId, cancellationToken);
|
||||
return new PagedList<ProductDto>(dtos, pageNumber, pageSize, count);
|
||||
}
|
||||
|
||||
public async Task<WishlistDto> DeletePersonalWishlistAsync(string wishlistId, CancellationToken cancellationToken)
|
||||
{
|
||||
if (!ObjectId.TryParse(wishlistId, out var wishlistObjectId))
|
||||
{
|
||||
throw new InvalidDataException("Provided id is invalid.");
|
||||
}
|
||||
|
||||
var entity = await TryGetPersonalWishlist(wishlistObjectId, cancellationToken);
|
||||
|
||||
entity.LastModifiedById = GlobalUser.Id;
|
||||
entity.LastModifiedDateUtc = DateTime.UtcNow;
|
||||
|
||||
await _wishlistsRepository.DeleteAsync(entity, cancellationToken);
|
||||
|
||||
return _mapper.Map<WishlistDto>(entity);
|
||||
}
|
||||
|
||||
private async Task<Wishlist> TryGetPersonalWishlist(ObjectId wishlistId, CancellationToken cancellationToken)
|
||||
{
|
||||
var entity = await _wishlistsRepository.GetWishlistAsync(x => x.Id == wishlistId, cancellationToken);
|
||||
|
||||
if (entity.CreatedById != GlobalUser.Id)
|
||||
{
|
||||
throw new UnAuthorizedException<Wishlist>();
|
||||
}
|
||||
|
||||
if (entity == null)
|
||||
{
|
||||
throw new EntityNotFoundException<Wishlist>();
|
||||
}
|
||||
|
||||
return entity;
|
||||
}
|
||||
}
|
||||
|
@ -23,9 +23,13 @@ public class DbInitialaizer
|
||||
|
||||
private readonly ITokensService _tokensService;
|
||||
|
||||
private readonly IWishlistsService _wishlistsService;
|
||||
|
||||
private readonly IMongoCollection<User> _userCollection;
|
||||
|
||||
private readonly IMongoCollection<Wishlist> _wishlistCollection;
|
||||
|
||||
private readonly IMongoCollection<Product> _productCollection;
|
||||
|
||||
public IEnumerable<RoleDto> Roles { get; set; }
|
||||
|
||||
@ -35,8 +39,10 @@ public class DbInitialaizer
|
||||
_rolesService = serviceProvider.GetService<IRolesService>();
|
||||
_userManager = serviceProvider.GetService<IUserManager>();
|
||||
_tokensService = serviceProvider.GetService<ITokensService>();
|
||||
_wishlistsService = serviceProvider.GetService<IWishlistsService>();
|
||||
_wishlistCollection = serviceProvider.GetService<MongoDbContext>().Db.GetCollection<Wishlist>("Wishlists");
|
||||
_userCollection = serviceProvider.GetService<MongoDbContext>().Db.GetCollection<User>("Users");
|
||||
_productCollection = serviceProvider.GetService<MongoDbContext>().Db.GetCollection<Product>("Product");
|
||||
}
|
||||
|
||||
public async Task InitialaizeDb(CancellationToken cancellationToken)
|
||||
@ -44,6 +50,7 @@ public class DbInitialaizer
|
||||
await AddRoles(cancellationToken);
|
||||
await AddUsers(cancellationToken);
|
||||
await AddWishlistsWithMessages(cancellationToken);
|
||||
await AddProducts(cancellationToken);
|
||||
}
|
||||
|
||||
public async Task AddUsers(CancellationToken cancellationToken)
|
||||
@ -186,11 +193,17 @@ public class DbInitialaizer
|
||||
{
|
||||
Text = "Prompt",
|
||||
Role = MessageRoles.User.ToString(),
|
||||
WishlistId = ObjectId.Parse("ab79cde6f69abcd3efab65cd"),
|
||||
CreatedById = user1.Id,
|
||||
CreatedDateUtc = DateTime.UtcNow
|
||||
},
|
||||
new Message
|
||||
{
|
||||
Text = "Answer",
|
||||
Role = MessageRoles.Application.ToString(),
|
||||
WishlistId = ObjectId.Parse("ab79cde6f69abcd3efab65cd"),
|
||||
CreatedById = user1.Id,
|
||||
CreatedDateUtc = DateTime.UtcNow
|
||||
},
|
||||
}
|
||||
},
|
||||
@ -206,6 +219,9 @@ public class DbInitialaizer
|
||||
{
|
||||
Text = "Prompt",
|
||||
Role = MessageRoles.User.ToString(),
|
||||
WishlistId = ObjectId.Parse("ab6c2c2d9edf39abcd1ef9ab"),
|
||||
CreatedById = user1.Id,
|
||||
CreatedDateUtc = DateTime.UtcNow
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -213,4 +229,83 @@ public class DbInitialaizer
|
||||
|
||||
await _wishlistCollection.InsertManyAsync(wishlists);
|
||||
}
|
||||
|
||||
public async Task AddProducts(CancellationToken cancellationToken)
|
||||
{
|
||||
var products = new Product[]
|
||||
{
|
||||
new Product()
|
||||
{
|
||||
Name = "Thermaltake Glacier 360 Liquid-Cooled PC",
|
||||
Description = "Cool PC for any task!",
|
||||
Rating = 4.3,
|
||||
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[]
|
||||
{
|
||||
"https://m.media-amazon.com/images/I/61cXu9yGldL._AC_SL1200_.jpg",
|
||||
"https://m.media-amazon.com/images/I/615gxSGp42L._AC_SL1200_.jpg"
|
||||
},
|
||||
CreatedDateUtc = DateTime.UtcNow,
|
||||
WasOpened = false,
|
||||
WishlistId = ObjectId.Parse("ab79cde6f69abcd3efab65cd")
|
||||
},
|
||||
|
||||
new Product()
|
||||
{
|
||||
Name = "Apple MagSafe Battery Pack",
|
||||
Description = "Portable Charger with Fast Charging Capability, Power Bank Compatible with iPhone",
|
||||
Rating = 4.3,
|
||||
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[]
|
||||
{
|
||||
"https://m.media-amazon.com/images/I/418SjFMB1wL._AC_SX679_.jpg",
|
||||
"https://m.media-amazon.com/images/I/51v4pgChtLL._AC_SX679_.jpg",
|
||||
"https://m.media-amazon.com/images/I/61mJ0z7uYQL._AC_SX679_.jpg"
|
||||
},
|
||||
CreatedDateUtc = DateTime.UtcNow,
|
||||
WasOpened = false,
|
||||
WishlistId = ObjectId.Parse("ab79cde6f69abcd3efab65cd")
|
||||
},
|
||||
|
||||
new Product()
|
||||
{
|
||||
Name = "Logitech K400 Plus Wireless Touch With Easy Media Control and Built-in Touchpad",
|
||||
Description = "Reliable membrane keyboard with touchpad!",
|
||||
Rating = 4.5,
|
||||
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[]
|
||||
{
|
||||
"https://m.media-amazon.com/images/I/51yjnWJ5urL._AC_SX466_.jpg",
|
||||
"https://m.media-amazon.com/images/I/71al70zP7QL._AC_SX466_.jpg",
|
||||
"https://m.media-amazon.com/images/I/71+JXDDY01L._AC_SX466_.jpg"
|
||||
},
|
||||
CreatedDateUtc = DateTime.UtcNow,
|
||||
WasOpened = false,
|
||||
WishlistId = ObjectId.Parse("ab6c2c2d9edf39abcd1ef9ab")
|
||||
},
|
||||
|
||||
new Product()
|
||||
{
|
||||
Name = "Logitech MX Anywhere 2S Wireless Mouse Use On Any Surface",
|
||||
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,
|
||||
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[]
|
||||
{
|
||||
"https://m.media-amazon.com/images/I/6170mJHIsYL._AC_SX466_.jpg",
|
||||
"https://m.media-amazon.com/images/I/71a5As76MDL._AC_SX466_.jpg"
|
||||
},
|
||||
CreatedDateUtc = DateTime.UtcNow,
|
||||
WasOpened = false,
|
||||
WishlistId = ObjectId.Parse("ab6c2c2d9edf39abcd1ef9ab")
|
||||
}
|
||||
};
|
||||
|
||||
await _productCollection.InsertManyAsync(products);
|
||||
}
|
||||
}
|
||||
|
@ -16,6 +16,7 @@ public static class RepositoriesExtention
|
||||
services.AddScoped<IRefreshTokensRepository, RefreshTokensRepository>();
|
||||
services.AddScoped<IWishlistsRepository, WishlistsRepository>();
|
||||
services.AddScoped<IMessagesRepository, MessagesRepository>();
|
||||
services.AddScoped<IProductsRepository, ProductsRepository>();
|
||||
|
||||
return services;
|
||||
}
|
||||
|
@ -81,4 +81,4 @@ public abstract class BaseRepository<TEntity> : IBaseRepository<TEntity> where T
|
||||
return await this._collection.FindOneAndUpdateAsync(
|
||||
Builders<TEntity>.Filter.Eq(e => e.Id, entity.Id), updateDefinition, options, cancellationToken);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1,3 +1,5 @@
|
||||
using System.Linq.Expressions;
|
||||
using MongoDB.Driver;
|
||||
using ShoppingAssistantApi.Application.IRepositories;
|
||||
using ShoppingAssistantApi.Domain.Entities;
|
||||
using ShoppingAssistantApi.Persistance.Database;
|
||||
@ -7,4 +9,13 @@ namespace ShoppingAssistantApi.Persistance.Repositories;
|
||||
public class MessagesRepository : BaseRepository<Message>, IMessagesRepository
|
||||
{
|
||||
public MessagesRepository(MongoDbContext db) : base(db, "Messages") { }
|
||||
|
||||
public async Task<List<Message>> GetPageStartingFromEndAsync(int pageNumber, int pageSize, Expression<Func<Message, bool>> predicate, CancellationToken cancellationToken)
|
||||
{
|
||||
return await _collection.Find(predicate)
|
||||
.SortByDescending(x => x.CreatedDateUtc)
|
||||
.Skip((pageNumber - 1) * pageSize)
|
||||
.Limit(pageSize)
|
||||
.ToListAsync(cancellationToken);
|
||||
}
|
||||
}
|
||||
|
@ -0,0 +1,10 @@
|
||||
using ShoppingAssistantApi.Application.IRepositories;
|
||||
using ShoppingAssistantApi.Domain.Entities;
|
||||
using ShoppingAssistantApi.Persistance.Database;
|
||||
|
||||
namespace ShoppingAssistantApi.Persistance.Repositories;
|
||||
|
||||
public class ProductsRepository : BaseRepository<Product>, IProductsRepository
|
||||
{
|
||||
public ProductsRepository(MongoDbContext db) : base(db, "Products") { }
|
||||
}
|
@ -97,29 +97,35 @@ public class DbInitializer
|
||||
public async Task InitializeWishlistsAsync()
|
||||
{
|
||||
var wishlistsCollection = _dbContext.Db.GetCollection<Wishlist>("Wishlists");
|
||||
var messagesCollection = _dbContext.Db.GetCollection<Message>("Messages");
|
||||
|
||||
var gamingPcWishlist = new Wishlist
|
||||
{
|
||||
Id = ObjectId.Parse("ab79cde6f69abcd3efab65cd"),
|
||||
Name = "Gaming PC",
|
||||
Type = WishlistTypes.Product.ToString(),
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6418fc"),
|
||||
Messages = new Message[]
|
||||
{
|
||||
new Message
|
||||
{
|
||||
Text = "Prompt",
|
||||
Role = MessageRoles.User.ToString(),
|
||||
},
|
||||
new Message
|
||||
{
|
||||
Text = "Answer",
|
||||
Role = MessageRoles.Application.ToString(),
|
||||
},
|
||||
}
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6418fc")
|
||||
};
|
||||
await wishlistsCollection.InsertOneAsync(gamingPcWishlist);
|
||||
|
||||
await messagesCollection.InsertManyAsync(new Message[]
|
||||
{
|
||||
new() {
|
||||
WishlistId = ObjectId.Parse("ab79cde6f69abcd3efab65cd"),
|
||||
Text = "Prompt",
|
||||
Role = MessageRoles.User.ToString(),
|
||||
CreatedDateUtc = DateTime.UtcNow.AddMinutes(-1),
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6418fc")
|
||||
},
|
||||
new() {
|
||||
WishlistId = ObjectId.Parse("ab79cde6f69abcd3efab65cd"),
|
||||
Text = "Answer",
|
||||
Role = MessageRoles.Application.ToString(),
|
||||
CreatedDateUtc = DateTime.UtcNow,
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6418fc")
|
||||
},
|
||||
});
|
||||
|
||||
var genericWishlist = new Wishlist
|
||||
{
|
||||
Id = ObjectId.Parse("ab6c2c2d9edf39abcd1ef9ab"),
|
||||
@ -130,11 +136,55 @@ public class DbInitializer
|
||||
{
|
||||
new Message
|
||||
{
|
||||
Text = "Prompt",
|
||||
Text = "One Message",
|
||||
Role = MessageRoles.User.ToString(),
|
||||
CreatedDateUtc = DateTime.UtcNow.AddMinutes(-1),
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6409fc")
|
||||
}
|
||||
}
|
||||
};
|
||||
await wishlistsCollection.InsertOneAsync(genericWishlist);
|
||||
await messagesCollection.InsertOneAsync(new Message
|
||||
{
|
||||
WishlistId = ObjectId.Parse("ab6c2c2d9edf39abcd1ef9ab"),
|
||||
Text = "One Message",
|
||||
Role = MessageRoles.User.ToString(),
|
||||
CreatedDateUtc = DateTime.UtcNow.AddMinutes(-1),
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6409fc")
|
||||
});
|
||||
|
||||
var mouseWishlist = new Wishlist
|
||||
{
|
||||
Id = ObjectId.Parse("ab79cde6f69abcd3efab95cd"),
|
||||
Name = "Mouse",
|
||||
Type = WishlistTypes.Product.ToString(),
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6418fc"),
|
||||
};
|
||||
await wishlistsCollection.InsertOneAsync(mouseWishlist);
|
||||
|
||||
await messagesCollection.InsertManyAsync(new List<Message>
|
||||
{
|
||||
new() {
|
||||
WishlistId = ObjectId.Parse("ab79cde6f69abcd3efab95cd"),
|
||||
Text = "First Message",
|
||||
Role = MessageRoles.User.ToString(),
|
||||
CreatedDateUtc = DateTime.UtcNow.AddMinutes(-2),
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6418fc"),
|
||||
},
|
||||
new() {
|
||||
WishlistId = ObjectId.Parse("ab79cde6f69abcd3efab95cd"),
|
||||
Text = "Second Message",
|
||||
Role = MessageRoles.Application.ToString(),
|
||||
CreatedDateUtc = DateTime.UtcNow.AddMinutes(-1),
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6418fc"),
|
||||
},
|
||||
new() {
|
||||
WishlistId = ObjectId.Parse("ab79cde6f69abcd3efab95cd"),
|
||||
Text = "Third Message",
|
||||
Role = MessageRoles.User.ToString(),
|
||||
CreatedDateUtc = DateTime.UtcNow,
|
||||
CreatedById = ObjectId.Parse("652c3b89ae02a3135d6418fc"),
|
||||
},
|
||||
});
|
||||
}
|
||||
}
|
||||
|
@ -1,4 +1,4 @@
|
||||
using Newtonsoft.Json.Linq;
|
||||
using Newtonsoft.Json.Linq;
|
||||
using ShoppingAssistantApi.Application.Models.Identity;
|
||||
using ShoppingAssistantApi.Tests.TestExtentions;
|
||||
using Xunit;
|
||||
|
@ -107,7 +107,7 @@ public class WishlistsTests : TestsBase
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task AddMessageToPersonalWishlist_ValidMessageModel_ReturnsNewMessageModel()
|
||||
public async Task AddMessageToPersonalWishlist_ValidMessage_ReturnsNewMessage()
|
||||
{
|
||||
await LoginAsync(TestingUserEmail, TestingUserPassword);
|
||||
const string MessageText = "Second Message";
|
||||
@ -138,6 +138,41 @@ public class WishlistsTests : TestsBase
|
||||
Assert.Equal(TestingUserId, message.CreatedById);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task GetMessagesPageFromPersonalWishlist_ValidPageNumberAndSize_ReturnsPage()
|
||||
{
|
||||
await LoginAsync(TestingUserEmail, TestingUserPassword);
|
||||
var mutation = new
|
||||
{
|
||||
query = @"
|
||||
query messagesPageFromPersonalWishlist($wishlistId: String!, $pageNumber: Int!, $pageSize: Int!) {
|
||||
messagesPageFromPersonalWishlist (wishlistId: $wishlistId, pageNumber: $pageNumber, pageSize: $pageSize) {
|
||||
hasNextPage,
|
||||
hasPreviousPage,
|
||||
items { id, text, role, createdById },
|
||||
pageNumber,
|
||||
pageSize,
|
||||
totalItems,
|
||||
totalPages
|
||||
}
|
||||
}",
|
||||
variables = new
|
||||
{
|
||||
wishlistId = "ab79cde6f69abcd3efab95cd", // From DbInitializer
|
||||
pageNumber = 1,
|
||||
pageSize = 2
|
||||
}
|
||||
};
|
||||
|
||||
var jsonObject = await SendGraphQlRequestAsync(mutation);
|
||||
var pagedList = (PagedList<MessageDto>?) jsonObject?.data?.messagesPageFromPersonalWishlist?.ToObject<PagedList<MessageDto>>();
|
||||
|
||||
Assert.NotNull(pagedList);
|
||||
Assert.NotEmpty(pagedList.Items);
|
||||
Assert.Equal("Third Message", pagedList.Items.FirstOrDefault()?.Text);
|
||||
Assert.Equal(MessageRoles.User.ToString(), pagedList.Items.FirstOrDefault()?.Role);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task StartPersonalWishlistAsync_InvalidWishlist_ReturnsErrors()
|
||||
{
|
||||
@ -245,4 +280,4 @@ public class WishlistsTests : TestsBase
|
||||
Assert.NotNull(errors);
|
||||
Assert.True(errors.Count > 0);
|
||||
}
|
||||
}
|
||||
}
|
1
ShoppingAssistantApi.UnitTests/GlobalUsings.cs
Normal file
1
ShoppingAssistantApi.UnitTests/GlobalUsings.cs
Normal file
@ -0,0 +1 @@
|
||||
global using Xunit;
|
@ -0,0 +1,31 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net7.0</TargetFramework>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
<Nullable>enable</Nullable>
|
||||
|
||||
<IsPackable>false</IsPackable>
|
||||
<IsTestProject>true</IsTestProject>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.7.1" />
|
||||
<PackageReference Include="Moq" Version="4.20.69" />
|
||||
<PackageReference Include="xunit" Version="2.4.2" />
|
||||
<PackageReference Include="xunit.runner.visualstudio" Version="2.4.5">
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="coverlet.collector" Version="3.2.0">
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
</PackageReference>
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\ShoppingAssistantApi.Infrastructure\ShoppingAssistantApi.Infrastructure.csproj" />
|
||||
<ProjectReference Include="..\ShoppingAssistantApi.Application\ShoppingAssistantApi.Application.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
@ -15,6 +15,8 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ShoppingAssistantApi.Api",
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ShoppingAssistantApi.Tests", "ShoppingAssistantApi.Tests\ShoppingAssistantApi.Tests.csproj", "{297B5378-79D7-406C-80A5-151C6B3EA147}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ShoppingAssistantApi.UnitTests", "ShoppingAssistantApi.UnitTests\ShoppingAssistantApi.UnitTests.csproj", "{B4EFE8F1-89F5-44E4-BD0A-4F63D09C8E6F}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
@ -45,6 +47,10 @@ Global
|
||||
{297B5378-79D7-406C-80A5-151C6B3EA147}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{297B5378-79D7-406C-80A5-151C6B3EA147}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{297B5378-79D7-406C-80A5-151C6B3EA147}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{B4EFE8F1-89F5-44E4-BD0A-4F63D09C8E6F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{B4EFE8F1-89F5-44E4-BD0A-4F63D09C8E6F}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{B4EFE8F1-89F5-44E4-BD0A-4F63D09C8E6F}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{B4EFE8F1-89F5-44E4-BD0A-4F63D09C8E6F}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
|
Loading…
Reference in New Issue
Block a user