Как использовать @ConstructorResult с полем Set<SomeEnum>

Я пытаюсь создать @NamedNativeQuery с @ConstructorResult для класса, который имеет поле с Set из enum ценности.

VeterinarianJPA.java:

@Entity
@Table(name = "veterinarians")
@Setter
@Getter
@NoArgsConstructor
@NamedNativeQueries({
        @NamedNativeQuery(
                name = VeterinarianJPA.FIND_ALL_VETS,
                query = "SELECT v.id, v.name, vs.specialisations " +
                        "FROM veterinarians v " +
                        "JOIN veterinarian_specialisations vs ON v.id = vs.vet_id",
                resultSetMapping = VeterinarianJPA.VETERINARIAN_RESULT_MAPPER
        )})
@SqlResultSetMappings({
        @SqlResultSetMapping(
                name = VeterinarianJPA.VETERINARIAN_RESULT_MAPPER,
                classes = @ConstructorResult(
                        targetClass = Veterinarian.class,
                        columns = {
                                @ColumnResult(name = "id", type = Long.class),
                                @ColumnResult(name = "name"),
                                @ColumnResult(name = "specialisations", type = Set.class)
                        }
                )
        )})
class VeterinarianJPA {

    static final String FIND_ALL_VETS = "net.kemitix.naolo.gateway.data.jpa.findAllVets";
    static final String VETERINARIAN_RESULT_MAPPER = "net.kemitix.naolo.gateway.data.jpa.Veterinarian";

    @Id
    @GeneratedValue
    private Long id;

    private String name;

    @ElementCollection
    @Enumerated(EnumType.STRING)
    @CollectionTable(
            name = "veterinarian_specialisations",
            joinColumns = @JoinColumn(name = "vet_id")
    )
    private final Set<VetSpecialisation> specialisations = new HashSet<>();
}

Veterinarian.java:

public final class Veterinarian {

    private Long id;
    private String name;
    private Set<VetSpecialisation> specialisations;

    public Veterinarian() {
    }

    public Veterinarian(final long id,
                        final String name,
                        final Set<VetSpecialisation> specialisations) {
        this.id = id;
        this.name = name;
        this.specialisations = new HashSet<>(specialisations);
    }

    public long getId() {
        return id;
    }

    public String getName() {
        return name;
    }

    public Set<VetSpecialisation> getSpecialisations() {
        return new HashSet<>(specialisations);
    }

}

VetSpecialisation.java:

public enum VetSpecialisation {

    RADIOLOGY,
    DENTISTRY,
    SURGERY

}

Когда я пытаюсь выполнить именованный запрос:

entityManager.createNamedQuery(VeterinarianJPA.FIND_ALL_VETS, Veterinarian.class)
             .getResultStream()

Я получаю следующее исключение:

java.lang.IllegalArgumentException: Could not locate appropriate constructor on class : net.kemitix.naolo.entities.Veterinarian

    at org.hibernate.loader.custom.ConstructorResultColumnProcessor.resolveConstructor(ConstructorResultColumnProcessor.java:92)
    at org.hibernate.loader.custom.ConstructorResultColumnProcessor.performDiscovery(ConstructorResultColumnProcessor.java:45)
    at org.hibernate.loader.custom.CustomLoader.autoDiscoverTypes(CustomLoader.java:494)
    at org.hibernate.loader.Loader.processResultSet(Loader.java:2213)
    at org.hibernate.loader.Loader.getResultSet(Loader.java:2169)
    at org.hibernate.loader.Loader.executeQueryStatement(Loader.java:1930)
    at org.hibernate.loader.Loader.executeQueryStatement(Loader.java:1892)
    at org.hibernate.loader.Loader.scroll(Loader.java:2765)
    at org.hibernate.loader.custom.CustomLoader.scroll(CustomLoader.java:383)
    at org.hibernate.internal.SessionImpl.scrollCustomQuery(SessionImpl.java:2198)
    at org.hibernate.internal.AbstractSharedSessionContract.scroll(AbstractSharedSessionContract.java:1058)
    at org.hibernate.query.internal.NativeQueryImpl.doScroll(NativeQueryImpl.java:217)
    at org.hibernate.query.internal.AbstractProducedQuery.scroll(AbstractProducedQuery.java:1462)
    at org.hibernate.query.internal.AbstractProducedQuery.stream(AbstractProducedQuery.java:1486)
    at org.hibernate.query.Query.getResultStream(Query.java:1110)

Я ожидаю, что SQL возвращает несколько строк для многозначного Set а не одно значение, которое приводит к тому, что конструктор не совпадает. Как изменить SQL, чтобы получить корректный ввод для конструктора, или мне нужно внести другое изменение конфигурации?

1 ответ

Решение

Ну, я не уверен, возможно ли это так, как вы хотите. Но вы можете использовать LISTAGG функция на specialisations таблица для встраивания specialisations с veterinarians используя какой-то разделитель.

Таким образом, запрос должен выглядеть так:

SELECT v.id, v.name 
(SELECT LISTAGG(vs.type, ';') 
  WITHIN GROUP (ORDER BY vs.type) 
  FROM veterinarian_specialisations vs 
  WHERE vs.vet_id = v.id) specialisations
FROM veterinarians v;

Запрос вернет ветеринар и специализацию, разделенную точкой с запятой:

1   NAME   DENTISTRY;RADIOLOGY

И тогда в вашем Veterinarian конструктор класса вы должны переназначить результат String обратно в набор VetSpecialisation, Я использовал Java 8 stream api просто для удобства.

public final class Veterinarian {

private Long id;
private String name;
private Set<VetSpecialisation> specialisations;

public Veterinarian() {
}

public Veterinarian(final long id,
                    final String name,
                    final String specialisations) {
    this.id = id;
    this.name = name;
    this.specialisations = Arrays.asList(specialisations.split(";"))
        .stream()
        .map(VetSpecialisation::valueOf) //Map string to VetSpecialisation enum.
        .collect(Collectors.toSet());
}
Другие вопросы по тегам