Проблема с отображением Hibernate многие ко многим - PullRequest
0 голосов
/ 28 сентября 2018

У меня есть программа hibernate + oracle.В этом у меня есть отношение «многие ко многим» между таблицей «Приложения» и «IdentityProviders» через таблицу «ApplicationIdentityProviders».Я получаю следующую ошибку при запуске программы:

Exception in thread "main" java.lang.RuntimeException: org.hibernate.MappingException: 
Could not determine type for: String, at table: identity_providers, for columns: [org.hibernate.mapping.Column(issuer)]
at com.oracle.hibernate.OCIAuthManager.setupTenant1Pdb1(OCIAuthManager.java:246)
at com.oracle.hibernate.OCIAuthManager.main(OCIAuthManager.java:283)

Я не могу понять, почему.Applications.java это здесь .IdentityProviders.java - здесь .ApplicationIdentityProviders.java - здесь .Основной класс здесь .Applications.hbm.xml выглядит следующим образом:

<?xml version = "1.0" encoding = "utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC 
"-//Hibernate/Hibernate Mapping DTD//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> 

<hibernate-mapping>
 <class name = "Applications" table = "applications">

  <meta attribute = "class-description">
     This class contains the applications details. 
  </meta>

  <id name = "application_id" type = "int" column = "application_id">
     <generator class="native"/>
  </id>

  <set name = "application_id_providers" cascade="save-update" table="application_identity_providers">
     <key column = "application_id"/>
     <many-to-many column = "identity_provider_id" class="IdentityProviders"/>
  </set>

  <property name = "application_name" column = "application_name" type = "string"/>
  <property name = "attr" column = "attr" type = "short"/>
  <property name = "salary" column = "salary" type = "int"/>

 </class>

 <class name = "IdentityProviders" table = "identity_providers">

  <meta attribute = "class-description">
     This class contains the identity providers' details. 
  </meta>

  <id name = "identity_provider_id" type = "int" column = "identity_provider_id">
     <generator class="native"/>
  </id>

  <property name = "attr" column = "attr" type = "short"/>
  <property name = "protocols" column = "protocols" type = "short"/>
  <property name = "issuer" column = "issuer" type = "String"/>
  <property name = "cert1" column = "cert1" type = "String"/>
  <property name = "cert2" column = "cert2" type = "String"/>
  <property name = "is_tenant_default" column = "is_tenant_default" type = "short"/>

 </class>

</hibernate-mapping>

hibernate-1.cfg.xml выглядит следующим образом:

<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
    "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
    "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
  <session-factory>

    <property name="connection.url">jdbc:oracle:thin:@//localhost:1521/t1p1.oradev.oraclecorp.com</property>
    <property name="connection.username">username</property>
    <property name="connection.password">password</property>
    <property name="connection.driver_class">oracle.jdbc.OracleDriver</property>
    <property name="dialect">org.hibernate.dialect.Oracle12cDialect</property>

    <property name="show_sql">true</property>

    <property name="format_sql">true</property>
    <property name="hbm2ddl.auto">create</property>

    <!-- JDBC connection pool (use the built-in) -->
    <property name="connection.pool_size">1</property>
    <property name="current_session_context_class">thread</property>

    <mapping resource = "Applications.hbm.xml"/>

  </session-factory>
</hibernate-configuration>

Где я иду не так?Пожалуйста, помогите.

Обновление 1: ответ @ Гийома сработал.Но теперь я получаю следующую ошибку:

Exception in thread "main" java.lang.RuntimeException:
org.hibernate.MappingException: An association from the table 
application_identity_providers refers to an unmapped class: 
IdentityProviders at
com.oracle.hibernate.OCIAuthManager.setupTenant1Pdb1(OCIAuthManager.java:246) 
at com.oracle.hibernate.OCIAuthManager.main(OCIAuthManager.java:283)

Обновление 2: я исправил ошибку в обновлении 1. Теперь я получаю следующую ошибку: ОШИБКА: ORA-01400: невозможно вставить NULL в ("SYS"." APPLICATION_IDENTITY_PROVIDERS "." APPLICATION_IDENTITY_PROVIDER_ID ")

Эта ошибка возникает в промежуточной таблице, содержащей первичные ключи двух других таблиц, которые имеют сопоставление" многие ко многим ".В ApplicationIdentityProviders.java у меня есть:

@Id
@SequenceGenerator(name="seq",sequenceName="oracle_seq",allocationSize=1)        
@GeneratedValue(strategy=GenerationType.SEQUENCE, generator="seq")
private int application_identity_provider_id;

Когда я запускаю программу, соответствующий выходной фрагмент выглядит так: Hibernate:

create table application_identity_providers (
application_identity_provider_id number(10,0) not null,
application_id number(10,0) not null,
identity_provider_id number(10,0) not null,
is_application_default number(5,0) not null,
primary key (application_id, identity_provider_id)
)

Где я иду не так?

Обновление 3: учитывая имя "application_name" в таблице "Applications", получить весь контент в таблицах "identity_providers" и "saml_identity_providers".Как это сделать?

Я попробовал следующее:

Query query = entitymanager.createQuery("select a from Applications a join 
fetch a.ApplicationIdentityProviders ap " +
"join fetch ap.IdentityProviders ip where a.application_name = 
'newApplicationName'");

Это дает мне эту ошибку:

org.springframework.beans.factory.BeanCreationException: Error creating bean 
with name 'applicationsRepository' defined in file 
[D:\filepath\ApplicationsRepository.class]: Bean instantiation via 
constructor failed; nested exception is 
org.springframework.beans.BeanInstantiationException: Failed to instantiate 
[ApplicationsRepository]: Constructor threw exception; nested exception is 
java.lang.IllegalArgumentException: org.hibernate.QueryException: could not 
resolve property: ApplicationIdentityProviders of: Applications [select a 
from com.package.entities.Applications a join fetch 
a.ApplicationIdentityProviders ap join fetch 
ap.IdentityProviders ip where a.application_name = 'newApplicationName']

Ответы [ 3 ]

0 голосов
/ 28 сентября 2018

Более быстрый способ - использовать инструменты Hibernate для автоматического создания файла сопоставления.Пожалуйста, обратитесь к следующим статьям.

Как установить Hibernate Tools в Eclipse? http://www.mkyong.com/hibernate/how-to-generate-code-with-hibernate-tools/

0 голосов
/ 10 октября 2018

In Application.java Удалите следующую строку:

private Set<IdentityProviders> application_id_providers;

Добавьте следующие строки:

@ManyToMany(cascade = { 
    CascadeType.PERSIST, 
    CascadeType.MERGE
})
@JoinTable(name = "application_identity_providers",
    joinColumns = @JoinColumn(name = "application_id"),
    inverseJoinColumns = @JoinColumn(name = "identity_provider_id")
)
private Set<IdentityProviders> id_providers;

В IdentityProviders.java Добавьте следующие строки:

@ManyToMany(mappedBy = "id_providers")
private Set<Applications> applications;

@OneToMany(
    cascade = CascadeType.ALL, 
    orphanRemoval = true
)
private Set<SamlIdentityProviders> si_providers;

Тогда запрос может быть:

SELECT a from Applications a JOIN fetch a.id_providers ip JOIN fetch ip.si_providers sip 
WHERE a.application_name = 'newApplicationName' AND ip.issuer = 'issuerName'
0 голосов
/ 28 сентября 2018

Я думаю, что ваш тип должен быть строчной строкой (не String) для этих строк здесь:

<property name = "issuer" column = "issuer" type = "string"/>
<property name = "cert1" column = "cert1" type = "string"/>
<property name = "cert2" column = "cert2" type = "string"/>

Я думаю, что вам также нужно объявить квалифицированные имена ваших классов Java в ваших файлах отображенияэто можно сделать, добавив имя пакета в элемент отображения hibernate:

<hibernate-mapping package="com.oracle.hibernate">
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...