HHH-11089 - Naming Strategy Does not affect the user-specified index/foreign-key names
Call the implicit naming strategy regardless if the client has specified foreign key name or not
This commit is contained in:
parent
e0da35e5e8
commit
e654e6c2c7
|
@ -1811,45 +1811,49 @@ public class InFlightMetadataCollectorImpl implements InFlightMetadataCollector
|
||||||
|
|
||||||
fk.setReferencedTable( referencedClass.getTable() );
|
fk.setReferencedTable( referencedClass.getTable() );
|
||||||
|
|
||||||
// todo : should we apply a physical naming too?
|
Identifier nameIdentifier;
|
||||||
if ( fk.getName() == null ) {
|
|
||||||
final Identifier nameIdentifier = getMetadataBuildingOptions().getImplicitNamingStrategy().determineForeignKeyName(
|
|
||||||
new ImplicitForeignKeyNameSource() {
|
|
||||||
final List<Identifier> columnNames = extractColumnNames( fk.getColumns() );
|
|
||||||
List<Identifier> referencedColumnNames = null;
|
|
||||||
|
|
||||||
@Override
|
ImplicitForeignKeyNameSource foreignKeyNameSource = new ImplicitForeignKeyNameSource() {
|
||||||
public Identifier getTableName() {
|
final List<Identifier> columnNames = extractColumnNames( fk.getColumns() );
|
||||||
return table.getNameIdentifier();
|
List<Identifier> referencedColumnNames = null;
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public List<Identifier> getColumnNames() {
|
public Identifier getTableName() {
|
||||||
return columnNames;
|
return table.getNameIdentifier();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Identifier getReferencedTableName() {
|
public List<Identifier> getColumnNames() {
|
||||||
return fk.getReferencedTable().getNameIdentifier();
|
return columnNames;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public List<Identifier> getReferencedColumnNames() {
|
public Identifier getReferencedTableName() {
|
||||||
if ( referencedColumnNames == null ) {
|
return fk.getReferencedTable().getNameIdentifier();
|
||||||
referencedColumnNames = extractColumnNames( fk.getReferencedColumns() );
|
}
|
||||||
}
|
|
||||||
return referencedColumnNames;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public MetadataBuildingContext getBuildingContext() {
|
public List<Identifier> getReferencedColumnNames() {
|
||||||
return buildingContext;
|
if ( referencedColumnNames == null ) {
|
||||||
}
|
referencedColumnNames = extractColumnNames( fk.getReferencedColumns() );
|
||||||
}
|
}
|
||||||
);
|
return referencedColumnNames;
|
||||||
|
}
|
||||||
|
|
||||||
fk.setName( nameIdentifier.render( getDatabase().getJdbcEnvironment().getDialect() ) );
|
@Override
|
||||||
}
|
public Identifier getUserProvidedIdentifier() {
|
||||||
|
return fk.getName() != null ? Identifier.toIdentifier( fk.getName() ) : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MetadataBuildingContext getBuildingContext() {
|
||||||
|
return buildingContext;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
nameIdentifier = getMetadataBuildingOptions().getImplicitNamingStrategy().determineForeignKeyName(foreignKeyNameSource);
|
||||||
|
|
||||||
|
fk.setName( nameIdentifier.render( getDatabase().getJdbcEnvironment().getDialect() ) );
|
||||||
|
|
||||||
fk.alignColumns();
|
fk.alignColumns();
|
||||||
}
|
}
|
||||||
|
@ -1959,34 +1963,39 @@ public class InFlightMetadataCollectorImpl implements InFlightMetadataCollector
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
final String originalKeyName = keyName;
|
||||||
|
|
||||||
if ( unique ) {
|
if ( unique ) {
|
||||||
if ( StringHelper.isEmpty( keyName ) ) {
|
final Identifier keyNameIdentifier = getMetadataBuildingOptions().getImplicitNamingStrategy().determineUniqueKeyName(
|
||||||
final Identifier keyNameIdentifier = getMetadataBuildingOptions().getImplicitNamingStrategy().determineUniqueKeyName(
|
new ImplicitUniqueKeyNameSource() {
|
||||||
new ImplicitUniqueKeyNameSource() {
|
@Override
|
||||||
@Override
|
public MetadataBuildingContext getBuildingContext() {
|
||||||
public MetadataBuildingContext getBuildingContext() {
|
return buildingContext;
|
||||||
return buildingContext;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Identifier getTableName() {
|
public Identifier getTableName() {
|
||||||
return table.getNameIdentifier();
|
return table.getNameIdentifier();
|
||||||
}
|
}
|
||||||
|
|
||||||
private List<Identifier> columnNameIdentifiers;
|
private List<Identifier> columnNameIdentifiers;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public List<Identifier> getColumnNames() {
|
public List<Identifier> getColumnNames() {
|
||||||
// be lazy about building these
|
// be lazy about building these
|
||||||
if ( columnNameIdentifiers == null ) {
|
if ( columnNameIdentifiers == null ) {
|
||||||
columnNameIdentifiers = toIdentifiers( columnNames );
|
columnNameIdentifiers = toIdentifiers( columnNames );
|
||||||
}
|
|
||||||
return columnNameIdentifiers;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
);
|
return columnNameIdentifiers;
|
||||||
keyName = keyNameIdentifier.render( getDatabase().getJdbcEnvironment().getDialect() );
|
}
|
||||||
}
|
|
||||||
|
@Override
|
||||||
|
public Identifier getUserProvidedIdentifier() {
|
||||||
|
return originalKeyName != null ? Identifier.toIdentifier( originalKeyName ) : null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
keyName = keyNameIdentifier.render( getDatabase().getJdbcEnvironment().getDialect() );
|
||||||
|
|
||||||
UniqueKey uk = table.getOrCreateUniqueKey( keyName );
|
UniqueKey uk = table.getOrCreateUniqueKey( keyName );
|
||||||
for ( int i = 0; i < columns.length; i++ ) {
|
for ( int i = 0; i < columns.length; i++ ) {
|
||||||
|
@ -1999,33 +2008,36 @@ public class InFlightMetadataCollectorImpl implements InFlightMetadataCollector
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if ( StringHelper.isEmpty( keyName ) ) {
|
final Identifier keyNameIdentifier = getMetadataBuildingOptions().getImplicitNamingStrategy().determineIndexName(
|
||||||
final Identifier keyNameIdentifier = getMetadataBuildingOptions().getImplicitNamingStrategy().determineIndexName(
|
new ImplicitIndexNameSource() {
|
||||||
new ImplicitIndexNameSource() {
|
@Override
|
||||||
@Override
|
public MetadataBuildingContext getBuildingContext() {
|
||||||
public MetadataBuildingContext getBuildingContext() {
|
return buildingContext;
|
||||||
return buildingContext;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Identifier getTableName() {
|
public Identifier getTableName() {
|
||||||
return table.getNameIdentifier();
|
return table.getNameIdentifier();
|
||||||
}
|
}
|
||||||
|
|
||||||
private List<Identifier> columnNameIdentifiers;
|
private List<Identifier> columnNameIdentifiers;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public List<Identifier> getColumnNames() {
|
public List<Identifier> getColumnNames() {
|
||||||
// be lazy about building these
|
// be lazy about building these
|
||||||
if ( columnNameIdentifiers == null ) {
|
if ( columnNameIdentifiers == null ) {
|
||||||
columnNameIdentifiers = toIdentifiers( columnNames );
|
columnNameIdentifiers = toIdentifiers( columnNames );
|
||||||
}
|
|
||||||
return columnNameIdentifiers;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
);
|
return columnNameIdentifiers;
|
||||||
keyName = keyNameIdentifier.render( getDatabase().getJdbcEnvironment().getDialect() );
|
}
|
||||||
}
|
|
||||||
|
@Override
|
||||||
|
public Identifier getUserProvidedIdentifier() {
|
||||||
|
return originalKeyName != null ? Identifier.toIdentifier( originalKeyName ) : null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
keyName = keyNameIdentifier.render( getDatabase().getJdbcEnvironment().getDialect() );
|
||||||
|
|
||||||
Index index = table.getOrCreateIndex( keyName );
|
Index index = table.getOrCreateIndex( keyName );
|
||||||
for ( int i = 0; i < columns.length; i++ ) {
|
for ( int i = 0; i < columns.length; i++ ) {
|
||||||
|
|
|
@ -16,4 +16,5 @@ import java.util.List;
|
||||||
public interface ImplicitConstraintNameSource extends ImplicitNameSource {
|
public interface ImplicitConstraintNameSource extends ImplicitNameSource {
|
||||||
public Identifier getTableName();
|
public Identifier getTableName();
|
||||||
public List<Identifier> getColumnNames();
|
public List<Identifier> getColumnNames();
|
||||||
|
public Identifier getUserProvidedIdentifier();
|
||||||
}
|
}
|
||||||
|
|
|
@ -198,7 +198,8 @@ public class ImplicitNamingStrategyJpaCompliantImpl implements ImplicitNamingStr
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Identifier determineForeignKeyName(ImplicitForeignKeyNameSource source) {
|
public Identifier determineForeignKeyName(ImplicitForeignKeyNameSource source) {
|
||||||
return toIdentifier(
|
Identifier userProvidedIdentifier = source.getUserProvidedIdentifier();
|
||||||
|
return userProvidedIdentifier != null ? userProvidedIdentifier : toIdentifier(
|
||||||
NamingHelper.INSTANCE.generateHashedFkName(
|
NamingHelper.INSTANCE.generateHashedFkName(
|
||||||
"FK",
|
"FK",
|
||||||
source.getTableName(),
|
source.getTableName(),
|
||||||
|
@ -211,7 +212,8 @@ public class ImplicitNamingStrategyJpaCompliantImpl implements ImplicitNamingStr
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Identifier determineUniqueKeyName(ImplicitUniqueKeyNameSource source) {
|
public Identifier determineUniqueKeyName(ImplicitUniqueKeyNameSource source) {
|
||||||
return toIdentifier(
|
Identifier userProvidedIdentifier = source.getUserProvidedIdentifier();
|
||||||
|
return userProvidedIdentifier != null ? userProvidedIdentifier : toIdentifier(
|
||||||
NamingHelper.INSTANCE.generateHashedConstraintName(
|
NamingHelper.INSTANCE.generateHashedConstraintName(
|
||||||
"UK",
|
"UK",
|
||||||
source.getTableName(),
|
source.getTableName(),
|
||||||
|
@ -223,7 +225,8 @@ public class ImplicitNamingStrategyJpaCompliantImpl implements ImplicitNamingStr
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Identifier determineIndexName(ImplicitIndexNameSource source) {
|
public Identifier determineIndexName(ImplicitIndexNameSource source) {
|
||||||
return toIdentifier(
|
Identifier userProvidedIdentifier = source.getUserProvidedIdentifier();
|
||||||
|
return userProvidedIdentifier != null ? userProvidedIdentifier : toIdentifier(
|
||||||
NamingHelper.INSTANCE.generateHashedConstraintName(
|
NamingHelper.INSTANCE.generateHashedConstraintName(
|
||||||
"IDX",
|
"IDX",
|
||||||
source.getTableName(),
|
source.getTableName(),
|
||||||
|
|
|
@ -4345,6 +4345,11 @@ public class ModelBinder {
|
||||||
public MetadataBuildingContext getBuildingContext() {
|
public MetadataBuildingContext getBuildingContext() {
|
||||||
return mappingDocument;
|
return mappingDocument;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Identifier getUserProvidedIdentifier() {
|
||||||
|
return uk.getName() != null ? Identifier.toIdentifier( uk.getName() ) : null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
uk.setName( ukName.render( mappingDocument.getMetadataCollector().getDatabase().getDialect() ) );
|
uk.setName( ukName.render( mappingDocument.getMetadataCollector().getDatabase().getDialect() ) );
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
/*
|
||||||
|
* 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.test.annotations.namingstrategy;
|
||||||
|
|
||||||
|
import org.hibernate.boot.model.naming.Identifier;
|
||||||
|
import org.hibernate.boot.model.naming.ImplicitForeignKeyNameSource;
|
||||||
|
import org.hibernate.boot.model.naming.ImplicitIndexNameSource;
|
||||||
|
import org.hibernate.boot.model.naming.ImplicitNamingStrategyJpaCompliantImpl;
|
||||||
|
import org.hibernate.boot.model.naming.ImplicitUniqueKeyNameSource;
|
||||||
|
|
||||||
|
@SuppressWarnings("serial")
|
||||||
|
public class LongIdentifierNamingStrategy
|
||||||
|
extends ImplicitNamingStrategyJpaCompliantImpl {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Identifier determineForeignKeyName(ImplicitForeignKeyNameSource source) {
|
||||||
|
return limitIdentifierName(super.determineForeignKeyName( source ));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Identifier determineUniqueKeyName(ImplicitUniqueKeyNameSource source) {
|
||||||
|
return limitIdentifierName(super.determineUniqueKeyName( source ));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Identifier determineIndexName(ImplicitIndexNameSource source) {
|
||||||
|
return limitIdentifierName(super.determineIndexName( source ));
|
||||||
|
}
|
||||||
|
|
||||||
|
public Identifier limitIdentifierName(Identifier identifier) {
|
||||||
|
String text = identifier.getText();
|
||||||
|
if(text.length() > 30) {
|
||||||
|
return new Identifier( text.substring( 0, 30 ), identifier.isQuoted() );
|
||||||
|
}
|
||||||
|
return identifier;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,149 @@
|
||||||
|
/*
|
||||||
|
* 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>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// $Id$
|
||||||
|
package org.hibernate.test.annotations.namingstrategy;
|
||||||
|
|
||||||
|
import javax.persistence.Entity;
|
||||||
|
import javax.persistence.ForeignKey;
|
||||||
|
import javax.persistence.Id;
|
||||||
|
import javax.persistence.Index;
|
||||||
|
import javax.persistence.JoinColumn;
|
||||||
|
import javax.persistence.ManyToOne;
|
||||||
|
import javax.persistence.Table;
|
||||||
|
import javax.persistence.UniqueConstraint;
|
||||||
|
|
||||||
|
import org.hibernate.boot.Metadata;
|
||||||
|
import org.hibernate.boot.MetadataSources;
|
||||||
|
import org.hibernate.cfg.Environment;
|
||||||
|
import org.hibernate.mapping.UniqueKey;
|
||||||
|
import org.hibernate.service.ServiceRegistry;
|
||||||
|
|
||||||
|
import org.hibernate.testing.ServiceRegistryBuilder;
|
||||||
|
import org.hibernate.testing.TestForIssue;
|
||||||
|
import org.hibernate.testing.junit4.BaseUnitTestCase;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import org.jboss.logging.Logger;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test harness for HHH-11089.
|
||||||
|
*
|
||||||
|
* @author Vlad Mihalcea
|
||||||
|
*/
|
||||||
|
@TestForIssue( jiraKey = "HHH-11089" )
|
||||||
|
public class LongKeyNamingStrategyTest extends BaseUnitTestCase {
|
||||||
|
private static final Logger log = Logger.getLogger( LongKeyNamingStrategyTest.class );
|
||||||
|
|
||||||
|
private ServiceRegistry serviceRegistry;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
serviceRegistry = ServiceRegistryBuilder.buildServiceRegistry( Environment.getProperties() );
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() {
|
||||||
|
if ( serviceRegistry != null ) {
|
||||||
|
ServiceRegistryBuilder.destroy( serviceRegistry );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testWithCustomNamingStrategy() throws Exception {
|
||||||
|
Metadata metadata = new MetadataSources( serviceRegistry )
|
||||||
|
.addAnnotatedClass(Address.class)
|
||||||
|
.addAnnotatedClass(Person.class)
|
||||||
|
.getMetadataBuilder()
|
||||||
|
.applyImplicitNamingStrategy( new LongIdentifierNamingStrategy() )
|
||||||
|
.build();
|
||||||
|
|
||||||
|
org.hibernate.mapping.ForeignKey foreignKey =
|
||||||
|
(org.hibernate.mapping.ForeignKey) metadata.getEntityBinding( Address.class.getName()).getTable().getForeignKeyIterator().next();
|
||||||
|
assertEquals( "FK_way_longer_than_the_30_char", foreignKey.getName() );
|
||||||
|
|
||||||
|
UniqueKey uniqueKey = metadata.getEntityBinding( Address.class.getName()).getTable().getUniqueKeyIterator().next();
|
||||||
|
assertEquals( "UK_way_longer_than_the_30_char", uniqueKey.getName() );
|
||||||
|
|
||||||
|
org.hibernate.mapping.Index index = metadata.getEntityBinding( Address.class.getName()).getTable().getIndexIterator().next();
|
||||||
|
assertEquals( "IDX_way_longer_than_the_30_cha", index.getName() );
|
||||||
|
}
|
||||||
|
|
||||||
|
@Entity(name = "Address")
|
||||||
|
@Table(uniqueConstraints = @UniqueConstraint(
|
||||||
|
name = "UK_way_longer_than_the_30_characters_limit",
|
||||||
|
columnNames = {
|
||||||
|
"city", "streetName", "streetNumber"
|
||||||
|
}),
|
||||||
|
indexes = @Index( name = "IDX_way_longer_than_the_30_characters_limit", columnList = "city, streetName, streetNumber")
|
||||||
|
)
|
||||||
|
public class Address {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
private Long id;
|
||||||
|
|
||||||
|
private String city;
|
||||||
|
|
||||||
|
private String streetName;
|
||||||
|
|
||||||
|
private String streetNumber;
|
||||||
|
|
||||||
|
@ManyToOne
|
||||||
|
@JoinColumn(name = "person_id", foreignKey = @ForeignKey(name = "FK_way_longer_than_the_30_characters_limit"))
|
||||||
|
private Person person;
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(Long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCity() {
|
||||||
|
return city;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCity(String city) {
|
||||||
|
this.city = city;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getStreetName() {
|
||||||
|
return streetName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setStreetName(String streetName) {
|
||||||
|
this.streetName = streetName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getStreetNumber() {
|
||||||
|
return streetNumber;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setStreetNumber(String streetNumber) {
|
||||||
|
this.streetNumber = streetNumber;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Entity
|
||||||
|
public class Person {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
private long id;
|
||||||
|
|
||||||
|
public long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue