package jpabook.jpashop.service;
import jakarta.persistence.EntityManager;
import jpabook.jpashop.domain.Address;
import jpabook.jpashop.domain.Member;
import jpabook.jpashop.domain.Order;
import jpabook.jpashop.domain.OrderStatus;
import jpabook.jpashop.domain.item.Book;
import jpabook.jpashop.domain.item.Item;
import jpabook.jpashop.exception.NotEnoughStockException;
import jpabook.jpashop.repository.OrderRepository;
import org.junit.Assert;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
import org.springframework.transaction.annotation.Transactional;
import static org.junit.Assert.*;
@RunWith(SpringRunner.class)
@SpringBootTest
@Transactional
public class OrderServiceTest {
@Autowired
EntityManager em;
@Autowired
OrderService orderService;
@Autowired
OrderRepository orderRepository;
@Test
public void 상품주문() throws Exception{
//given
Member member = createMember();
Book book = createBook("시골 jpa", 10000, 10);
int orderCount = 2;
//when
Long orderId = orderService.order(member.getId(), book.getId(), orderCount);
System.out.println(book.getId());
//then
Order getOrder = orderRepository.findOne(orderId);
assertEquals("상품주문시 상태는 ORDER", OrderStatus.ORDER, getOrder.getStatus());
assertEquals("주문한 상품 종류 수가 정확해야 한다", 1, getOrder.getOrderItems().size());
assertEquals("주문 가격은 가격 * 수량이다", 10000*orderCount, getOrder.getTotalPrice());
assertEquals("주문 수량만큼 재고가 줄어야한다", 8, book.getStockQuantity());
}
private Book createBook(String name, int price, int stockQuantity) {
Book book = new Book();
book.setName(name);
book.setPrice(price);
book.setStockQuantity(stockQuantity);
em.persist(book); //이걸 해줘야 book의 id 필드에 pk 값이 존재하게 된다.
return book;
}
private Member createMember() {
Member member = new Member();
member.setName("회원1");
member.setAddress(new Address("서울", "강가", "123-123"));
em.persist(member);
return member;
}
@Test(expected = NotEnoughStockException.class)
public void 상품주문_재고수량초과() throws Exception{
//given
Member member = createMember();
Item item = createBook("시골 jpa", 10000, 10);
int orderCount = 11;
//when
orderService.order(member.getId(), item.getId(), orderCount);
//then
fail("재고 수량 부족 예외가 발생해야한다.");
}
@Test
public void 주문취소() throws Exception{
//given
Member member = createMember();
Book item = createBook("시골 jpa", 10000, 10);
int orderCount = 2;
Long orderId = orderService.order(member.getId(), item.getId(), orderCount);
//when
orderService.cancelOrder(orderId);
//then
Order getOrder = orderRepository.findOne(orderId);
assertEquals("주문 취소시 상태는 cancel이다", OrderStatus.CANCEL, getOrder.getStatus());
assertEquals("주문이 취소된 상품은 그만큼 재고가 증가해야한다", 10, item.getStockQuantity());
}
}
오류 : org.springframework.dao.InvalidDataAccessApiUsageException: id to load is required for loading; nested exception is java.lang.IllegalArgumentException: id to load is required for loading
book.getId()를 호출하는데 id가 존재하지 않는다는 오류가 났다.
id가 @GeneratedValue로 되어있어 id를 생성하려면 엔티티를 db에 저장해야 한다.
알고보니 book을 생성해주고 마지막에 entityManager가 영속성 컨텍스트 할 수 있도록 em.persist(book)을 빼먹었다. 추가해주니 오류 해결! em.persist(book)하면 이후에 book의 id 필드에 pk값이 저장된다고 한다!
본 게시글은 인프런 김영한 강사의 실전! 스프링 부트와 JPA 활용 1 - 웹 어플리케이션 개발 수강 후 작성한 코드 입니다.
'BackEnd > JPA' 카테고리의 다른 글
@Transactional과 @Transactional( readOnly = true ) (0) | 2024.11.21 |
---|---|
[Spring Data JPA] 회원가입 기능 및 회원 조회 + 의존성 주입 (0) | 2024.03.03 |
[Spring Data JPA] 연관관계 매핑 & 테이블 설계 (0) | 2024.03.01 |
[JPA] 인텔리제이 jpa 사용시 테이블이 drop 되지 않는 오류 (0) | 2023.12.22 |