Skip to content

Commit

Permalink
Merge branch 'main' into fix/allow_call_http_root_path
Browse files Browse the repository at this point in the history
  • Loading branch information
alexandrepa authored Jan 8, 2025
2 parents 1e52188 + c2ba4e2 commit fbddc78
Show file tree
Hide file tree
Showing 3 changed files with 187 additions and 62 deletions.
4 changes: 2 additions & 2 deletions pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -79,15 +79,15 @@
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-dependencies</artifactId>
<version>2023.0.3</version>
<version>2024.0.0</version>
<type>pom</type>
<scope>import</scope>
</dependency>
<dependency>
<!-- Import dependency management from Spring Boot -->
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>3.3.5</version>
<version>3.4.1</version>
<type>pom</type>
<scope>import</scope>
</dependency>
Expand Down
128 changes: 120 additions & 8 deletions tzatziki-spring-jpa/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -98,20 +98,18 @@ public class User {
You can now insert rows with:
```gherkin
Given that the users table will contain:
| id | firstName | lastName |
| 1 | Darth | Vader |
| firstName | lastName |
| Darth | Vader |
# or alternatively
Given that the User entities will contain:
"""
- id: 1
firstName: Darth
- firstName: Darth
lastName: Vader
"""
# single rows work as well
Given that the users table will contain:
"""
id: 1
firstName: Darth
lastName: Vader
"""
Expand All @@ -120,7 +118,6 @@ Given that the UserRepository repository will contain:
"""
[
{
"id": 1,
"firstName": "Darth",
"lastName": "Vader"
}
Expand All @@ -131,8 +128,123 @@ Given that the UserRepository repository will contain:
Adding `only` to the step will also empty the table before inserting the data:
```gherkin
But when the users table will contain only:
| id | firstName | lastName |
| 1 | Han | Solo |
| firstName | lastName |
| Han | Solo |
```

### Important Note on Generated IDs

Starting with Hibernate 6.6+ and Spring Boot 3.4+, you can no longer manually specify values for IDs that are generated by the framework.

#### Example
Suppose you have an entity mapped as follows:

```java
@Getter
@Entity
@NoArgsConstructor
@Table(name = "users")
public class User {

@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Integer id;

@Column(name = "first_name")
private String firstName;
}
```

In this example, Hibernate **will not allow you** to insert data with explicit `id` values like this:

```gherkin
Given the users table will contain:
| id | firstName | lastName |
| 42 | Darth | Vader |
```

The test above fails with the following exception:

```
org.springframework.orm.ObjectOptimisticLockingFailureException: Row was updated or deleted by another transaction (or unsaved-value mapping was incorrect)
```
This exception occurs because Hibernate expects the database to assign the primary key when `GenerationType.IDENTITY` is used.
Providing an explicit `id` value conflicts with Hibernate’s assumptions, leading to a failure.

#### Correct Approach

To avoid this issue, ensure that the input data does not include the `id` column, as shown below:

```gherkin
Given the users table will contain:
| firstName | lastName |
| Darth | Vader |
```

### Inserting Data into Parent and Child Tables
When working with parent and child tables where the child table has a foreign key referencing the parent table, you may need to ensure correct key mapping during data insertion.

When the primary key of the parent table is auto-generated, you may not know its exact value at the time of insertion into the child table.

However, since the primary key values in the parent table follow a predictable sequence starting at 1, you can infer the foreign key values.

This behavior is consistent because Tzatziki resets the sequence at the beginning of each scenario.

The following code demonstrates an example where the `User` entity references the `Group` entity:

```java
@NoArgsConstructor
@Getter
@Entity
@Table(name = "groups")
public class Group {

@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
Integer id;

@Column(name = "name")
String name;

@OneToMany(mappedBy = "group")
List<User> users;
}
```
```java
@NoArgsConstructor
@Getter
@Entity
@Table(name = "users")
public class User {

@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
Integer id;

@Column(name = "first_name")
String firstName;

@Column(name = "last_name")
String lastName;

@ManyToOne
@JoinColumn(name = "group_id")
Group group;
}

```
The example below inserts the data properly:

```gherkin
Given that the groups table will contain:
| name |
| admins |
| guests |
And the users table will contain:
| firstName | lastName | group.id |
| Chuck | Norris | 1 |
| Uma | Thurman | 2 |
| Jackie | Chan | 2 |
```

## Asserting data
Expand Down
Loading

0 comments on commit fbddc78

Please sign in to comment.