-
Notifications
You must be signed in to change notification settings - Fork 72
Open
Description
Problem Description
The PUT /coffees/{id} endpoint throws a JpaSystemException when the request body doesn't include the id field, even though the id is provided in the URL path.
Error Message
org.hibernate.id.IdentifierGenerationException: Identifier of entity 'com.thehecklers.sbur_rest_demo.Coffee' must be manually assigned before calling 'persist()'
Root Cause
The issue occurs because:
- The
@PathVariable String idparameter extracts the ID from the URL path (e.g.,/coffees/1111→id = "1111"). - The
@RequestBody Coffee coffeeobject is created by Jackson from the JSON body only. - These are two separate variables/objects — the URL path parameter and the request body object are independent.
- When the JSON body doesn't include an
idfield (e.g.,{"name": "byecoffee"}), Jackson creates aCoffeeobject using the default constructor, leaving theidfield asnull. - Hibernate/JPA requires the
idfield to be set before callingsave(), causing the exception.
Steps to Reproduce
- Send a PUT request to
/coffees/1111with the following JSON body:{ "name": "byecoffee" } - The request fails with a 500 Internal Server Error.
Expected Behavior
The endpoint should use the id from the URL path to update or create the entity, regardless of whether the JSON body includes the id field.
Suggested Fix
Add coffee.setId(id); before calling coffeeRepository.save(coffee) to ensure the entity has the correct ID from the URL path:
@PutMapping("/{id}")
ResponseEntity<Coffee> putCoffee(@PathVariable String id,
@RequestBody Coffee coffee) {
coffee.setId(id); // Add this line
return (!coffeeRepository.existsById(id))
? new ResponseEntity<>(coffeeRepository.save(coffee),
HttpStatus.CREATED)
: new ResponseEntity<>(coffeeRepository.save(coffee),
HttpStatus.OK);
}Additional Context
- Spring Boot version: 3.5.8
- Using JPA with Hibernate
- The
Coffeeentity uses@Idannotation without auto-generation, requiring manual ID assignment
Metadata
Metadata
Assignees
Labels
No labels