javahibernatejpa

Why doesn't INET_NTOA work in ColumnTransformer?


I have a MySQL table that stores information about label printers:

ip port nickname
181012304 9100 PRLMA213
181012370 9100 PRLMA212
181012398 9100 PRLMA211
181012559 9100 PRLMA210

There is a corresponding JPA entity for this table in my application:

@Entity
@Table(name = "printers", catalog = "gage_requests")
public class PrinterEntity {

    @Id
    @Column(name = "ip")
    @ColumnTransformer(read = "INET_NTOA(ip)", write = "INET_ATON(?)")
    private String address;
    private int port;
    private String nickname;

    public String getAddress() {
        return this.address;
    }

    public int getPort() {
        return this.port;
    }

    public String getNickname() {
        return this.nickname;
    }

}

When I use CriteriaQuery to retrieve the list of printers, the IP address is not properly converted and remains an integer:

public List<PrinterEntity> findPrinters() {
    CriteriaBuilder criteriaBuilder = this.entityManager.getCriteriaBuilder();
    CriteriaQuery<PrinterEntity> criteriaQuery = criteriaBuilder.createQuery(PrinterEntity.class);

    Root<PrinterEntity> root = criteriaQuery.from(PrinterEntity.class);
    criteriaQuery.select(root);

    TypedQuery<PrinterEntity> query = this.entityManager.createQuery(criteriaQuery);
    return query.getResultStream().peek(this.entityManager::detach)
            .collect(Collectors.toCollection(LinkedList::new));
}

Returned list:

[
    {
        "address": "181012304",
        "nickname": "PRLMA213",
        "port": 9100
    },
    {
        "address": "181012370",
        "nickname": "PRLMA212",
        "port": 9100
    },
    {
        "address": "181012398",
        "nickname": "PRLMA211",
        "port": 9100
    },
    {
        "address": "181012559",
        "nickname": "PRLMA210",
        "port": 9100
    }
]

With hibernate.show_sql set to true, the following SQL is logged:

select printerent0_.ip as ip1_5_, printerent0_.nickname as nickname2_5_, printerent0_.port as port3_5_ from gage_requests.printers printerent0_

This is clearly missing the INET_NTOA function used in my @ColumnTransformer annotation, so it appears that the annotation is not being applied.

Why is this?


Solution

  • @ColumnTransformer not supported at @Id fields

    Hibernate's @ColumnTransformer might not be supported for a primary-key field as given by the annotation @Id here.

    This was rejected in feature-request HHH-9808 by the core-maintainer Steve Ebersole in 2016:

    @ColumnTransformer is not applied on @Id members

    Ok then, until this gets some votes or something else changes, going to reject this.

    For a detailed explanation see: Why can't we apply @ColumnTransformer to @Id field (primary key)? .

    Related

    This restriction was also be the issue in this case: @ColumnTransformer isn't used in select . The solution was to use a different column as primary-key with @Id.

    Workarounds

    To get the transformation for the IP-address working, I could think of these workarounds:

    1. Fix the source: Reformat the persisted IP-addresses in the source-table, e.g. ensure that the column-type is VARCHAR and execute SQL UPDATE TABLE gage_requests.printers SET ip = INET_NTOA(ip). Then you need no transformation in Java but a validation on the address field, to ensure the correct format.
    2. Transform in Java: Port this internet-address/IP transform-functions INET_NTOA and INET_ATON into the Java getters/setters for field address.
    3. Change primary-key: Use a different column as primary-key, e.g. move @Id to nickname