diff --git a/README.md b/README.md index 77bb8b29..64463452 100644 --- a/README.md +++ b/README.md @@ -1,10 +1,8 @@ -# Xperience by Kentico - KX 13 E-Commerce integration +# Xperience by Kentico - Kentico Xperience 13 E-commerce -[![Kentico Labs](https://img.shields.io/badge/Kentico_Labs-grey?labelColor=orange&logo=)](https://github.com/Kentico/.github/blob/main/SUPPORT.md#labs-limited-support) +[![7-day bug-fix policy](https://img.shields.io/badge/-7--days_bug--fixing_policy-grey?labelColor=orange&logo=)](https://github.com/Kentico/.github/blob/main/SUPPORT.md#full-support) [![CI: Build and Test](https://github.com/Kentico/xperience-by-kentico-k13ecommerce/actions/workflows/ci.yml/badge.svg)](https://github.com/Kentico/xperience-by-kentico-k13ecommerce/actions/workflows/ci.yml) -**This integration is currently a Proof of Concept (PoC). For further details, please refer to the [Support](#support) section and the [Kentico Labs](https://github.com/Kentico/.github/blob/main/SUPPORT.md#labs-limited-support) tag associated with this feature.** - | Name | Package | | ------------- |:-------------:| | Kentico.Xperience.K13Ecommerce | [![NuGet Package](https://img.shields.io/nuget/v/Kentico.Xperience.K13Ecommerce.svg)](https://www.nuget.org/packages/Kentico.Xperience.K13Ecommerce) | @@ -43,8 +41,10 @@ is located in [Dancing Goat XbyK example project](./examples/DancingGoat-K13Ecom The integration provides an API with services for implementing the following scenarios: - Listing products based on parameters, product categories, prices and inventory - Actions with shopping cart, changing currency and order creation - - Listing of orders (currently suitable for implementing listing orders in administration) - - **Order updates and listing for specific customers are under development** + - Listing of orders in the administration + - Listing of orders on the live site for the current customer + - Updates of existing orders + - Retrieving and listing order statuses - Listing site cultures and currencies - Check [this part of User Guide](./docs/Usage-Guide.md#kx-13-e-commerce-integration-in-xperience-by-kentico) for more specific description @@ -169,6 +169,7 @@ public void Configure(IApplicationBuilder app, IWebHostEnvironment environment) "ClientId": "3ef7fe1b-696c-4afa-8b56-d3176b7bea95", "ClientSecret": "********************", "ProductSyncEnabled": true, + "StandaloneProductSync": true, "ProductSyncInterval": 10 } } @@ -207,8 +208,8 @@ Distributed under the MIT License. See [`LICENSE.md`](./LICENSE.md) for more inf ## Support -This contribution has __Kentico Labs limited support__. +This contribution has __Full Support__. -See [`SUPPORT.md`](https://github.com/Kentico/.github/blob/main/SUPPORT.md#labs-limited-support) for more information. +See [`SUPPORT.md`](https://github.com/Kentico/.github/blob/main/SUPPORT.md) for more information. For any security issues see [`SECURITY.md`](https://github.com/Kentico/.github/blob/main/SECURITY.md). diff --git a/docs/Usage-Guide.md b/docs/Usage-Guide.md index 53de2128..bccd9948 100644 --- a/docs/Usage-Guide.md +++ b/docs/Usage-Guide.md @@ -77,6 +77,10 @@ via API. ### Orders - Endpoint `api/store/order/list` for retrieving list of orders for current customer based on request (supports paging) - Endpoint `api/store/order/admin/list` for retrieving list of orders (for all customers) based on request (supports paging) to display in XbyK administration (supports paging) +- Endpoint `api/store/order/detail/{orderID}` for retrieving order details for the current customer. If the order belongs to another customer, no order is retrieved +- Endpoint `api/store/order/admin/detail/{orderID}` for retrieving order details (without verifying if the order belongs to the current customer) +- Endpoint `api/store/order/statuses/list` for retrieving all order statuses +- Endpoint `api/store/order/update` for updating orders (update order status, set order payment, etc.). Primarily intended to be used via `IOrderService` available in the [integration API](https://github.com/Kentico/xperience-by-kentico-k13ecommerce/pull/16#kx-13-e-commerce-integration-in-xperience-by-kentico) ### Customers - Endpoint `api/store/customer/addresses` for retrieving current customer's addresses @@ -90,7 +94,7 @@ via API. When [member](https://docs.kentico.com/x/BIsuCw) is created on XbyK (for example when a new customer registers), this member needs to be synchronized to KX 13 as a user. It is subsequently used for API authorization (member/user identity is generated in JWT). Before you start using the Store API, you need to synchronize all website members between the client (XbyK) and your KX 13 application. -Complete synchronization is not part of this PoC solution. +Complete synchronization is currently not a part of this solution. - Endpoint `api/store/synchronization/user-synchronization` can be used to create a new user in KX 13 - The client application (XbyK) should use this to ensure that all new members are synchronized to KX 13. This is necessary when client's @@ -218,8 +222,12 @@ and to browser cookie (uses `IShoppingCartClientStorage`) - Service is used e.g. in [CheckoutService in Dancing Goat example](../examples/DancingGoat-K13Ecommerce/Services/CheckoutService.cs) where customer's addresses are retrieved in cart's second step. - `IOrderService` - - List of orders - currently suitable for implementing listing orders in administration - - **Order updates and listing for specific customers are under development** + - List orders from all customers (for implementing order listings in the administration) + - List orders for the current customer (based on the request context) + - Retrieve order details for the current customer (only for orders that belong to the customer) + - Retrieve order details for administrators (without verifying if the order belongs to the current customer) + - List all order statuses + - Update existing orders (order status, payment, etc.) - `ISiteStoreService` - Use for retrieving site's [list of enabled cultures](https://github.com/Kentico/xperience-by-kentico-ecommerce/blob/main/src/Kentico.Xperience.K13Ecommerce/SiteStore/ISiteStoreService.cs#L13), e.g. for implementation of language selector - Use for retrieving site's [list of enabled currencies](https://github.com/Kentico/xperience-by-kentico-ecommerce/blob/main/src/Kentico.Xperience.K13Ecommerce/SiteStore/ISiteStoreService.cs#L18), e.g. for implementation of currency selector @@ -232,7 +240,7 @@ and to browser cookie (uses `IShoppingCartClientStorage`) Library also implements product synchronization to Content hub. These are 3 entities synchronized to reusable content items: - Products - Content type `K13Store.ProductSKU` - - All products associated with product pages are synced. **Standalone SKUs** aren't currently supported. + - All products associated with product pages are synced. **Standalone SKUs** synchronization can be set via `StandaloneProductSync` setting. - Product variants - Content type `K13Store.ProductVariant` - All products variant for parent products - Product images - Content type `K13Store.ProductImage` @@ -241,7 +249,7 @@ Library also implements product synchronization to Content hub. These are 3 enti The synchronization runs in a background thread worker periodically and can be disabled (`ProductSyncEnabled` setting). Interval can be set in minutes (`ProductSyncInterval` setting). Synchronized data is updated when source value changes, so data cannot be edited in XbyK safely, but new custom or reusable fields can be added and edited -safely. +safely. You can decide, whether include [standalone SKUs](https://docs.kentico.com/x/3gqRBg) or not (`StandaloneProductSync` setting). You can select content item folders where content items are synchronized. Content item folders can be selected independently for each content type in XbyK administration UI. Go to **Configuration** -> **K13Ecommerce** -> **K13Ecommerce settings**. Content items are not moved if root folder is selected. @@ -302,6 +310,7 @@ dotnet add package Kentico.Xperience.Store.Rcl "ClientId": "3ef7fe1b-696c-4afa-8b56-d3176b7bea95", "ClientSecret": "********************", "ProductSyncEnabled": true, + "StandaloneProductSync": true, "ProductSyncInterval": 10 } } @@ -314,6 +323,7 @@ dotnet add package Kentico.Xperience.Store.Rcl | ClientId | Fill same value which is defined on KX 13 side | | ClientSecret | Fill same value which is defined on KX 13 side | | ProductSyncEnabled | If true, product synchronization is enabled | +| StandaloneProductSync | If this setting along with `ProductSyncEnabled` is true, [standalone SKUs](https://docs.kentico.com/x/3gqRBg) are synchronized as well (if `ProductSyncEnabled` is false, no products are synchronized). | | ProductSyncInterval | Interval in minutes specifies how often synchronization is running | @@ -420,7 +430,7 @@ Here are links for some specific parts of shopping cart: - [Discount / Coupon codes](https://github.com/Kentico/xperience-by-kentico-ecommerce/blob/main/examples/DancingGoat-K13Ecommerce/Controllers/KStore/CheckoutController.cs#L163) - [Delivery details + shipping](https://github.com/Kentico/xperience-by-kentico-ecommerce/blob/main/examples/DancingGoat-K13Ecommerce/Controllers/KStore/CheckoutController.cs#L194) - [Payment](https://github.com/Kentico/xperience-by-kentico-ecommerce/blob/main/examples/DancingGoat-K13Ecommerce/Controllers/KStore/CheckoutController.cs#L330) -- Payment gateway - Is not part of this PoC solution, you need to implement integration with specific payment gateway. **API for updating orders (and their statuses) is under development**. +- Payment gateway - Is not currently part of the solution. You need to implement integration with a specific payment gateway. - [Order creation](https://github.com/Kentico/xperience-by-kentico-ecommerce/blob/main/examples/DancingGoat-K13Ecommerce/Controllers/KStore/CheckoutController.cs#L315) ### How to enable automatic product page synchronization? @@ -473,4 +483,9 @@ mapped e.g. in another folder or website channel: #### Known limitations Avoid creating rules that map the NodeAliasPath of different content items to a single product page tree path. Such rules cause the linked content item to be overwritten for particular pages. -If you change existing mapping rules, already created pages will not be moved accordingly. Instead, they will be created in the new location. \ No newline at end of file +If you change existing mapping rules, already created pages will not be moved accordingly. Instead, they will be created in the new location. + +### How to handle order payments? +1. Implement your own payment method. +2. Retrieve all order statuses using `IOrderService` if needed. +3. Use `UpdateOrder` method of `IOrderService` to update order status and to set `OrderIsPaid` flag according to the payment result. \ No newline at end of file diff --git a/examples/DancingGoat-K13Ecommerce/Controllers/TestController.cs b/examples/DancingGoat-K13Ecommerce/Controllers/TestController.cs index de833ceb..3e4ca008 100644 --- a/examples/DancingGoat-K13Ecommerce/Controllers/TestController.cs +++ b/examples/DancingGoat-K13Ecommerce/Controllers/TestController.cs @@ -29,5 +29,43 @@ public async Task TestSetCurrency(string currencyCode) public async Task TestOrders([FromServices] IOrderService orderService) => Json(await orderService.GetCurrentCustomerOrderList(new OrderListRequest { Page = 1, PageSize = 10, OrderBy = "OrderID DESC" })); + + public async Task TestOrderStatuses([FromServices] IOrderService orderService) + { + var statuses = await orderService.GetOrderStatuses(); + return Json(statuses); + } + + public async Task TestUpdateOrder([FromServices] IOrderService orderService) + { + var orders = + await orderService.GetAdminOrderList(new OrderListRequest { Page = 1, PageSize = 10, OrderBy = "OrderID DESC" }); + + var order = orders.Orders.First(o => o.OrderId == 25); + + order.OrderGrandTotal = 999; + order.OrderIsPaid = true; + var newStatus = (await orderService.GetOrderStatuses()).First(o => o.StatusName == "Completed"); + order.OrderStatus = newStatus; + + order.OrderBillingAddress.AddressLine1 = "123 Main St"; + order.OrderBillingAddress.AddressCity = "New York"; + + order.OrderShippingAddress.AddressZip = "10001"; + + order.OrderShippingOption = new KShippingOption { ShippingOptionId = 2 }; + order.OrderPaymentOption = new KPaymentOption { PaymentOptionId = 1 }; + + order.OrderPaymentResult = new KPaymentResult + { + PaymentIsCompleted = true, + PaymentMethodName = "Test", + PaymentStatusName = "Test status" + }; + + await orderService.UpdateOrder(order); + + return Ok(); + } } #endif diff --git a/examples/DancingGoat-K13Ecommerce/appsettings.json b/examples/DancingGoat-K13Ecommerce/appsettings.json index e97cc90b..8ffed5c8 100644 --- a/examples/DancingGoat-K13Ecommerce/appsettings.json +++ b/examples/DancingGoat-K13Ecommerce/appsettings.json @@ -19,6 +19,7 @@ "ClientId": "YourUniqueClientIdentifier", "ClientSecret": "********************", "ProductSyncEnabled": true, + "StandaloneProductSync": true, "ProductSyncInterval": 10 } } \ No newline at end of file diff --git a/examples/Kentico13_DancingGoatStore/ShopApi/CustomSKUConverter.cs b/examples/Kentico13_DancingGoatStore/ShopApi/CustomSKUConverter.cs index 33140a37..5dcf28ff 100644 --- a/examples/Kentico13_DancingGoatStore/ShopApi/CustomSKUConverter.cs +++ b/examples/Kentico13_DancingGoatStore/ShopApi/CustomSKUConverter.cs @@ -10,9 +10,9 @@ namespace DancingGoat.ShopApi; /// public class CustomSKUConverter : ProductSKUConverter { - public override CustomSKU Convert(SKUInfo skuInfo, string currencyCode, bool withVariants) + public override CustomSKU Convert(SKUInfo skuInfo, string currencyCode, bool withVariants, bool withLongDescription) { - var model = base.Convert(skuInfo, currencyCode, withVariants); + var model = base.Convert(skuInfo, currencyCode, withVariants, withLongDescription); return model; } diff --git a/src/Kentico.Xperience.K13Ecommerce/Config/KenticoStoreConfig.cs b/src/Kentico.Xperience.K13Ecommerce/Config/KenticoStoreConfig.cs index 6d99c2f9..12a822ab 100644 --- a/src/Kentico.Xperience.K13Ecommerce/Config/KenticoStoreConfig.cs +++ b/src/Kentico.Xperience.K13Ecommerce/Config/KenticoStoreConfig.cs @@ -20,6 +20,11 @@ public class KenticoStoreConfig /// public required string ClientSecret { get; set; } = string.Empty; + /// + /// Synchronize also standalone products without page representation. + /// + public required bool StandaloneProductSync { get; set; } = true; + /// /// When true, product synchronization is enabled. /// diff --git a/src/Kentico.Xperience.K13Ecommerce/Orders/IOrderService.cs b/src/Kentico.Xperience.K13Ecommerce/Orders/IOrderService.cs index fd397946..f87dcb53 100644 --- a/src/Kentico.Xperience.K13Ecommerce/Orders/IOrderService.cs +++ b/src/Kentico.Xperience.K13Ecommerce/Orders/IOrderService.cs @@ -21,4 +21,31 @@ public interface IOrderService /// Request parameters for order listing. /// Paged list of orders. Task GetAdminOrderList(OrderListRequest request); + + + /// + /// Get order by id for current customer. + /// + /// Order ID. + Task GetCurrentCustomerOrder(int orderId); + + + /// + /// Get order by ID to display in XbyK administration. + /// + /// Order ID. + Task GetAdminOrder(int orderId); + + + /// + /// Returns list of enabled order statuses. + /// + Task> GetOrderStatuses(); + + + /// + /// Updates order. + /// + /// Order Dto, send full data for order - retrieve order data first. + Task UpdateOrder(KOrder order); } diff --git a/src/Kentico.Xperience.K13Ecommerce/Orders/OrderListRequest.cs b/src/Kentico.Xperience.K13Ecommerce/Orders/OrderListRequest.cs index 5b17d59d..565e8ed7 100644 --- a/src/Kentico.Xperience.K13Ecommerce/Orders/OrderListRequest.cs +++ b/src/Kentico.Xperience.K13Ecommerce/Orders/OrderListRequest.cs @@ -19,4 +19,9 @@ public class OrderListRequest /// Order by. /// public required string OrderBy { get; set; } + + /// + /// Customer ID, leave null or zero for all customers. + /// + public int? CustomerId { get; set; } } diff --git a/src/Kentico.Xperience.K13Ecommerce/Orders/OrderService.cs b/src/Kentico.Xperience.K13Ecommerce/Orders/OrderService.cs index 4680615e..b8f28e2d 100644 --- a/src/Kentico.Xperience.K13Ecommerce/Orders/OrderService.cs +++ b/src/Kentico.Xperience.K13Ecommerce/Orders/OrderService.cs @@ -12,4 +12,20 @@ public async Task GetCurrentCustomerOrderList(OrderListReques /// public async Task GetAdminOrderList(OrderListRequest request) => await storeApiClient.AdminOrderListAsync(request.Page, request.PageSize, request.OrderBy); + + + /// + public async Task GetCurrentCustomerOrder(int orderId) => await storeApiClient.OrderDetailAsync(orderId); + + + /// + public async Task GetAdminOrder(int orderId) => await storeApiClient.AdminOrderDetailAsync(orderId); + + + /// + public async Task> GetOrderStatuses() => await storeApiClient.OrderStatusesListAsync(); + + + /// + public async Task UpdateOrder(KOrder order) => await storeApiClient.UpdateOrderAsync(order); } diff --git a/src/Kentico.Xperience.K13Ecommerce/Products/IProductService.cs b/src/Kentico.Xperience.K13Ecommerce/Products/IProductService.cs index 49d49537..479f2d7b 100644 --- a/src/Kentico.Xperience.K13Ecommerce/Products/IProductService.cs +++ b/src/Kentico.Xperience.K13Ecommerce/Products/IProductService.cs @@ -36,4 +36,12 @@ public interface IProductService /// SKUID for product or variant. /// Currency code. Task GetVariantInventoryPriceInfo(int skuId, string? currencyCode = null); + + + /// + /// Gets all standalone products. + /// + /// Stanadlone product params. + /// + Task> GetStandaloneProducts(ProductRequest request); } diff --git a/src/Kentico.Xperience.K13Ecommerce/Products/ProductPageRequest.cs b/src/Kentico.Xperience.K13Ecommerce/Products/ProductPageRequest.cs index a0d35390..dff2db93 100644 --- a/src/Kentico.Xperience.K13Ecommerce/Products/ProductPageRequest.cs +++ b/src/Kentico.Xperience.K13Ecommerce/Products/ProductPageRequest.cs @@ -3,7 +3,7 @@ /// /// Model for request to filter product pages /// -public class ProductPageRequest +public class ProductPageRequest : ProductRequest { /// /// Node alias path prefix @@ -11,36 +11,6 @@ public class ProductPageRequest public required string Path { get; set; } - /// - /// Document culture - /// - public string? Culture { get; set; } - - - /// - /// Product currency - /// - public string? Currency { get; set; } - - - /// - /// Order by SQL expression - /// - public string? OrderBy { get; set; } - - - /// - /// Limit how many products to return. - /// - public int? Limit { get; set; } - - - /// - /// If true variants are loaded too for products with variants (default false). - /// - public bool WithVariants { get; set; } - - /// /// If true, DocumentSKUDescription is filled too (default false). /// diff --git a/src/Kentico.Xperience.K13Ecommerce/Products/ProductRequest.cs b/src/Kentico.Xperience.K13Ecommerce/Products/ProductRequest.cs new file mode 100644 index 00000000..1758d47a --- /dev/null +++ b/src/Kentico.Xperience.K13Ecommerce/Products/ProductRequest.cs @@ -0,0 +1,39 @@ +using System.ComponentModel.DataAnnotations; + +namespace Kentico.Xperience.K13Ecommerce.Products; + +/// +/// Model for request to filter standalone products +/// +public class ProductRequest +{ + /// + /// To determine if product has its page, culture needs to be provided + /// so documents in the specific culture will be checked. + /// + [RegularExpression("[a-zA-Z]{2}-[a-zA-Z]{2}")] + public string? Culture { get; set; } + + /// + /// Product currency. + /// + public string? Currency { get; set; } + + /// + /// Order by SQL expression. + /// + public string? OrderBy { get; set; } + + /// + /// Limit how many products to return. + /// + [DefaultValue(12)] + [Range(1, 1000)] + public int Limit { get; set; } + + /// + /// If true variants are loaded too for products with variants (default false). + /// + public bool WithVariants { get; set; } +} + diff --git a/src/Kentico.Xperience.K13Ecommerce/Products/ProductService.cs b/src/Kentico.Xperience.K13Ecommerce/Products/ProductService.cs index fc7ce94b..776b4d8e 100644 --- a/src/Kentico.Xperience.K13Ecommerce/Products/ProductService.cs +++ b/src/Kentico.Xperience.K13Ecommerce/Products/ProductService.cs @@ -6,8 +6,8 @@ internal class ProductService(IKenticoStoreApiClient storeApiClient) : IProductS { /// public async Task> GetProductPages(ProductPageRequest request) - => await storeApiClient.GetProductPagesAsync(request.Path, request.Culture, request.Currency, request.OrderBy, - request.Limit, request.WithVariants, request.WithLongDescription, request.NoLinks); + => await storeApiClient.GetProductPagesAsync(request.Path, request.WithLongDescription, request.NoLinks, request.Culture, + request.Currency, request.OrderBy, request.Limit, request.WithVariants); /// @@ -24,4 +24,8 @@ public async Task GetProductPrices(int skuId, string? cur /// public async Task GetVariantInventoryPriceInfo(int skuId, string? currencyCode = null) => await storeApiClient.GetInventoryPricesAsync(skuId, currencyCode); + + /// + public async Task> GetStandaloneProducts(ProductRequest request) => + await storeApiClient.GetStandaloneProductsAsync(request.Culture, request.Currency, request.OrderBy, request.Limit, request.WithVariants); } diff --git a/src/Kentico.Xperience.K13Ecommerce/StoreApi/StoreItemIdentifiers.cs b/src/Kentico.Xperience.K13Ecommerce/StoreApi/StoreItemIdentifiers.cs index 1263c1f3..246ddb3a 100644 --- a/src/Kentico.Xperience.K13Ecommerce/StoreApi/StoreItemIdentifiers.cs +++ b/src/Kentico.Xperience.K13Ecommerce/StoreApi/StoreItemIdentifiers.cs @@ -9,6 +9,20 @@ public partial class KProductNode : IItemIdentifier { /// public int ExternalId => Sku.Skuid; + + /// + /// Create product node from product SKU. + /// + /// + public KProductNode(KProductSKU sku) + { + Sku = sku; + DocumentSKUName = sku.SkuName; + DocumentSKUDescription = sku.SkuLongDescription; + DocumentSKUShortDescription = sku.SkuShortDescription; + ClassName = string.Empty; + NodeAliasPath = string.Empty; + } } /// diff --git a/src/Kentico.Xperience.K13Ecommerce/StoreApi/swagger.json b/src/Kentico.Xperience.K13Ecommerce/StoreApi/swagger.json index ba166abd..b24af5c4 100644 --- a/src/Kentico.Xperience.K13Ecommerce/StoreApi/swagger.json +++ b/src/Kentico.Xperience.K13Ecommerce/StoreApi/swagger.json @@ -243,6 +243,14 @@ "schema": { "type": "string" } + }, + { + "name": "CustomerId", + "in": "query", + "schema": { + "type": "integer", + "format": "int32" + } } ], "responses": { @@ -301,6 +309,14 @@ "schema": { "type": "string" } + }, + { + "name": "CustomerId", + "in": "query", + "schema": { + "type": "integer", + "format": "int32" + } } ], "responses": { @@ -327,6 +343,170 @@ } } }, + "/api/store/order/detail/{orderId}": { + "get": { + "tags": [ + "Order" + ], + "summary": "Returns order by ID only if order belongs to current customer.", + "operationId": "OrderDetail", + "parameters": [ + { + "name": "orderId", + "in": "path", + "description": "Order ID.", + "required": true, + "schema": { + "type": "integer", + "format": "int32" + } + } + ], + "responses": { + "200": { + "description": "Success", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/KOrder" + } + } + } + }, + "401": { + "description": "Unauthorized", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ProblemDetails" + } + } + } + } + } + } + }, + "/api/store/order/admin/detail/{orderId}": { + "get": { + "tags": [ + "Order" + ], + "summary": "Returns order by ID.", + "operationId": "AdminOrderDetail", + "parameters": [ + { + "name": "orderId", + "in": "path", + "description": "Order ID.", + "required": true, + "schema": { + "type": "integer", + "format": "int32" + } + } + ], + "responses": { + "200": { + "description": "Success", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/KOrder" + } + } + } + }, + "401": { + "description": "Unauthorized", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ProblemDetails" + } + } + } + } + } + } + }, + "/api/store/order/statuses/list": { + "get": { + "tags": [ + "Order" + ], + "summary": "Endpoint for listing order statuses.", + "operationId": "OrderStatusesList", + "responses": { + "200": { + "description": "Success", + "content": { + "application/json": { + "schema": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KOrderStatus" + } + } + } + } + }, + "401": { + "description": "Unauthorized", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ProblemDetails" + } + } + } + } + } + } + }, + "/api/store/order/update": { + "put": { + "tags": [ + "Order" + ], + "summary": "Updates order. Updates all fields on order level and addresses on sub-level. Customer data and order items cannot be updated.", + "operationId": "UpdateOrder", + "requestBody": { + "description": "Order dto", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/KOrder" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/KOrder" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/KOrder" + } + } + } + }, + "responses": { + "200": { + "description": "Success" + }, + "401": { + "description": "Unauthorized", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ProblemDetails" + } + } + } + } + } + } + }, "/api/store/products/test": { "get": { "tags": [ @@ -367,10 +547,26 @@ "type": "string" } }, + { + "name": "WithLongDescription", + "in": "query", + "description": "If true, DocumentSKUDescription is filled too (default false).", + "schema": { + "type": "boolean" + } + }, + { + "name": "NoLinks", + "in": "query", + "description": "If true, only not-linked product pages are returned (default false).", + "schema": { + "type": "boolean" + } + }, { "name": "Culture", "in": "query", - "description": "Document culture.", + "description": "To determine if product has its page, culture needs to be provided\r\nso documents in the specific culture will be checked.", "schema": { "pattern": "[a-zA-Z]{2}-[a-zA-Z]{2}", "type": "string" @@ -411,22 +607,6 @@ "schema": { "type": "boolean" } - }, - { - "name": "WithLongDescription", - "in": "query", - "description": "If true, DocumentSKUDescription is filled too (default false).", - "schema": { - "type": "boolean" - } - }, - { - "name": "NoLinks", - "in": "query", - "description": "If true, only not-linked product pages are returned (default false).", - "schema": { - "type": "boolean" - } } ], "responses": { @@ -659,6 +839,87 @@ } } }, + "/api/store/products/standalone-products": { + "get": { + "tags": [ + "Products" + ], + "summary": "Returns all standalone products.", + "operationId": "GetStandaloneProducts", + "parameters": [ + { + "name": "Culture", + "in": "query", + "description": "To determine if product has its page, culture needs to be provided\r\nso documents in the specific culture will be checked.", + "schema": { + "pattern": "[a-zA-Z]{2}-[a-zA-Z]{2}", + "type": "string" + } + }, + { + "name": "Currency", + "in": "query", + "description": "Product currency.", + "schema": { + "type": "string" + } + }, + { + "name": "OrderBy", + "in": "query", + "description": "Order by SQL expression.", + "schema": { + "type": "string" + } + }, + { + "name": "Limit", + "in": "query", + "description": "Limit how many products to return.", + "schema": { + "maximum": 1000, + "minimum": 1, + "type": "integer", + "format": "int32", + "default": 12 + } + }, + { + "name": "WithVariants", + "in": "query", + "description": "If true variants are loaded too for products with variants (default false).", + "schema": { + "type": "boolean" + } + } + ], + "responses": { + "200": { + "description": "Success", + "content": { + "application/json": { + "schema": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KProductSKU" + } + } + } + } + }, + "400": { + "description": "Bad Request", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ProblemDetails" + } + } + } + } + } + } + }, "/api/store/cart/content": { "get": { "tags": [ @@ -2279,6 +2540,10 @@ "KCurrency": { "type": "object", "properties": { + "currencyId": { + "type": "integer", + "format": "int32" + }, "currencyCode": { "type": "string", "nullable": true @@ -2289,7 +2554,7 @@ } }, "additionalProperties": false, - "description": "Represents currency from Store configuration." + "description": "Represents currency from Store configuration. CMS.Ecommerce.CurrencyInfo" }, "KCustomer": { "required": [ @@ -2399,6 +2664,12 @@ "description": "Dto for CMS.Ecommerce.ManufacturerInfo." }, "KOrder": { + "required": [ + "orderGrandTotal", + "orderId", + "orderTotalPrice", + "orderTotalTax" + ], "type": "object", "properties": { "orderId": { @@ -2417,9 +2688,8 @@ "type": "string", "nullable": true }, - "orderCurrencyCode": { - "type": "string", - "nullable": true + "orderCurrency": { + "$ref": "#/components/schemas/KCurrency" }, "orderShippingOption": { "$ref": "#/components/schemas/KShippingOption" @@ -2518,6 +2788,9 @@ "$ref": "#/components/schemas/KOrderItem" }, "nullable": true + }, + "orderPaymentResult": { + "$ref": "#/components/schemas/KPaymentResult" } }, "additionalProperties": false, @@ -2631,6 +2904,57 @@ "additionalProperties": false, "description": "Dto for CMS.Ecommerce.PaymentOptionInfo." }, + "KPaymentResult": { + "type": "object", + "properties": { + "paymentDate": { + "type": "string", + "nullable": true + }, + "paymentIsCompleted": { + "type": "boolean" + }, + "paymentIsFailed": { + "type": "boolean" + }, + "paymentIsAuthorized": { + "type": "boolean" + }, + "paymentDescription": { + "type": "string", + "nullable": true + }, + "paymentTransactionId": { + "type": "string", + "nullable": true + }, + "paymentAuthorizationID": { + "type": "string", + "nullable": true + }, + "paymentMethodName": { + "type": "string", + "nullable": true + }, + "paymentMethodID": { + "type": "integer", + "format": "int32" + }, + "paymentStatusValue": { + "type": "string", + "nullable": true + }, + "paymentStatusName": { + "type": "string", + "nullable": true + }, + "paymentApprovalUrl": { + "type": "string", + "nullable": true + } + }, + "additionalProperties": false + }, "KProductCatalogPrices": { "type": "object", "properties": { @@ -2751,6 +3075,10 @@ "type": "string", "nullable": true }, + "skuLongDescription": { + "type": "string", + "nullable": true + }, "skuNumber": { "type": "string", "nullable": true @@ -3311,7 +3639,7 @@ "nullable": true } }, - "additionalProperties": { } + "additionalProperties": {} }, "ProductInventoryPriceInfo": { "type": "object", @@ -3418,7 +3746,7 @@ }, "security": [ { - "Bearer": [ ] + "Bearer": [] } ] } \ No newline at end of file diff --git a/src/Kentico.Xperience.K13Ecommerce/Synchronization/Products/ProductSynchronizationItem.cs b/src/Kentico.Xperience.K13Ecommerce/Synchronization/Products/ProductSynchronizationItem.cs index 526ba7b3..1c8de2f5 100644 --- a/src/Kentico.Xperience.K13Ecommerce/Synchronization/Products/ProductSynchronizationItem.cs +++ b/src/Kentico.Xperience.K13Ecommerce/Synchronization/Products/ProductSynchronizationItem.cs @@ -90,6 +90,7 @@ public bool GetModifiedProperties(ProductSKU contentItem, out Dictionary config) : SynchronizationServiceCommon(httpClientFactory), IProductSynchronizationService { public async Task SynchronizeProducts() { @@ -33,7 +36,7 @@ public async Task SynchronizeProducts() string language = defaultCultureCode[..2]; var kenticoStoreProducts = - await productService.GetProductPages(new ProductPageRequest + (await productService.GetProductPages(new ProductPageRequest { Path = "/", Culture = defaultCultureCode, @@ -41,7 +44,20 @@ await productService.GetProductPages(new ProductPageRequest WithVariants = true, WithLongDescription = true, NoLinks = true - }); + })).ToList(); + + if (config.CurrentValue.StandaloneProductSync) + { + var kenticoStandaloneProducts = (await productService.GetStandaloneProducts(new ProductRequest + { + Culture = defaultCultureCode, + Limit = 1000, + WithVariants = true + })).Select(p => new KProductNode(p)); + + kenticoStoreProducts.AddRange(kenticoStandaloneProducts); + } + var contentItemProducts = await contentItemService.GetContentItems(ProductSKU.CONTENT_TYPE_NAME, linkedItemsLevel: 2); diff --git a/src/Kentico.Xperience.StoreApi/Currencies/KCurrency.cs b/src/Kentico.Xperience.StoreApi/Currencies/KCurrency.cs index f261bb27..d8b0f213 100644 --- a/src/Kentico.Xperience.StoreApi/Currencies/KCurrency.cs +++ b/src/Kentico.Xperience.StoreApi/Currencies/KCurrency.cs @@ -1,10 +1,11 @@ namespace Kentico.Xperience.StoreApi.Currencies; /// -/// Represents currency from Store configuration. +/// Represents currency from Store configuration. /// public class KCurrency { + public int CurrencyId { get; set; } public string CurrencyCode { get; set; } public string CurrencyFormatString { get; set; } } diff --git a/src/Kentico.Xperience.StoreApi/Mapping/StoreApiMappingProfile.cs b/src/Kentico.Xperience.StoreApi/Mapping/StoreApiMappingProfile.cs index bb479123..04d3ffbd 100644 --- a/src/Kentico.Xperience.StoreApi/Mapping/StoreApiMappingProfile.cs +++ b/src/Kentico.Xperience.StoreApi/Mapping/StoreApiMappingProfile.cs @@ -56,21 +56,34 @@ public StoreApiMappingProfile() CreateMap() .AfterMap((source, dest, ctx) => { - dest.OrderCurrencyCode = CurrencyInfoProvider.ProviderObject.Get(source.OrderCurrencyID)?.CurrencyCode; + dest.OrderCurrency = + ctx.Mapper.Map(CurrencyInfoProvider.ProviderObject.Get(source.OrderCurrencyID)); dest.OrderCustomer = ctx.Mapper.Map(CustomerInfoProvider.ProviderObject.Get(source.OrderCustomerID)); dest.OrderShippingOption = ctx.Mapper.Map(ShippingOptionInfo.Provider.Get(source.OrderShippingOptionID)); - dest.OrderPaymentOption = ctx.Mapper.Map(PaymentOptionInfo.Provider.Get(source.OrderPaymentOptionID)); - dest.OrderItems = ctx.Mapper.Map>(OrderItemInfoProvider.GetOrderItems(source.OrderID)); - dest.OrderStatus = ctx.Mapper.Map(OrderStatusInfoProvider.ProviderObject.Get(source.OrderStatusID)); - }); + dest.OrderPaymentOption = + ctx.Mapper.Map(PaymentOptionInfo.Provider.Get(source.OrderPaymentOptionID)); + dest.OrderItems = + ctx.Mapper.Map>(OrderItemInfoProvider.GetOrderItems(source.OrderID)); + dest.OrderStatus = + ctx.Mapper.Map(OrderStatusInfoProvider.ProviderObject.Get(source.OrderStatusID)); + }) + .ReverseMap() + .ForPath(s => s.OrderCurrencyID, m => m.MapFrom(d => d.OrderCurrency.CurrencyId)) + .ForPath(s => s.OrderCustomerID, m => m.MapFrom(d => d.OrderCustomer.CustomerId)) + .ForPath(s => s.OrderShippingOptionID, m => m.MapFrom(d => d.OrderShippingOption.ShippingOptionId)) + .ForPath(s => s.OrderPaymentOptionID, m => m.MapFrom(d => d.OrderPaymentOption.PaymentOptionId)) + .ForPath(s => s.OrderStatusID, m => m.MapFrom(d => d.OrderStatus.StatusId)) + .ForPath(s => s.OrderBillingAddress, m => m.MapFrom(d => d.OrderBillingAddress)) + .ForPath(s => s.OrderShippingAddress, m => m.MapFrom(d => d.OrderShippingAddress)); CreateMap(); - CreateMap(); + CreateMap().ReverseMap(); CreateMap(); CreateMap>().ConvertUsing(); CreateMap(); + CreateMap().ReverseMap(); CreateMap() .AfterMap((source, dest, ctx) => diff --git a/src/Kentico.Xperience.StoreApi/Orders/KOrder.cs b/src/Kentico.Xperience.StoreApi/Orders/KOrder.cs index 4d10a332..901daf44 100644 --- a/src/Kentico.Xperience.StoreApi/Orders/KOrder.cs +++ b/src/Kentico.Xperience.StoreApi/Orders/KOrder.cs @@ -1,22 +1,25 @@ -using Kentico.Xperience.StoreApi.Customers; +using System.ComponentModel.DataAnnotations; + +using Kentico.Xperience.StoreApi.Currencies; +using Kentico.Xperience.StoreApi.Customers; using Kentico.Xperience.StoreApi.ShoppingCart; namespace Kentico.Xperience.StoreApi.Orders; /// -/// Dto for . +/// Dto for . /// public class KOrder { - public int OrderId { get; set; } + [Required] public int OrderId { get; set; } - public decimal OrderTotalTax { get; set; } + [Required] public decimal OrderTotalTax { get; set; } public string OrderTaxSummary { get; set; } public string OrderInvoiceNumber { get; set; } - public string OrderCurrencyCode { get; set; } + public KCurrency OrderCurrency { get; set; } public KShippingOption OrderShippingOption { get; set; } @@ -28,11 +31,11 @@ public class KOrder public KOrderStatus OrderStatus { get; set; } - public decimal OrderGrandTotal { get; set; } + [Required] public decimal OrderGrandTotal { get; set; } public decimal OrderGrandTotalInMainCurrency { get; set; } - public decimal OrderTotalPrice { get; set; } + [Required] public decimal OrderTotalPrice { get; set; } public decimal OrderTotalPriceInMainCurrency { get; set; } @@ -67,4 +70,6 @@ public class KOrder public KAddress OrderCompanyAddress { get; set; } public IEnumerable OrderItems { get; set; } + + public KPaymentResult OrderPaymentResult { get; set; } } diff --git a/src/Kentico.Xperience.StoreApi/Orders/KPaymentResult.cs b/src/Kentico.Xperience.StoreApi/Orders/KPaymentResult.cs new file mode 100644 index 00000000..e3ff9e58 --- /dev/null +++ b/src/Kentico.Xperience.StoreApi/Orders/KPaymentResult.cs @@ -0,0 +1,28 @@ +namespace Kentico.Xperience.StoreApi.Orders; + +public class KPaymentResult +{ + public string PaymentDate { get; set; } + + public bool PaymentIsCompleted { get; set; } + + public bool PaymentIsFailed { get; set; } + + public bool PaymentIsAuthorized { get; set; } + + public string PaymentDescription { get; set; } + + public string PaymentTransactionId { get; set; } + + public string PaymentAuthorizationID { get; set; } + + public string PaymentMethodName { get; set; } + + public int PaymentMethodID { get; set; } + + public string PaymentStatusValue { get; set; } + + public string PaymentStatusName { get; set; } + + public string PaymentApprovalUrl { get; set; } +} diff --git a/src/Kentico.Xperience.StoreApi/Orders/OrderController.cs b/src/Kentico.Xperience.StoreApi/Orders/OrderController.cs index 2a0b78d2..d1199fb9 100644 --- a/src/Kentico.Xperience.StoreApi/Orders/OrderController.cs +++ b/src/Kentico.Xperience.StoreApi/Orders/OrderController.cs @@ -25,17 +25,20 @@ public class OrderController : ControllerBase private readonly IMapper mapper; private readonly ISiteService siteService; private readonly IShoppingService shoppingService; + private readonly IOrderStatusInfoProvider orderStatusInfoProvider; public OrderController( IOrderInfoProvider orderInfoProvider, IMapper mapper, ISiteService siteService, - IShoppingService shoppingService) + IShoppingService shoppingService, + IOrderStatusInfoProvider orderStatusInfoProvider) { this.orderInfoProvider = orderInfoProvider; this.mapper = mapper; this.siteService = siteService; this.shoppingService = shoppingService; + this.orderStatusInfoProvider = orderStatusInfoProvider; } /// @@ -83,7 +86,7 @@ public async Task> CurrentCustomerOrderList([Fro /// /// Endpoint for getting list of orders based on request to display in XbyK administration. /// - /// + /// Order list request. /// [HttpGet("admin/list", Name = nameof(AdminOrderList))] public async Task> AdminOrderList([FromQuery] OrderListRequest request) @@ -93,11 +96,17 @@ public async Task> AdminOrderList([FromQuery] Or { request.OrderBy = $"{nameof(OrderInfo.OrderDate)} DESC"; } + var orderQuery = orderInfoProvider.Get() .OnSite(siteService.CurrentSite.SiteID) .Page(page, request.PageSize) .OrderBy(request.OrderBy); + if (request.CustomerId is > 0) + { + orderQuery = orderQuery.WhereEquals(nameof(OrderInfo.OrderCustomerID), request.CustomerId); + } + var orders = mapper.Map>(await orderQuery.GetEnumerableTypedResultAsync()); return Ok(new OrderListResponse @@ -107,4 +116,94 @@ public async Task> AdminOrderList([FromQuery] Or MaxPage = (orderQuery.TotalRecords / request.PageSize) + 1 }); } + + + /// + /// Returns order by ID only if order belongs to current customer. + /// + /// Order ID. + [HttpGet("detail/{orderId:int}", Name = nameof(OrderDetail))] + public async Task> OrderDetail([FromRoute] int orderId) + { + var customer = shoppingService.GetCurrentCustomer(); + if (customer == null) + { + return NotFound(); + } + + var order = await orderInfoProvider.GetAsync(orderId); + + if (order == null || order.OrderCustomerID != customer.CustomerID) + { + return NotFound(); + } + + return Ok(mapper.Map(order)); + } + + + /// + /// Returns order by ID. + /// + /// Order ID. + [HttpGet("admin/detail/{orderId:int}", Name = nameof(AdminOrderDetail))] + public async Task> AdminOrderDetail([FromRoute] int orderId) + { + var order = await orderInfoProvider.GetAsync(orderId); + + if (order == null) + { + return NotFound(); + } + + return Ok(mapper.Map(order)); + } + + + /// + /// Endpoint for listing order statuses. + /// + /// List of order statuses. + [HttpGet("statuses/list", Name = nameof(OrderStatusesList))] + public async Task>> OrderStatusesList() + { + int siteId = ECommerceHelper.GetSiteID(siteService.CurrentSite.SiteID, "CMSStoreUseGlobalOrderStatus"); + + var orderStatuses = await orderStatusInfoProvider.Get() + .OnSite(siteId, includeGlobal: siteId == 0) + .OrderBy(nameof(OrderStatusInfo.StatusOrder)) + .GetEnumerableTypedResultAsync(); + + return Ok(mapper.Map>(orderStatuses)); + } + + /// + /// Updates order. Updates all fields on order level and addresses on sub-level. Customer data and order items cannot be updated. + /// + /// Order dto + [HttpPut("update", Name = nameof(UpdateOrder))] + public async Task UpdateOrder([FromBody] KOrder order) + { + var orderInfo = await orderInfoProvider.GetAsync(order.OrderId); + if (orderInfo == null) + { + return NotFound(); + } + + orderInfo = mapper.Map(order, orderInfo); + + if (orderInfo.OrderBillingAddress.HasChanged) + { + orderInfo.OrderBillingAddress.Update(); + } + + if (orderInfo.OrderShippingAddress.HasChanged) + { + orderInfo.OrderShippingAddress.Update(); + } + + orderInfoProvider.Set(orderInfo); + + return Ok(); + } } diff --git a/src/Kentico.Xperience.StoreApi/Orders/OrderListRequest.cs b/src/Kentico.Xperience.StoreApi/Orders/OrderListRequest.cs index 795c59f2..bdcb2e2a 100644 --- a/src/Kentico.Xperience.StoreApi/Orders/OrderListRequest.cs +++ b/src/Kentico.Xperience.StoreApi/Orders/OrderListRequest.cs @@ -13,4 +13,6 @@ public class OrderListRequest public int PageSize { get; set; } public string OrderBy { get; set; } + + public int? CustomerId { get; set; } } diff --git a/src/Kentico.Xperience.StoreApi/Products/IKProductService.cs b/src/Kentico.Xperience.StoreApi/Products/IKProductService.cs index 41fc8f5e..5a6c2643 100644 --- a/src/Kentico.Xperience.StoreApi/Products/IKProductService.cs +++ b/src/Kentico.Xperience.StoreApi/Products/IKProductService.cs @@ -1,6 +1,7 @@ using Kentico.Xperience.StoreApi.Products.Categories; using Kentico.Xperience.StoreApi.Products.Pages; using Kentico.Xperience.StoreApi.Products.Prices; +using Kentico.Xperience.StoreApi.Products.SKU; namespace Kentico.Xperience.StoreApi.Products; @@ -49,4 +50,12 @@ public interface IKProductService /// Currency code. /// Task GetProductInventoryAndPrices(int skuId, string currencyCode); + + + /// + /// Get all standalone products in specific culture. + /// + /// Standalone products parameters. + /// + Task> GetStandaloneProducts(ProductRequest request); } diff --git a/src/Kentico.Xperience.StoreApi/Products/KProductService.cs b/src/Kentico.Xperience.StoreApi/Products/KProductService.cs index cfc49d78..8068498a 100644 --- a/src/Kentico.Xperience.StoreApi/Products/KProductService.cs +++ b/src/Kentico.Xperience.StoreApi/Products/KProductService.cs @@ -10,6 +10,7 @@ using Kentico.Xperience.StoreApi.Products.Categories; using Kentico.Xperience.StoreApi.Products.Pages; using Kentico.Xperience.StoreApi.Products.Prices; +using Kentico.Xperience.StoreApi.Products.SKU; namespace Kentico.Xperience.StoreApi.Products; @@ -18,6 +19,7 @@ internal class KProductService : IKProductService { private readonly IPageRetriever pageRetriever; private readonly IProductPageConverter productPageConverter; + private readonly IProductSKUConverter productSKUConverter; private readonly ISKUInfoProvider skuInfoProvider; private readonly ICatalogPriceCalculatorFactory catalogPriceCalculatorFactory; private readonly ISiteService siteService; @@ -25,12 +27,14 @@ internal class KProductService : IKProductService private readonly IShoppingService shoppingService; - public KProductService(IPageRetriever pageRetriever, IProductPageConverter productPageConverter, + public KProductService( + IPageRetriever pageRetriever, IProductPageConverter productPageConverter, IProductSKUConverter productSKUConverter, ISKUInfoProvider skuInfoProvider, ICatalogPriceCalculatorFactory catalogPriceCalculatorFactory, ISiteService siteService, IMapper mapper, IShoppingService shoppingService) { this.pageRetriever = pageRetriever; this.productPageConverter = productPageConverter; + this.productSKUConverter = productSKUConverter; this.skuInfoProvider = skuInfoProvider; this.catalogPriceCalculatorFactory = catalogPriceCalculatorFactory; this.siteService = siteService; @@ -48,10 +52,7 @@ public async Task> GetProductPages(ProductPageRequest orderBy = "DocumentSKUName ASC"; } - var productTypes = (await DataClassInfoProvider.ProviderObject.Get() - .WhereTrue(nameof(DataClassInfo.ClassIsProduct)) - .Columns(nameof(DataClassInfo.ClassName), nameof(DataClassInfo.ClassFormDefinition)) - .GetEnumerableTypedResultAsync()) + var productTypes = (await GetProductDataClasses()) .Select(p => new { p.ClassName, @@ -161,6 +162,35 @@ public async Task GetProductInventoryAndPrices(int sk } + /// + public async Task> GetStandaloneProducts(ProductRequest request) + { + var (culture, currencyCode, orderBy, limit, withVariants) = request; + + var skuInfos = skuInfoProvider.Get() + .WhereEqualsOrNull(nameof(SKUInfo.SKUOptionCategoryID), 0) + .WhereEqualsOrNull(nameof(SKUInfo.SKUParentSKUID), 0) + .TopN(limit) + .OrderBy(orderBy); + + + var query = DocumentHelper.GetDocuments() + .WhereNotNull(nameof(TreeNode.NodeSKUID)) + .Column(nameof(TreeNode.NodeSKUID)); + + if (!string.IsNullOrEmpty(culture)) + { + query = query.Culture(culture); + } + + skuInfos = skuInfos.WhereNotIn(nameof(SKUInfo.SKUID), query); + + + return (await skuInfos.GetEnumerableTypedResultAsync()) + .Select(x => productSKUConverter.Convert(x, currencyCode, withVariants, true)); + } + + private ProductPricesResponse GetProductPrices(SKUInfo sku, string currencyCode) { var calculator = catalogPriceCalculatorFactory.GetCalculator(siteService.CurrentSite.SiteID); @@ -185,4 +215,11 @@ private ProductPricesResponse GetProductPrices(SKUInfo sku, string currencyCode) return response; } + + + private async Task> GetProductDataClasses() + => await DataClassInfoProvider.ProviderObject.Get() + .WhereTrue(nameof(DataClassInfo.ClassIsProduct)) + .Columns(nameof(DataClassInfo.ClassName), nameof(DataClassInfo.ClassFormDefinition)) + .GetEnumerableTypedResultAsync(); } diff --git a/src/Kentico.Xperience.StoreApi/Products/Pages/ProductPageConverter.cs b/src/Kentico.Xperience.StoreApi/Products/Pages/ProductPageConverter.cs index 266f5307..9efdf768 100644 --- a/src/Kentico.Xperience.StoreApi/Products/Pages/ProductPageConverter.cs +++ b/src/Kentico.Xperience.StoreApi/Products/Pages/ProductPageConverter.cs @@ -30,7 +30,7 @@ public virtual TProduct Convert(SKUTreeNode skuTreeNode, IEnumerable cus { var model = mapper.Map(skuTreeNode); model.AbsoluteUrl = DocumentURLProvider.GetAbsoluteUrl(skuTreeNode); - model.SKU = skuConverter.Convert(skuTreeNode.SKU, currencyCode, withVariants); + model.SKU = skuConverter.Convert(skuTreeNode.SKU, currencyCode, withVariants, false); model.CustomFields = customFields.ToDictionary(f => f, f => skuTreeNode.GetValue(f)); if (withLongDescription) { diff --git a/src/Kentico.Xperience.StoreApi/Products/ProductPageRequest.cs b/src/Kentico.Xperience.StoreApi/Products/ProductPageRequest.cs index b26a1326..498203ee 100644 --- a/src/Kentico.Xperience.StoreApi/Products/ProductPageRequest.cs +++ b/src/Kentico.Xperience.StoreApi/Products/ProductPageRequest.cs @@ -1,14 +1,11 @@ -using System.ComponentModel; -using System.ComponentModel.DataAnnotations; - -using Kentico.Xperience.StoreApi.Currencies; +using System.ComponentModel.DataAnnotations; namespace Kentico.Xperience.StoreApi.Products; /// /// Model for product pages request used in API. /// -public class ProductPageRequest +public class ProductPageRequest : ProductRequest { /// /// Node alias path prefix. @@ -16,35 +13,6 @@ public class ProductPageRequest [Required] public string Path { get; set; } - /// - /// Document culture. - /// - [RegularExpression("[a-zA-Z]{2}-[a-zA-Z]{2}")] - public string Culture { get; set; } - - /// - /// Product currency. - /// - [CurrencyValidation] - public string Currency { get; set; } - - /// - /// Order by SQL expression. - /// - public string OrderBy { get; set; } - - /// - /// Limit how many products to return. - /// - [DefaultValue(12)] - [Range(1, 1000)] - public int Limit { get; set; } - - /// - /// If true variants are loaded too for products with variants (default false). - /// - public bool WithVariants { get; set; } - /// /// If true, DocumentSKUDescription is filled too (default false). /// diff --git a/src/Kentico.Xperience.StoreApi/Products/ProductRequest.cs b/src/Kentico.Xperience.StoreApi/Products/ProductRequest.cs new file mode 100644 index 00000000..7d4773c5 --- /dev/null +++ b/src/Kentico.Xperience.StoreApi/Products/ProductRequest.cs @@ -0,0 +1,49 @@ +using System.ComponentModel.DataAnnotations; +using System.ComponentModel; + +using Kentico.Xperience.StoreApi.Currencies; + +namespace Kentico.Xperience.StoreApi.Products; + +public class ProductRequest +{ + /// + /// To determine if product has its page, culture needs to be provided + /// so documents in the specific culture will be checked. + /// + [RegularExpression("[a-zA-Z]{2}-[a-zA-Z]{2}")] + public string Culture { get; set; } + + /// + /// Product currency. + /// + [CurrencyValidation] + public string Currency { get; set; } + + /// + /// Order by SQL expression. + /// + public string OrderBy { get; set; } + + /// + /// Limit how many products to return. + /// + [DefaultValue(12)] + [Range(1, 1000)] + public int Limit { get; set; } + + /// + /// If true variants are loaded too for products with variants (default false). + /// + public bool WithVariants { get; set; } + + public void Deconstruct(out string culture, out string currency, out string orderBy, + out int limit, out bool withVariants) + { + culture = Culture; + currency = Currency; + orderBy = OrderBy; + limit = Limit; + withVariants = WithVariants; + } +} diff --git a/src/Kentico.Xperience.StoreApi/Products/ProductsController.cs b/src/Kentico.Xperience.StoreApi/Products/ProductsController.cs index c4f32a35..e02f4ac4 100644 --- a/src/Kentico.Xperience.StoreApi/Products/ProductsController.cs +++ b/src/Kentico.Xperience.StoreApi/Products/ProductsController.cs @@ -8,6 +8,7 @@ using Kentico.Xperience.StoreApi.Products.Categories; using Kentico.Xperience.StoreApi.Products.Pages; using Kentico.Xperience.StoreApi.Products.Prices; +using Kentico.Xperience.StoreApi.Products.SKU; using Kentico.Xperience.StoreApi.Routing; using Microsoft.AspNetCore.Http; @@ -146,4 +147,24 @@ public async Task> GetInventoryPrices(in return ValidationProblem(); } } + + + /// + /// Returns all standalone products. + /// + /// Product request parameters. + /// + [HttpGet("standalone-products", Name = nameof(GetStandaloneProducts))] + [AuthorizeStore] + public async Task>> GetStandaloneProducts([FromQuery] ProductRequest request) + { + if (request.Culture is not null && + !CultureSiteInfoProvider.IsCultureOnSite(request.Culture, SiteContext.CurrentSiteName)) + { + return BadRequest($"Culture '{request.Culture}' is not assigned for site"); + } + + var products = await productService.GetStandaloneProducts(request); + return Ok(products); + } } diff --git a/src/Kentico.Xperience.StoreApi/Products/SKU/IProductSKUConverter.cs b/src/Kentico.Xperience.StoreApi/Products/SKU/IProductSKUConverter.cs index b6ab646a..c3e5e76c 100644 --- a/src/Kentico.Xperience.StoreApi/Products/SKU/IProductSKUConverter.cs +++ b/src/Kentico.Xperience.StoreApi/Products/SKU/IProductSKUConverter.cs @@ -14,6 +14,7 @@ public interface IProductSKUConverter /// SKU info. /// Currency code in which evaluates prices. /// If true, variants are included. + /// If true, is included. /// - TModel Convert(SKUInfo skuInfo, string currencyCode, bool withVariants); + TModel Convert(SKUInfo skuInfo, string currencyCode, bool withVariants, bool withLongDescription); } diff --git a/src/Kentico.Xperience.StoreApi/Products/SKU/KProductSKU.cs b/src/Kentico.Xperience.StoreApi/Products/SKU/KProductSKU.cs index cddeeeb6..095e59c3 100644 --- a/src/Kentico.Xperience.StoreApi/Products/SKU/KProductSKU.cs +++ b/src/Kentico.Xperience.StoreApi/Products/SKU/KProductSKU.cs @@ -18,6 +18,8 @@ public class KProductSKU public string SKUShortDescription { get; set; } + public string SKULongDescription { get; set; } + public string SKUNumber { get; set; } public bool SKUEnabled { get; set; } diff --git a/src/Kentico.Xperience.StoreApi/Products/SKU/ProductSKUConverter.cs b/src/Kentico.Xperience.StoreApi/Products/SKU/ProductSKUConverter.cs index aeb68d6b..4a15dbbd 100644 --- a/src/Kentico.Xperience.StoreApi/Products/SKU/ProductSKUConverter.cs +++ b/src/Kentico.Xperience.StoreApi/Products/SKU/ProductSKUConverter.cs @@ -37,7 +37,7 @@ public ProductSKUConverter(ICatalogPriceCalculatorFactory catalogPriceCalculator /// - public virtual TModel Convert(SKUInfo skuInfo, string currencyCode, bool withVariants) + public virtual TModel Convert(SKUInfo skuInfo, string currencyCode, bool withVariants, bool withLongDescription) { var model = mapper.Map(skuInfo); @@ -66,6 +66,11 @@ public virtual TModel Convert(SKUInfo skuInfo, string currencyCode, bool withVar model.Variants = variants; } + if (withLongDescription) + { + model.SKULongDescription = skuInfo.SKUDescription; + } + model.Prices = mapper.Map(prices); return model; diff --git a/submodules/xperience-by-kentico-ecommerce-common b/submodules/xperience-by-kentico-ecommerce-common index cf7540fd..91a62d6b 160000 --- a/submodules/xperience-by-kentico-ecommerce-common +++ b/submodules/xperience-by-kentico-ecommerce-common @@ -1 +1 @@ -Subproject commit cf7540fddfe1e5ca7e376aaf1467d29dce65267e +Subproject commit 91a62d6bcbcde1faa8d06d5f8d5cc0fa68fc68a4 diff --git a/test/Kentico.Xperience.K13Ecommerce.IntegrationTests/StoreApiOrderTests.cs b/test/Kentico.Xperience.K13Ecommerce.IntegrationTests/StoreApiOrderTests.cs new file mode 100644 index 00000000..8bc46856 --- /dev/null +++ b/test/Kentico.Xperience.K13Ecommerce.IntegrationTests/StoreApiOrderTests.cs @@ -0,0 +1,85 @@ +using Kentico.Xperience.K13Ecommerce.StoreApi; + +namespace Kentico.Xperience.KStore.Tests; + +/// +/// Store API order integration tests. +/// +[TestFixture] +[Category("IntegrationTests")] +public class StoreApiOrderTests : StoreApiTestBase +{ + /// + /// Test for existing order. + /// + /// Order ID. + /// Customer email. + [Test] + [TestCase(27, "automation.test@test.com")] + public async Task OrderDetail_ExistingOrder(int orderId, string customerEmail) + { + var order = await StoreApiClient.OrderDetailAsync(orderId); + + Assert.That(order.OrderId, Is.EqualTo(orderId)); + Assert.That(order.OrderItems, Has.Count.EqualTo(2)); + + Assert.That(order.OrderCustomer.CustomerEmail, Is.EqualTo(customerEmail)); + Assert.That(order.OrderBillingAddress.AddressLine1, Is.EqualTo("Automation Street 1")); + Assert.That(order.OrderBillingAddress.AddressCity, Is.EqualTo("Automation City A")); + Assert.That(order.OrderBillingAddress.AddressZip, Is.EqualTo("00001")); + Assert.That(order.OrderBillingAddress.AddressCountryId, Is.EqualTo(326)); + + Assert.That(order.OrderShippingAddress.AddressLine1, Is.EqualTo("Automation Street 2")); + Assert.That(order.OrderShippingAddress.AddressCity, Is.EqualTo("Automation City B")); + Assert.That(order.OrderShippingAddress.AddressZip, Is.EqualTo("00002")); + Assert.That(order.OrderShippingAddress.AddressCountryId, Is.EqualTo(326)); + + Assert.That(order.OrderStatus.StatusName, Is.EqualTo("New")); + Assert.That(order.OrderShippingOption.ShippingOptionName, Is.EqualTo("StandardDelivery")); + Assert.That(order.OrderPaymentOption.PaymentOptionName, Is.EqualTo("DancingGoatCore.MoneyTransfer")); + } + + /// + /// Test order detail when order does not exist. + /// + /// Order ID. + [Test] + [TestCase(99999)] + public async Task OrderDetail_NonExistingOrder(int orderId) + { + // test that exception has status code 404 + var exception = Assert.ThrowsAsync(async () => await StoreApiClient.OrderDetailAsync(orderId)); + Assert.That(exception!.StatusCode, Is.EqualTo(404)); + } + + /// + /// Test for order update. + /// + /// Order ID. + [Test] + [TestCase(27)] + public async Task OrderUpdate_ChangeStatus_And_IsPaid(int orderId) + { + var order = await StoreApiClient.OrderDetailAsync(orderId); + + order.OrderStatus = new KOrderStatus { StatusId = 2 }; + order.OrderIsPaid = true; + + await StoreApiClient.UpdateOrderAsync(order); + + var updatedOrder = await StoreApiClient.OrderDetailAsync(orderId); + + Assert.That(updatedOrder.OrderStatus.StatusName, Is.EqualTo("PaymentReceived")); + Assert.That(updatedOrder.OrderIsPaid, Is.True); + + order.OrderIsPaid = false; + order.OrderStatus = new KOrderStatus { StatusId = 3 }; + + await StoreApiClient.UpdateOrderAsync(order); + + var rollbackedOrder = await StoreApiClient.OrderDetailAsync(orderId); + + Assert.That(rollbackedOrder.OrderStatus.StatusName, Is.EqualTo("New")); + Assert.That(rollbackedOrder.OrderIsPaid, Is.False); + } +}