Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

HHH-18880 proof of concept for formula support in @ManyToMany #5976

Draft
wants to merge 1 commit into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -433,6 +433,26 @@ static AnnotatedJoinColumn buildExplicitJoinTableJoinColumn(
return column;
}

static AnnotatedJoinColumn buildExplicitJoinTableJoinFormula(
AnnotatedJoinColumns parent,
PropertyHolder propertyHolder,
PropertyData inferredData,
JoinFormula joinFormula) {
final AnnotatedJoinColumn column = new AnnotatedJoinColumn();
column.setImplicit( true );
// column.setPropertyHolder( propertyHolder );
// column.setPropertyName( getRelativePath( propertyHolder, propertyName ) );
// column.setJoins( secondaryTables );
// column.setContext( context );
column.setNullable( false ); //I break the spec, but it's for good
//done after the annotation to override it
column.setParent( parent );
column.setFormula( joinFormula.value() );
column.setReferencedColumn( joinFormula.referencedColumnName() );
column.bind();
return column;
}

@Override
public String toString() {
final StringBuilder string = new StringBuilder();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -209,6 +209,26 @@ public static AnnotatedJoinColumns buildJoinTableJoinColumns(
return parent;
}

/**
* Called for join tables in {@link jakarta.persistence.ManyToMany} associations.
*/
public static AnnotatedJoinColumns buildJoinTableJoinFormula(
JoinFormula joinFormula,
Map<String, Join> secondaryTables,
PropertyHolder propertyHolder,
PropertyData inferredData,
String mappedBy,
MetadataBuildingContext context) {
final AnnotatedJoinColumns parent = new AnnotatedJoinColumns();
parent.setBuildingContext( context );
parent.setJoins( secondaryTables );
parent.setPropertyHolder( propertyHolder );
parent.setPropertyName( getRelativePath( propertyHolder, inferredData.getPropertyName() ) );
parent.setMappedBy( mappedBy );
AnnotatedJoinColumn.buildExplicitJoinTableJoinFormula( parent, propertyHolder, inferredData, joinFormula );
return parent;
}

public List<AnnotatedJoinColumn> getJoinColumns() {
return columns;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@
import org.hibernate.annotations.Formula;
import org.hibernate.annotations.HQLSelect;
import org.hibernate.annotations.Immutable;
import org.hibernate.annotations.JoinFormula;
import org.hibernate.annotations.LazyCollection;
import org.hibernate.annotations.LazyCollectionOption;
import org.hibernate.annotations.LazyGroup;
Expand Down Expand Up @@ -149,6 +150,7 @@
import static org.hibernate.boot.model.internal.AnnotatedColumn.buildFormulaFromAnnotation;
import static org.hibernate.boot.model.internal.AnnotatedJoinColumns.buildJoinColumnsWithDefaultColumnSuffix;
import static org.hibernate.boot.model.internal.AnnotatedJoinColumns.buildJoinTableJoinColumns;
import static org.hibernate.boot.model.internal.AnnotatedJoinColumns.buildJoinTableJoinFormula;
import static org.hibernate.boot.model.internal.BinderHelper.buildAnyValue;
import static org.hibernate.boot.model.internal.GeneratorBinder.buildGenerators;
import static org.hibernate.boot.model.internal.BinderHelper.createSyntheticPropertyReference;
Expand Down Expand Up @@ -641,6 +643,7 @@ private static void bindJoinedTableAssociation(
final CollectionTable collectionTable = property.getAnnotation( CollectionTable.class );
final JoinColumn[] annJoins;
final JoinColumn[] annInverseJoins;
final JoinFormula annJoinFormula;
if ( assocTable != null || collectionTable != null ) {

final String catalog;
Expand Down Expand Up @@ -689,10 +692,25 @@ private static void bindJoinedTableAssociation(
//set check constraint in the second pass
annJoins = joins.length == 0 ? null : joins;
annInverseJoins = inverseJoins == null || inverseJoins.length == 0 ? null : inverseJoins;
annJoinFormula = null;
}
else {
if ( property.isAnnotationPresent(JoinFormula.class) ) {
annJoinFormula = property.getAnnotation(JoinFormula.class);
}
else {
annJoinFormula = null;
}
if ( property.isAnnotationPresent(JoinColumns.class) ) {
annInverseJoins = property.getAnnotation(JoinColumns.class).value();
}
else if ( property.isAnnotationPresent(JoinColumn.class) ) {
annInverseJoins = new JoinColumn[] { property.getAnnotation(JoinColumn.class) };
}
else {
annInverseJoins = null;
}
annJoins = null;
annInverseJoins = null;
}
associationTableBinder.setBuildingContext( buildingContext );
collectionBinder.setTableBinder( associationTableBinder );
Expand All @@ -704,14 +722,26 @@ private static void bindJoinedTableAssociation(
mappedBy,
buildingContext
) );
collectionBinder.setInverseJoinColumns( buildJoinTableJoinColumns(
annInverseJoins,
entityBinder.getSecondaryTables(),
propertyHolder,
inferredData,
mappedBy,
buildingContext
) );
if ( annJoinFormula != null ) {
collectionBinder.setInverseJoinColumns( buildJoinTableJoinFormula(
annJoinFormula,
entityBinder.getSecondaryTables(),
propertyHolder,
inferredData,
mappedBy,
buildingContext
) );
}
else {
collectionBinder.setInverseJoinColumns( buildJoinTableJoinColumns(
annInverseJoins,
entityBinder.getSecondaryTables(),
propertyHolder,
inferredData,
mappedBy,
buildingContext
) );
}
}

protected MetadataBuildingContext getBuildingContext() {
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
/*
* Hibernate, Relational Persistence for Idiomatic Java
*
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
*/
package org.hibernate.orm.test.annotations.formula;

import jakarta.persistence.Column;
import jakarta.persistence.Entity;
import jakarta.persistence.Id;
import jakarta.persistence.ManyToMany;
import jakarta.persistence.Table;
import org.hibernate.annotations.JoinFormula;
import org.hibernate.testing.orm.junit.DomainModel;
import org.hibernate.testing.orm.junit.SessionFactory;
import org.hibernate.testing.orm.junit.SessionFactoryScope;
import org.junit.jupiter.api.Test;

import java.util.HashSet;
import java.util.Set;

import static org.junit.jupiter.api.Assertions.assertEquals;

@SessionFactory
@DomainModel(annotatedClasses = {JoinColumnOrFormula2Test.A.class, JoinColumnOrFormula2Test.D.class})
public class JoinColumnOrFormula2Test {
@Test
public void test(SessionFactoryScope scope) {
scope.inTransaction(s -> {
A a = new A();
a.id = 3;
a.dId = 5;
D d = new D();
d.id = 5;
s.persist(a);
s.persist(d);
a.ds.add(d);
});
scope.inSession(s -> {
Set<D> ds = s.get(A.class, 3).ds;
assertEquals(1, ds.size());
});
}

@Entity( name = "A" )
@Table( name = "A" )
public static class A {
@Id
@Column( name = "aid")
public Integer id;

@Column( name = "did")
public Integer dId;

@ManyToMany
@JoinFormula(value = "A_aid+2", referencedColumnName = "did")
Set<D> ds = new HashSet<>();
}

@Entity( name = "D" )
@Table( name = "D" )
public static class D {
@Id
@Column( name = "did")
public Integer id;
}
}