Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Populate child bean with Transformers.aliasToBean in Hibernate

Tags:

java

hibernate

I have the next couple of beans:

Address {
    String name;
    String number;
    String zipcode;
    String town;
}

MyEntity {
    Address address;
    String value1;
    String value2;
}

I'm trying to do the next Hibernate query:

private final List<String> propertiesDistinct = Arrays.asList("address.name");
private final List<String> properties = Arrays.asList("address.number",
        "address.zipcode", "address.town")

ProjectionList projectionList = Projections.projectionList();

if (propertiesDistinct != null) {
    ProjectionList projectionListDistinct = Projections.projectionList();
for (String propertyDistinct : propertiesDistinct)
         projectionListDistinct.add(Projections.property(propertyDistinct).as(propertyDistinct));

    projectionList.add(Projections.distinct(projectionListAgrupar));
}

if (properties != null)
    for (String property : properties)
         projectionList.add(Projections.property(property).as(property));
criterio.setProjection(projectionList);

// MORE FILTERS ON MyEntity FIELDS
//... criterio.add(Restrinctions...);

// I want to recover the results on my bean MyEntity so I don't have to create a new one
criterio.setResultTransformer(Transformers.aliasToBean(MyEntity.class));

Problem:

Caused by: org.hibernate.PropertyNotFoundException: Could not find setter for address.name on class com.entities.MyEntity

I understand that Hibernate is looking for something like:

public String getAddressName() {} // This should be in MyEntity

Instead of:

public String getName() {} // In my Address bean

Ideas about how can I fix this without creating a new bean?

Thanks!

like image 437
maqjav Avatar asked Sep 10 '13 07:09

maqjav


2 Answers

I wrote a ResultTransformer that can fix your problem. It's name is AliasToBeanNestedResultTransformer, check it out on github.

like image 190
Sami Andoni Avatar answered Nov 08 '22 11:11

Sami Andoni


Code provided in Github works fine but there is change in import for new versions of hibernate. Its as follow.

org.hibernate.property.PropertyAccessor replaced byorg.hibernate.property.access.spi.PropertyAccess

and

org.hibernate.property.PropertyAccessorFactory replaced by org.hibernate.property.access.internal.PropertyAccessStrategyBasicImpl

So you'll have change the code from

PropertyAccessor accessor = PropertyAccessorFactory.getPropertyAccessor("property");
accessor.getSetter(resultClass, (String)subclassToAlias.get(subclass).get(2)).set(root, subObject, null);

to

PropertyAccess propertyAccess = PropertyAccessStrategyBasicImpl.INSTANCE.buildPropertyAccess(resultClass, (String)subclassToAlias.get(subclass).get(2));
propertyAccess.getSetter().set(root, subObject, null);
like image 26
Vicky Thakor Avatar answered Nov 08 '22 11:11

Vicky Thakor