Я обращался к ряду вопросов о подходах / публикациях / stackoverflow, чтобы справиться со следующей ошибкой (полная трассировка стека) при запуске приложения Kotlin / SpringBoot:
2020-04-22 18:33:56.823 ERROR 46345 --- [ restartedMain] o.s.boot.SpringApplication : Application run failed
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'entityManagerFactory' defined in class path resource [org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaConfiguration.class]: Invocation of init method failed; nested exception is javax.persistence.PersistenceException: [PersistenceUnit: default] Unable to build Hibernate SessionFactory; nested exception is org.hibernate.MappingException: No Dialect mapping for JDBC type: 2118910070
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1803)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:595)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:517)
at org.springframework.beans.factory.support.AbstractBeanFactory.lambda$doGetBean$0(AbstractBeanFactory.java:323)
at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:222)
at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:321)
at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:202)
at org.springframework.context.support.AbstractApplicationContext.getBean(AbstractApplicationContext.java:1108)
at org.springframework.context.support.AbstractApplicationContext.finishBeanFactoryInitialization(AbstractApplicationContext.java:868)
at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:550)
at org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext.refresh(ServletWebServerApplicationContext.java:141)
at org.springframework.boot.SpringApplication.refresh(SpringApplication.java:747)
at org.springframework.boot.SpringApplication.refreshContext(SpringApplication.java:397)
at org.springframework.boot.SpringApplication.run(SpringApplication.java:315)
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1226)
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1215)
at app.ApplicationKt.main(Application.kt:13)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:566)
at org.springframework.boot.devtools.restart.RestartLauncher.run(RestartLauncher.java:49)
Caused by: javax.persistence.PersistenceException: [PersistenceUnit: default] Unable to build Hibernate SessionFactory; nested exception is org.hibernate.MappingException: No Dialect mapping for JDBC type: 2118910070
at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean.buildNativeEntityManagerFactory(AbstractEntityManagerFactoryBean.java:403)
at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean.afterPropertiesSet(AbstractEntityManagerFactoryBean.java:378)
at org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean.afterPropertiesSet(LocalContainerEntityManagerFactoryBean.java:341)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFactory.java:1862)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1799)
... 21 common frames omitted
Caused by: org.hibernate.MappingException: No Dialect mapping for JDBC type: 2118910070
at org.hibernate.dialect.TypeNames.get(TypeNames.java:71)
at org.hibernate.dialect.TypeNames.get(TypeNames.java:103)
at org.hibernate.dialect.Dialect.getTypeName(Dialect.java:369)
at org.hibernate.mapping.Column.getSqlType(Column.java:238)
at org.hibernate.tool.schema.internal.AbstractSchemaValidator.validateColumnType(AbstractSchemaValidator.java:156)
at org.hibernate.tool.schema.internal.AbstractSchemaValidator.validateTable(AbstractSchemaValidator.java:143)
at org.hibernate.tool.schema.internal.GroupedSchemaValidatorImpl.validateTables(GroupedSchemaValidatorImpl.java:42)
at org.hibernate.tool.schema.internal.AbstractSchemaValidator.performValidation(AbstractSchemaValidator.java:89)
at org.hibernate.tool.schema.internal.AbstractSchemaValidator.doValidation(AbstractSchemaValidator.java:68)
at org.hibernate.tool.schema.spi.SchemaManagementToolCoordinator.performDatabaseAction(SchemaManagementToolCoordinator.java:192)
at org.hibernate.tool.schema.spi.SchemaManagementToolCoordinator.process(SchemaManagementToolCoordinator.java:73)
at org.hibernate.internal.SessionFactoryImpl.<init>(SessionFactoryImpl.java:320)
at org.hibernate.boot.internal.SessionFactoryBuilderImpl.build(SessionFactoryBuilderImpl.java:462)
at org.hibernate.jpa.boot.internal.EntityManagerFactoryBuilderImpl.build(EntityManagerFactoryBuilderImpl.java:1249)
at org.springframework.orm.jpa.vendor.SpringHibernateJpaPersistenceProvider.createContainerEntityManagerFactory(SpringHibernateJpaPersistenceProvider.java:58)
at org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean.createNativeEntityManagerFactory(LocalContainerEntityManagerFactoryBean.java:365)
at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean.buildNativeEntityManagerFactory(AbstractEntityManagerFactoryBean.java:391)
... 25 common frames omitted
Проблема заключается в отображение типа данных PostgreSQL JSONB с помощью Hibernate.
2 подхода, которые я тщательно пробовал и отлаживал, следующие:
- Реализация пользовательского отображения Hibernate и создание пользовательского UserType для JSONB. Ссылки: здесь , здесь , здесь и здесь
- Используйте типы Hibernate. Список литературы здесь , здесь и здесь
Я пытался обильно в обоих случаях, но безуспешно, и я очень заинтересован чтобы понять, где я иду не так и чего мне не хватает.
Подход 1
My Entity:
@Entity
@TypeDef(name = "JsonUserType", typeClass = JsonUserType::class)
@Table(name = "entity")
data class MyEntity(
@Column(nullable = false)
val id: UUID,
@Column(nullable = false)
@Enumerated(value = EnumType.STRING)
@Column(nullable = false)
val type: Type,
@Type(type = "JsonUserType")
@Column(columnDefinition = "jsonb")
@Basic(fetch = FetchType.LAZY)
var event_data: Event
) : SomeEntity<UUID>(), SomeOtherStuff {
override fun getName(): String {
return id
}
}
enum class Type(val value: String) {
TYPE1("Type1"),
TYPE2("Type2")
}
My PoJO:
data class Event(
val someContent: String,
val someBoolean: Boolean
) : Serializable { //equals, hashcode etc are omitted }
Мой пользовательский диалект Hibernate:
class CustomPostgreSQLDialect : PostgreSQL95Dialect {
constructor() : super() {
this.registerColumnType(Types.JAVA_OBJECT, "jsonb")
}
}
Мой пользовательский тип (абстрактный класс)
abstract class JsonDataUserType : UserType {
override fun sqlTypes(): IntArray? {
return intArrayOf(Types.JAVA_OBJECT)
}
override fun equals(value1: Any?, value2: Any?): Boolean {
return value1 == value2
}
override fun hashCode(value1: Any?): Int {
return value1!!.hashCode()
}
override fun assemble(value1: Serializable?, value2: Any?): Any {
return deepCopy(value1)
}
override fun disassemble(value1: Any?): Serializable {
return deepCopy(value1) as Serializable
}
override fun deepCopy(p0: Any?): Any {
return try {
val bos = ByteArrayOutputStream()
val oos = ObjectOutputStream(bos)
oos.writeObject(p0)
oos.flush()
oos.close()
bos.close()
val bais = ByteArrayInputStream(bos.toByteArray())
ObjectInputStream(bais).readObject()
} catch (ex: ClassNotFoundException) {
throw HibernateException(ex)
} catch (ex: IOException) {
throw HibernateException(ex)
}
}
override fun replace(p0: Any?, p1: Any?, p2: Any?): Any {
return deepCopy(p0)
}
override fun nullSafeSet(p0: PreparedStatement?, p1: Any?, p2: Int, p3: SharedSessionContractImplementor?) {
if (p1 == null) {
p0?.setNull(p2, Types.OTHER)
return
}
try {
val mapper = ObjectMapper()
val w = StringWriter()
mapper.writeValue(w, p1)
w.flush()
p0?.setObject(p2, w.toString(), Types.OTHER)
} catch (ex: java.lang.Exception) {
throw RuntimeException("Failed to convert Jsonb to String: " + ex.message, ex)
}
}
override fun nullSafeGet(p0: ResultSet?, p1: Array<out String>?, p2: SharedSessionContractImplementor?, p3: Any?): Any {
val cellContent = p0?.getString(p1?.get(0))
return try {
val mapper = ObjectMapper()
mapper.readValue(cellContent?.toByteArray(charset("UTF-8")), returnedClass())
} catch (ex: Exception) {
throw RuntimeException("Failed to convert String to Jsonb: " + ex.message, ex)
}
}
override fun isMutable(): Boolean {
return true
}
}
Такой класс был взят из этого Stackoverflow вопрос
Мой конкретный класс:
class JsonType : JsonDataUserType() {
override fun returnedClass(): Class<Event> {
return Event::class.java
}
}
Мои свойства hibernate jpa application.yml
jpa.properties.database.database-platform: org.hibernate.dialect.PostgreSQL95Dialect
jpa.properties.hibernate.dialect: org.myapp.util.CustomPostgreSQLDialect
Подход 2
Свойства Hibernate точно такие же, как и класс PoJo, пользовательский картограф не включен.
Сущность
@Entity
@TypeDef(
name = "jsonb",
typeClass = JsonBinaryType::class
)
@Table(name = "entity")
data class MyEntity(
@Column(nullable = false)
val id: UUID,
@Column(nullable = false)
@Enumerated(value = EnumType.STRING)
@Column(nullable = false)
val type: Type,
@Type(type = "jsonb")
@Column(columnDefinition = "jsonb")
@Basic(fetch = FetchType.LAZY)
var event_data: Event
) : SomeEntity<UUID>(), SomeOtherStuff {
override fun getName(): String {
return id
}
}
enum class Type(val value: String) {
TYPE1("Type1"),
TYPE2("Type2")
}
Пользовательский диалект (с использованием типов гибернации):
class CustomPostgreSQLDialect : PostgreSQL95Dialect {
constructor() : super() {
this.registerHibernateType(Types.OTHER, JsonNodeBinaryType::class.java.name)
this.registerHibernateType(Types.OTHER, JsonStringType::class.java.name)
this.registerHibernateType(Types.OTHER, JsonBinaryType::class.java.name)
this.registerHibernateType(Types.OTHER, JsonNodeBinaryType::class.java.name)
this.registerHibernateType(Types.OTHER, JsonNodeStringType::class.java.name)
}
}
Обратите внимание, что я также пытался использовать только:
this.registerHibernateType(Types.OTHER, "jsonb")
а также наличие всего этого в моей сущности или базовой сущности, из которой она вышла (без изменений):
@TypeDefs({
@TypeDef(name = "string-array", typeClass = StringArrayType.class),
@TypeDef(name = "int-array", typeClass = IntArrayType.class),
@TypeDef(name = "json", typeClass = JsonStringType.class),
@TypeDef(name = "jsonb", typeClass = JsonBinaryType.class),
@TypeDef(name = "jsonb-node", typeClass = JsonNodeBinaryType.class),
@TypeDef(name = "json-node", typeClass = JsonNodeStringType.class),
})
Есть ли что-то явно неправильное, что я делаю в обоих подходах? Я не могу заставить его работать, и, не уверен, имеет ли какое-либо отношение числовое значение после Нет отображения диалектов для типа JDB C: всегда отличается. Я добавляю это, поскольку видел некоторые идентификаторы, относящиеся к определенным категориям ошибок.
Вы можете помочь?
Спасибо
РЕДАКТИРОВАТЬ: Я хотел предоставить больше информации о версиях jpa, postgres и hibernate. В настоящее время я работаю со следующим:
- postgres: 10-alpine
- PostgreSQL JDB C Driver JDB C 4.2 »42.2.8
- org.springframework.boot: spring-boot-starter-data-jpa: 2.2.1.RELEASE
org.hibernate: hibernate-core: 5.4.8.Final
Есть ли какие-то конкретные проблемы с версиями среди них?
РЕДАКТИРОВАТЬ 2 Я пытался успешно использовать hibernate-типы (подход 2, как описано выше). Я сделал следующее изменение в соответствии с Postgres версией (10):
class CustomPostgreSQLDialect : PostgreSQL10Dialect {
constructor() : super() {
this.registerHibernateType(Types.OTHER, StringArrayType::class.java.name)
this.registerHibernateType(Types.OTHER, IntArrayType::class.java.name)
this.registerHibernateType(Types.OTHER, JsonStringType::class.java.name)
this.registerHibernateType(Types.OTHER, JsonBinaryType::class.java.name)
this.registerHibernateType(Types.OTHER, JsonNodeBinaryType::class.java.name)
this.registerHibernateType(Types.OTHER, JsonNodeStringType::class.java.name)
}
}
Тогда в моей сущности у меня есть
@TypeDefs({
@TypeDef(name = "string-array", typeClass = StringArrayType.class),
@TypeDef(name = "int-array", typeClass = IntArrayType.class),
@TypeDef(name = "json", typeClass = JsonStringType.class),
@TypeDef(name = "jsonb", typeClass = JsonBinaryType.class)
})
и
@Type(type = "jsonb")
@Column(columnDefinition = "jsonb")
@Basic(fetch = FetchType.LAZY)
var event_data: Event
Затем я отладил метод get в TypeNames , откуда исходит ошибка:
public String get(final int typeCode) throws MappingException {
final Integer integer = Integer.valueOf( typeCode );
final String result = defaults.get( integer );
if ( result == null ) {
throw new MappingException( "No Dialect mapping for JDBC type: " + typeCode );
}
return result;
}
, и вот что я получаю:
defaults = {HashMap@12093} size = 27
{Integer@12124} -1 -> "text"
{Integer@12126} 1 -> "char(1)"
{Integer@12128} -2 -> "bytea"
{Integer@12130} 2 -> "numeric($p, $s)"
{Integer@12132} -3 -> "bytea"
{Integer@12133} -4 -> "bytea"
{Integer@12134} 4 -> "int4"
{Integer@12136} -5 -> "int8"
{Integer@12138} -6 -> "int2"
{Integer@12140} 5 -> "int2"
{Integer@12141} -7 -> "bool"
{Integer@12143} 6 -> "float4"
{Integer@12145} 7 -> "real"
{Integer@12147} 8 -> "float8"
{Integer@12149} -9 -> "nvarchar($l)"
{Integer@12151} 12 -> "varchar($l)"
{Integer@12153} -15 -> "nchar($l)"
{Integer@12155} -16 -> "nvarchar($l)"
{Integer@12156} 16 -> "boolean"
{Integer@12158} 2000 -> "json"
{Integer@12160} 2004 -> "oid"
{Integer@12162} 2005 -> "text"
{Integer@12163} 1111 -> "uuid"
{Integer@12165} 91 -> "date"
{Integer@12167} 2011 -> "nclob"
{Integer@12169} 92 -> "time"
{Integer@12171} 93 -> "timestamp"
Нет jsonb можно найти, и когда я отлаживаю свой собственный диалект, я получаю следующее:
{Integer@10846} 1111 -> "com.vladmihalcea.hibernate.type.json.JsonStringType"
key = {Integer@10846} 1111
value = "com.vladmihalcea.hibernate.type.json.JsonStringType"
Почему это так? Почему я не получаю тип JSONB?