JPA entitygraph fetchgraph не работает, а loadgraph работает - PullRequest
0 голосов
/ 22 января 2019

Я пытаюсь использовать динамические графы сущностей JPA в Spring MVC. Когда я использую стратегию fetchgraph, она выдает

«java.lang.illegalArgumentException» с сообщением «Невозможно найти Атрибут с данным именем [экосистемы] на этом ManagedType [com.ith.betta.web.models.BaseModel]».

Но когда я выбрал loadgraph, он работает нормально. Ниже приведены мои сущности

Order.java

package com.ith.betta.web.models;

import javax.persistence.*;
import java.util.List;

@Entity
@Table(name = "orders")
public class Order extends BaseModel {

    private String orderNumber;
    private String type;
    private Integer quantity;

    @ManyToOne
    private User user;

    @OneToMany(mappedBy = "order")
    private List<Invoice> invoices;

    public Order() {
    }

    public Order(String orderNumber, String type, Integer quantity) {
        this.orderNumber = orderNumber;
        this.type = type;
        this.quantity = quantity;
    }

    public String getOrderNumber() {
        return orderNumber;
    }

    public void setOrderNumber(String orderNumber) {
        this.orderNumber = orderNumber;
    }

    public String getType() {
        return type;
    }

    public void setType(String type) {
        this.type = type;
    }

    public Integer getQuantity() {
        return quantity;
    }

    public void setQuantity(Integer quantity) {
        this.quantity = quantity;
    }
}

Invoice.java

package com.ith.betta.web.models;

import javax.persistence.Entity;
import javax.persistence.ManyToOne;
import javax.persistence.Table;

@Entity
@Table(name = "invoices")
public class Invoice extends BaseModel {

    @ManyToOne
    private Order order;
}

User.java

package com.ith.betta.web.models;

import javax.persistence.*;
import javax.validation.constraints.NotNull;
import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

@Entity
@Table(name = "users")

public class User extends BaseModel {

    @NotNull
    @Column(unique = true)
    private String email;

    @Column(name = "first_name")
    private String firstName;

    @Column(name = "last_name")
    private String lastName;

    private String password;
    private String username;
    private Boolean enabled;

    @OneToOne(mappedBy = "user", cascade = CascadeType.ALL, fetch = FetchType.EAGER)
    private UserProfile userProfile;

    @ManyToMany(cascade = CascadeType.ALL, fetch = FetchType.EAGER)
    private Set<Role> roles = new HashSet<Role>();

    @OneToMany(mappedBy = "user", cascade = CascadeType.ALL)
    private List<Notification> notifications = new ArrayList<Notification>();

    @OneToMany(mappedBy = "user", cascade = CascadeType.ALL)
    private List<Order> orders;

    @OneToMany(mappedBy = "user", cascade = CascadeType.ALL)
    private List<Ecosystem> ecosystems;

    public User() {
    }

    public User(String email, String firstName, String lastName, String password, String username) {

        this.email = email;
        this.firstName = firstName;
        this.lastName = lastName;
        this.password = password;
        this.username = username;

    }

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }

    public String getFirstName() {
        return firstName;
    }

    public void setFirstName(String firstName) {
        this.firstName = firstName;
    }

    public String getLastName() {
        return lastName;
    }

    public void setLastName(String lastName) {
        this.lastName = lastName;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public Boolean getEnabled() {
        return enabled;
    }

    public void setEnabled(Boolean enabled) {
        this.enabled = enabled;
    }

    public UserProfile getUserProfile() {
        return userProfile;
    }

    public void setUserProfile(UserProfile userProfile) {
        this.userProfile = userProfile;
    }

    public Set<Role> getRoles() {
        return roles;
    }

    public void setRoles(Set<Role> roles) {
        this.roles = roles;
    }

    public List<Notification> getNotifications() {
        return notifications;
    }

    public void setNotifications(List<Notification> notifications) {
        this.notifications = notifications;
    }

    @PrePersist
    private void initializeCreatedAndUpdatedAt() {
        LocalDateTime now = LocalDateTime.now();
        this.createdAt = this.updatedAt = now;
    }

    @PreUpdate
    private void updateUpdatedAt() {
        this.updatedAt = LocalDateTime.now();
    }

    public List<Order> getOrders() {
        return orders;
    }

    public void setOrders(List<Order> orders) {
        this.orders = orders;
    }

    public List<Ecosystem> getEcosystems() {
        return ecosystems;
    }

    public void setEcosystems(List<Ecosystem> ecosystems) {
        this.ecosystems = ecosystems;
    }
}

BaseModel.java

package com.ith.betta.web.models;

import org.hibernate.annotations.CreationTimestamp;
import org.hibernate.annotations.UpdateTimestamp;

import javax.persistence.*;
import java.time.LocalDateTime;

@MappedSuperclass
public class BaseModel {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    protected Long id;
    @CreationTimestamp
    @Column(name = "created_at")
    protected LocalDateTime createdAt;
    @UpdateTimestamp
    @Column(name = "updated_at")
    protected LocalDateTime updatedAt;

    public Long getId() {
        return id;
    }

    public LocalDateTime getCreatedAt() {
        return createdAt;
    }

    public void setCreatedAt(LocalDateTime createdAt) {
        this.createdAt = createdAt;
    }

    public LocalDateTime getUpdatedAt() {
        return updatedAt;
    }

    public void setUpdatedAt(LocalDateTime updatedAt) {
        this.updatedAt = updatedAt;
    }
}

и, наконец, класс обслуживания

package com.ith.betta.web.services.impl;

import com.ith.betta.web.models.Order;
import com.ith.betta.web.services.OrderService;
import org.springframework.stereotype.Service;

import javax.persistence.EntityGraph;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;

@Service("orderService")
public class OrderServiceImpl implements OrderService {

    @PersistenceContext
    private EntityManager em;

    @Override
    public Optional<Order> findOrderWithInvoices(Long orderId) {
        EntityGraph graph = em.createEntityGraph(Order.class);
        graph.addSubgraph("invoices");
        Map hints = new HashMap();
        hints.put("javax.persistence.fetchgraph", graph);
        Order order = em.find(Order.class, orderId, hints);

        return Optional.of(order);

    }
}

Теперь при настройке выше пользовательские отношения создают проблему. Когда я использую loadgraph, он работает и загружает пользователя, но никакой активной нагрузки не объявляется. Но когда я переключаюсь на fetchgraph, он пытается найти экосистему пользователя -> и по какой-то причине он не находит ее и переходит к базовой модели.

Соответствующие версии используются

  • spring-release-releasetrain: Lovelace-RELEASE
  • Hibernate: 5.4.0.FINAL

Редактировать 1: Когда я меняю тип извлечения ассоциации пользователей ManyToOne на Lazy (по умолчанию это Eager), он начинает работать, не знаю причину

Редактировать 2 Далее углубившись в это, я удалил уникальную аннотацию в столбце электронной почты, и теперь он начал работать с типами извлечения Lazy и Eager. До сих пор не знаю, почему

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...