Overview
This endpoint updates an existing basket entity or creates a new one if it doesn’t exist. The basket represents an aggregate root in our domain model that maintains its own consistency boundaries.
The operation follows the Repository Pattern for persistence and triggers domain events when the basket state changes. This aligns with our DDD approach by:
- Treating the basket as a complete aggregate with items as value objects
- Maintaining invariants during the update operation
- Encapsulating business rules within the domain model
- Ensuring atomic updates to maintain data consistency
- Publishing domain events for cross-boundary communication
When a basket is updated, the system publishes integration events that other bounded contexts (like Catalog and Ordering) can subscribe to if needed.
Implementation Details
The Update Basket operation is implemented using the CQRS pattern with a dedicated command handler:
Loading graph...
Key Components
- UpdateBasketCommand: Implements
ICommand
to update an existing basket with new items - UpdateBasketHandler: Processes the command using repository pattern
- UpdateBasketValidator: Validates the command parameters using FluentValidation
- CustomerBasket: Domain entity that encapsulates basket data and business rules
- BasketItem: Value object representing an item in the basket
- IBasketRepository: Repository interface for basket persistence operations
Technical Implementation
The update operation follows these steps:
- Authentication: Extracts the user ID from the JWT token claims
- Basket Retrieval: Fetches the existing basket using the user ID as the key
- Domain Update: Calls the
Update
method on theCustomerBasket
entity - Persistence: Saves the updated basket to the repository
- Response: Returns a 204 No Content response on success
The implementation enforces several key principles:
- Command Validation: Ensures the request contains valid data before processing
- Domain Encapsulation: The basket entity controls its own state changes
- Repository Abstraction: Data access is abstracted behind a repository interface
- Error Handling: Specific exceptions for different error scenarios
Authentication
This endpoint requires authentication. The user must be logged in with a valid JWT token containing a Keycloak subject claim. The basket is associated with the authenticated user’s ID.
Validation Rules
The following validation rules are enforced:
- The basket items collection must not be empty
- For each basket item:
- Book ID must not be empty
- Quantity must be greater than 0
Architecture
PUT (/api/v1/baskets)
Request Body
Example Usage
curl -X PUT https://api.bookworm.com/api/v1/baskets \ -H "Authorization: Bearer <your-jwt-token>" \ -H "Content-Type: application/json" \ -d '{ "items": [ { "id": "0195e52d-4f5b-7280-9a17-ec3bf0a5f733", "quantity": 2 }, { "id": "0195e52d-4f5b-7a69-8489-2c0985141a63", "quantity": 1 } ]}'
Responses
204 No Content
Returned when the basket is successfully updated.
400 Bad Request
Returned when the request validation fails.
401 Unauthorized
Returned when:
- The user is not authenticated
- The authentication token is missing or invalid
404 Not Found
Returned when the basket for the authenticated user cannot be found.
Error Handling
The service handles various error scenarios:
- Validation Errors: Returns 400 Bad Request with detailed validation messages
- Authentication Errors: Returns 401 Unauthorized if user is not properly authenticated
- Not Found Errors: Returns 404 if the basket doesn’t exist
Implementation Notes
- The endpoint is versioned (v1) and follows REST principles
- Uses optimistic concurrency for basket updates
- Implements domain-driven design patterns with
Basket
as an aggregate root - Provides immediate consistency for basket operations