Как исправить HttpMessageNotWritableException в oneToMany при загрузке Spring - PullRequest
0 голосов
/ 11 января 2019

Я использую весеннюю загрузку и использую hibernate для создания некоторых отношений между Entity. Но когда я получаю данные, это выдает мне ошибку:

org.springframework.http.converter.HttpMessageNotWritableException.

Ниже мой код:

Сущность:

Singer.class:

@Entity
public class Singer implements Serializable {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private long id;
    @Column(name = "FIRST_NAME")
    private String firstName;
    @Column(name = "LAST_NAME")
    private String lastName;
    @Column(name = "BIRTH_DAY")
    private Date birthDay;
//    @JsonIdentityInfo(generator = ObjectIdGenerators.PropertyGenerator.class, property="id", scope=ServerRequest.class)
    @JsonIgnore
    @OneToMany(mappedBy = "singer", fetch = FetchType.EAGER, cascade = CascadeType.ALL, orphanRemoval = true)
    private List<Album> albums = new ArrayList<>();

//    @JsonIgnore
    @ManyToMany(fetch = FetchType.EAGER)
    @JoinTable(name = "singer_instrument", joinColumns = @JoinColumn(name = "SINGER_ID"), inverseJoinColumns = @JoinColumn(name = "INSTRUMENT_ID"))
    private Set<Instrument> instruments = new HashSet<>();



    public Singer() {
    }

    public Singer(String firstName, String lastName, Date birthDay, List<Album> albums, Set<Instrument> instruments) {
        this.firstName = firstName;
        this.lastName = lastName;
        this.birthDay = birthDay;
        this.albums = albums;
        this.instruments = instruments;
    }

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    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 Date getBirthDay() {
        return birthDay;
    }

    public void setBirthDay(Date birthDay) {
        this.birthDay = birthDay;
    }

    public List<Album> getAlbums() {
        return albums;
    }

    public void setAlbums(List<Album> albums) {
        this.albums = albums;
    }

    public Set<Instrument> getInstruments() {
        return instruments;
    }

    public void setInstruments(Set<Instrument> instruments) {
        this.instruments = instruments;
    }

    public boolean addAbum(Album album) {
        album.setSinger(this);
        return getAlbums().add(album);
    }

    public boolean addInstrument(Instrument instrument) {
        return instruments.add(instrument);
    }

    @Override
    public String toString() {
        return "Singer - Id: " + id + ", First name: " + firstName
                + ", Last name: " + lastName + ", Birthday: " + birthDay;
    }

}

AlbumClass:

   @Entity
public class Album implements Serializable {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private long id;
    @Column(name = "TITLE")
    private String title;
    @Column(name = "RELEASE_DATE")
    private Date releaseDate;
    @ManyToOne(fetch = FetchType.EAGER)
    @JoinColumn(name = "singer_id")
    private Singer singer;

    public Album() {
    }

    public Album(String title, Date releaseDate, Singer singer) {
        this.title = title;
        this.releaseDate = releaseDate;
        this.singer = singer;
    }

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public Date getReleaseDate() {
        return releaseDate;
    }

    public void setReleaseDate(Date releaseDate) {
        this.releaseDate = releaseDate;
    }

    public Singer getSinger() {
        return singer;
    }

    public void setSinger(Singer singer) {
        this.singer = singer;
    }

    @Override
    public String toString() {
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
        return String.format("Album - id: %d, Singer id: %d, Title: %s, Release Date: %s",
                id, singer.getId(), title, sdf.format(releaseDate));
    }

}

Инструмент:

@Entity
public class Instrument implements Serializable {

    @Id
    @Column(name = "INSTRUMENT_ID")
    private String instrumentId;

    @ManyToMany(fetch = FetchType.EAGER)
    @JoinTable(name = "singer_instrument", joinColumns = @JoinColumn(name = "INSTRUMENT_ID"), inverseJoinColumns = @JoinColumn(name = "SINGER_ID"))
    private Set<Singer> singers = new HashSet<>();

    public Instrument() {
    }

    public Instrument(String instrumentId) {
        this.instrumentId = instrumentId;
    }

    public String getInstrumentId() {
        return instrumentId;
    }

    public void setInstrumentId(String instrumentId) {
        this.instrumentId = instrumentId;
    }

    public Set<Singer> getSingers() {
        return singers;
    }

    public void setSingers(Set<Singer> singers) {
        this.singers = singers;
    }

    @Override
    public String toString() {
        return "Instrument :" + getInstrumentId();
    }


  }

SingerController:

@RestController
@RequestMapping(path = "/singers")
public class SingerController {

    @Autowired
    private SingerRepository singerRepository;

    @Autowired
    private SingerService singerService;

    @GetMapping(path = "/findAll")
    public List<Singer> findAllSinger() {
        List<Singer> signers = (List<Singer>) singerRepository.findAll();
        return signers;
    }

    @PostMapping(path = "/createSinger",consumes = "application/json")
    public ResponseEntity<Singer> createSinger(@RequestBody Singer singer) {
        Singer singer1 = singerRepository.save(singer);
        URI uri = ServletUriComponentsBuilder.fromCurrentRequest().path("/{id}").buildAndExpand(singer.getId()).toUri();
        return ResponseEntity.created(uri).body(singer1);
    }

    @GetMapping(path = "{/id}")
    public Singer findById(@PathVariable long id) {
        Optional<Singer> singer = singerRepository.findById(id);
        if(!singer.isPresent()) {
            throw new SingerNotFoundException("Can't find Singer by id"+id);
        }
        return singer.get();
    }

    @PutMapping(path = "{/id}")
    public Singer updateSingerById(@PathVariable long id,@RequestBody Singer singer) {
        Optional<Singer> singers = this.singerRepository.findById(id);
        if(!singers.isPresent()) {
            throw new SingerNotFoundException("Can't find Singer by id"+id);
        }
        Singer singer1 = singers.get();
        singer1.setFirstName(singer.getLastName());
        singer1.setLastName(singer.getLastName());
        return singer1;
    }

    @DeleteMapping("/{id}")
    @ResponseStatus(HttpStatus.NO_CONTENT)
    public void deleteSingerById(@PathVariable long id) {
        Optional<Singer> singers = this.singerRepository.findById(id);
        if(!singers.isPresent()) {
            throw new SingerNotFoundException("Can't find Singer by id"+id);
        }
        singerRepository.deleteById(id);
    }

    @GetMapping(path = "/findByName")
    public SingerDto findAlbumBySingerName(@RequestParam String firstName) {
        SingerDto singers = singerService.findSingerFirstName(firstName);
        return singers;
    }

SingerRepository:

public interface SingerRepository extends CrudRepository<Singer,Long> {

   Singer findByFirstName(String firstName);
}

Певица Сервис:

@Slf4j
@Service
public class SingerServiceImpl implements SingerService {
    @Autowired
    private SingerRepository singerRepository;


    private SingerDto findSingerWithAlbum(Singer singer) {
        SingerDto singerDto = new SingerDto();
        singerDto.setFirstName(singer.getFirstName());
        singerDto.setLastName(singer.getLastName());
        singerDto.setBirth_date(singer.getBirthDay());
        singerDto.setAlbums(singer.getAlbums());
        singerDto.setInstruments(singer.getInstruments());
        return singerDto;

    }

    @Override
    public SingerDto findSingerFirstName(String firstName) {
        Singer singer = singerRepository.findByFirstName(firstName);
        SingerDto singerDto = findSingerWithAlbum(singer);
        return singerDto;
    }
}

Когда я вызываю метод: findAlbumBySingerName в почтальоне выглядит так:

http://localhost:8080/singers/findByName?firstName=Eric

выдает ошибку:

org.springframework.http.converter.HttpMessageNotWritableException 

Но когда я добавляю @JsonIgnore в поле: private List<Album> albums = new ArrayList<>(); и @JsonIgnore private Set<Instrument> instruments = new HashSet<>(); в SingerDTO, это нормально. Он получает успех firstName,lastname и birthDay, но не имеет альбомов и инструментов.

Я хочу получить все альбомы и инструменты при вызове метода findAlbumBySingerName. Как я мог получить это?

1 Ответ

0 голосов
/ 11 января 2019

Я нашел решение. Я добавляю @JsonIgnoreProperties (), и это работает. Спасибо

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