BAEL-6753: Storing PostgreSQL jsonb Using SpringBoot and JPA (#15632)

* BAEL-6753: Storing PostgreSQL jsonb Using SpringBoot and JPA

* BAEL-6753: Storing PostgreSQL jsonb Using SpringBoot and JPA
This commit is contained in:
Manfred 2024-01-28 01:14:00 +00:00 committed by GitHub
parent d458fe57e9
commit 002384b668
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 264 additions and 0 deletions

View File

@ -30,6 +30,11 @@
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>io.hypersistence</groupId>
<artifactId>hypersistence-utils-hibernate-55</artifactId>
<version>${hypersistance-utils-hibernate-55.version}</version>
</dependency>
<dependency>
<groupId>com.h2database</groupId>
<artifactId>h2</artifactId>
@ -47,6 +52,30 @@
<artifactId>guava</artifactId>
<version>${guava.version}</version>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
</dependency>
<dependency>
<groupId>org.postgresql</groupId>
<artifactId>postgresql</artifactId>
<version>${postgresql.version}</version>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
<version>${org.slf4j.version}</version>
</dependency>
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-core</artifactId>
<version>${logback.version}</version>
</dependency>
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>${logback.version}</version>
</dependency>
</dependencies>
<build>
@ -98,4 +127,9 @@
</plugins>
</build>
<properties>
<hypersistance-utils-hibernate-55.version>3.7.0</hypersistance-utils-hibernate-55.version>
<postgresql.version>42.7.1</postgresql.version>
</properties>
</project>

View File

@ -0,0 +1,15 @@
package com.baeldung.spring.data.persistence.json;
import lombok.*;
@Data
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode
public class Address {
private String postCode;
private String city;
}

View File

@ -0,0 +1,36 @@
package com.baeldung.spring.data.persistence.json;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.extern.slf4j.Slf4j;
import javax.persistence.AttributeConverter;
import javax.persistence.Converter;
@Converter
@Slf4j
public class AddressAttributeConverter implements AttributeConverter<Address, String> {
private static final ObjectMapper objectMapper = new ObjectMapper();
@Override
public String convertToDatabaseColumn(Address address) {
try {
return objectMapper.writeValueAsString(address);
} catch (JsonProcessingException jpe) {
log.warn("Cannot convert Address into JSON");
return null;
}
}
@Override
public Address convertToEntityAttribute(String value) {
try {
return objectMapper.readValue(value, Address.class);
} catch (JsonProcessingException e) {
log.warn("Cannot convert JSON into Address");
return null;
}
}
}

View File

@ -0,0 +1,13 @@
package com.baeldung.spring.data.persistence.json;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class JsonAttributeApplication {
public static void main(String[] args) {
SpringApplication.run(JsonAttributeApplication.class, args);
}
}

View File

@ -0,0 +1,31 @@
package com.baeldung.spring.data.persistence.json;
import javax.persistence.*;
import io.hypersistence.utils.hibernate.type.json.JsonBinaryType;
import lombok.*;
import org.hibernate.annotations.Type;
import org.hibernate.annotations.TypeDef;
@Entity
@Table(name = "student")
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(of = {"id"})
@TypeDef(name = "jsonb", typeClass = JsonBinaryType.class)
public class StudentEntity {
@Id
@Column(name = "student_id", length = 8)
private String id;
@Column(name = "admit_year", length = 4)
private String admitYear;
@Type(type = "jsonb")
@Column(name = "address", columnDefinition = "jsonb")
private Address address;
}

View File

@ -0,0 +1,16 @@
package com.baeldung.spring.data.persistence.json;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.CrudRepository;
import org.springframework.data.repository.query.Param;
import org.springframework.stereotype.Repository;
import java.util.List;
@Repository
public interface StudentRepository extends CrudRepository<StudentEntity, String> {
@Query(value = "SELECT * FROM student WHERE address->>'postCode' = :postCode", nativeQuery = true)
List<StudentEntity> findByAddressPostCode(@Param("postCode") String postCode);
}

View File

@ -0,0 +1,27 @@
package com.baeldung.spring.data.persistence.json;
import javax.persistence.*;
import lombok.*;
@Entity
@Table(name = "student_str")
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(of = {"id"})
public class StudentStrEntity {
@Id
@Column(name = "student_id", length = 8)
private String id;
@Column(name = "admit_year", length = 4)
private String admitYear;
@Convert(converter = AddressAttributeConverter.class)
@Column(name = "address", length = 500)
private Address address;
}

View File

@ -0,0 +1,16 @@
package com.baeldung.spring.data.persistence.json;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.CrudRepository;
import org.springframework.data.repository.query.Param;
import org.springframework.stereotype.Repository;
import java.util.List;
@Repository
public interface StudentStrRepository extends CrudRepository<StudentStrEntity, String> {
@Query(value = "SELECT * FROM student WHERE address->>'postCode' = :postCode", nativeQuery = true)
List<StudentStrEntity> findByAddressPostCode(@Param("postCode") String postCode);
}

View File

@ -0,0 +1,76 @@
package com.baeldung.spring.data.persistence.json;
import org.junit.jupiter.api.*;
import org.springframework.boot.test.context.SpringBootTest;
import javax.inject.Inject;
import java.util.List;
import java.util.Optional;
import static org.assertj.core.api.Assertions.assertThat;
@SpringBootTest(classes = JsonAttributeApplication.class)
@TestMethodOrder(MethodOrderer.OrderAnnotation.class)
class JsonAttributeLiveTest {
@Inject
private StudentStrRepository studentStrRepository;
@Inject
private StudentRepository studentRepository;
@Test
@Order(10)
void whenSaveAnStudentStrEntityAndFindById_thenTheRecordPresentsInDb() {
String studentId = "23876371";
String postCode = "KT6 7BB";
Address address = new Address(postCode, "London");
StudentStrEntity studentStrEntity = StudentStrEntity.builder()
.id(studentId)
.admitYear("2023")
.address(address)
.build();
StudentStrEntity savedStudentStrEntity = studentStrRepository.save(studentStrEntity);
Optional<StudentStrEntity> studentEntityOptional = studentStrRepository.findById(studentId);
assertThat(studentEntityOptional.isPresent()).isTrue();
studentStrEntity = studentEntityOptional.get();
assertThat(studentStrEntity.getId()).isEqualTo(studentId);
assertThat(studentStrEntity.getAddress().getPostCode()).isEqualTo(postCode);
}
@Test
@Order(20)
void whenSaveAnStudentEntityAndFindById_thenTheRecordPresentsInDb() {
String studentId = "23876371";
String postCode = "KT6 7BB";
Address address = new Address(postCode, "London");
StudentEntity studentEntity = StudentEntity.builder()
.id(studentId)
.admitYear("2023")
.address(address)
.build();
StudentEntity savedStudentEntity = studentRepository.save(studentEntity);
Optional<StudentEntity> studentEntityOptional = studentRepository.findById(studentId);
assertThat(studentEntityOptional.isPresent()).isTrue();
studentEntity = studentEntityOptional.get();
assertThat(studentEntity.getId()).isEqualTo(studentId);
assertThat(studentEntity.getAddress().getPostCode()).isEqualTo(postCode);
}
@Test
@Order(50)
void whenFindByAddressPostCode_thenReturnListIsNotEmpty() {
String postCode = "KT6 7BB";
List<StudentStrEntity> studentStrEntityList = studentStrRepository.findByAddressPostCode(postCode);
assertThat(studentStrEntityList).isNotEmpty();
}
}