주문 도메인 개발
@Entity
@Table(name = "orders")
@Getter @Setter
public class Order {
@Id @GeneratedValue
@Column(name = "order_id")
private Long id;
@ManyToOne(fetch = FetchType.LAZY) //여러 상품이 하나의 고객에 의해
@JoinColumn(name = "member_id")
private Member member;
@OneToMany(mappedBy = "order", cascade = CascadeType.ALL)
private List<OrderItem> orderItems = new ArrayList<>();
@OneToOne(fetch = FetchType.LAZY, cascade = CascadeType.ALL)
@JoinColumn(name = "delivery_id")
private Delivery delivery;
private LocalDateTime orderDate;
@Enumerated(EnumType.STRING)
private OrderStatus status; //주문 상태 [ORDER, CANCEL]
//==연관 관계 편의 메서드 ==//
public void setMember(Member member) {
this.member = member;
member.getOrders().add(this);
}
public void addOrderItem(OrderItem orderItem) {
orderItems.add(orderItem);
orderItem.setOrder(this);
}
public void setDelivery(Delivery delivery) {
this.delivery = delivery;
delivery.setOrder(this);
}
//==생성 메서드==/
public static Order createOrder(Member member, Delivery delivery, OrderItem... orderItems) {
Order order = new Order();
order.setMember(member);
order.setDelivery(delivery);
for (OrderItem orderItem:orderItems) {
order.addOrderItem(orderItem);
}
order.setStatus(OrderStatus.ORDER);
order.setOrderDate(LocalDateTime.now());
return order;
}
//==비즈니스 로직==//
/**
* 주문 취소
*/
public void cancel() {
if(delivery.getStatus() == DeliveryStatus.COMP) {
throw new IllegalStateException("이미 배송 완료된 상품은 취소가 불가능합니다.");
}
this.setStatus(OrderStatus.CANCEL);
for (OrderItem orderItem : orderItems) {
orderItem.cancel();
}
}
//==조회 로직==//
/**
* 전체 주문 가격 조회
*/
public int getTotalPrice() {
int totalPrice = 0;
for (OrderItem orderItem:orderItems) {
totalPrice += orderItem.getTotalPrice();
}
return totalPrice;
}
}
- createOrder
- 주문 회원, 배송 정보, 주문 상품을 받아서 set한 후에 order를 리턴한다.
- 주문 생성에 대한 로직을 모두 여기에 넣어둔다.
- 앞으로 주문 생성에 대한 코드를 수정하려면 이 함수만 고치면 된다.
- cancel()
- DeliveryStatus를 통해 배송 상태를 보고 이미 배송 완료된 상품이면 취소를 할 수 없게 한다.
- 주문 상태를 바꿔주고, 루프를 돌면서 OrderItem의 cancel()을 통해 주문 상품에 대한 처리를 해준다. (재고 수량 다시 증가시키기)
- getTotalPrice()
- 주문 상품들을 하나씩 loop를 돌면서 OrderItem의 getTotalPrice() (상품 금액 * 주문 수량)을 받고 totalPrice에 더하면서 주문 총 금액을 구하고, 리턴한다.
@Entity
@Getter @Setter
public class OrderItem {
@Id @GeneratedValue
@Column(name = "order_item_id")
private Long id;
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "item_id")
private Item item;
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "order_item")
private Order order;
private int orderPrice;
private int count;
//==생성 메서드==//
public static OrderItem createOrderItem(Item item, int orderPrice, int count) { //item에 price가 있지만, 할인 가격일 수 있어서 orderPrice로 따로 사용
OrderItem orderItem = new OrderItem();
orderItem.setItem(item);
orderItem.setOrderPrice(orderPrice);
orderItem.setCount(count);
item.removeStock(count);
return orderItem;
}
//==비즈니스 로직==/
public void cancel() {
getItem().addStock(count);
}
public int getTotalPrice() {
return getOrderPrice() * getCount();
}
}
- createOrderItem
- 주문 상품, 주문 금액, 주문 수량을 set해서 return한다.
- item에 상품 금액이 있지만, 할인 적용 금액을 위해 orderPrice로 따로 사용한다.
- 주문 수량 만큼 Item의 removeStock()을 이용해서 재고 수량을 감소시킨다.
- cancel()
- item을 가져와서 주문 수량 만큼 재고를 증가시킨다.
- getTotalPrice()
- 주문 가격에 주문 수량을 곱해서 상품의 총 금액을 구한다.
주문 서비스 개발
@Service
@RequiredArgsConstructor
@Transactional(readOnly = true)
public class OrderService {
private final OrderRepository orderRepository;
private final MemberRepository memberRepository;
private final ItemRepository itemRepository;
/**
* 주문
*/
@Transactional
public Long order(Long memberId, Long itemId, int count) {
//엔티티 조회
Member member = memberRepository.findOne(memberId);
Item item = itemRepository.findOne(itemId);
//배송 정보 생성
Delivery delivery = new Delivery();
delivery.setAddress(member.getAddress());
//주문 상품 생성
OrderItem orderItem = OrderItem.createOrderItem(item ,item.getPrice(), count);
//주문 생성
Order order = Order.createOrder(member, delivery, orderItem);
//주문 저장
orderRepository.save(order); //앞에서 도메인 설계할 때 cascade 옵션을 넣어줘서 Order만 persist해줘도 delivery와 orderitem도 persist된다.
return order.getId();
}
/**
* 주문 취소
*/
@Transactional
public void cancelOrder(Long orderId) {
//주문 엔티티 조회
Order order = orderRepository.findOne(orderId);
//주문 취소
order.cancel();
}
/**
* 주문 검색
*/
// public List<Order> findOrders(OrderSearch orderSearch) {
// return orderRepository.findAll(orderSearch);
// }
}
- order()
- 엔티티 조회
- 파라미터로 받은 memberId와 itemId를 통해 주문한 회원과 주문 아이템을 가져온다.
- 배송 정보 생성
- 배송 정보는 일단 주문한 회원의 주소로 설정한다.
- 주문 상품 생성
- OrderItem에 만들어둔 createOrderItem()을 통해 주문 상품을 생성한다.
- 주문 생성
- Order에 만들어둔 createOrder()를 통해 주문을 생성한다.
- orderItem 여러개를 넘겨도 처리할 수 있음.
- 주문 저장
- orderRepository.save()를 통해 주문을 저장한다.
- cascade옵션이 있어서 Order와 Delivery도 함께 persist 된다.
- 엔티티 조회
cascade의 범위: 현재 프로젝트에서 Order만 OrderItem을 쓰고, Order만 Delivery를 쓴다. 다른 곳에서 쓰지 않는다. 그리고 persist해야하는 라이프 사이클도 똑같다. 이럴때만 cascade를 써야 한다. 잘못하면 의도치 않은 것도 지워질 수 있다.
비즈니스 로직 대부분이 엔티티에 있고, 서비스 계층은 단순히 엔티티에 필요한 요청을 위임하는 역할을 한다 . 이처럼 엔티티가 비즈니스 로직을 가지고 객체 지향의 특성을 적극 활용하는 것을 '도메인 모델 패턴'이라고 한다.
주문 기능 테스트
@RunWith(SpringRunner.class)
@SpringBootTest
@Transactional
public class OrderServiceTest {
@Autowired
EntityManager entityManager;
@Autowired
OrderService orderService;
@Autowired
OrderRepository orderRepository;
@Test
public void 상품주문() throws Exception {
//given
Member member = createMember();
Book book = createBook(10000, 10);
int orderCount = 2;
//when
Long orderId = orderService.order(member.getId(), book.getId(), orderCount);
//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());
}
@Test(expected = NotEnoughStockException.class)
public void 상품주문_재고수량초과() throws Exception {
//given
Member member = createMember();
Book book = createBook(10000, 10);
int orderCount = 11;
//when
orderService.order(member.getId(), book.getId(), orderCount);
//then
fail("재고 수량 부족 예외가 발생해야 한다.");
}
@Test
public void 주문취소() throws Exception {
//given
Member member = createMember();
Book book = createBook(10000, 10);
int orderCount = 2;
Long orderId = orderService.order(member.getId(), book.getId(), orderCount);
//when
orderService.cancelOrder(orderId);
//then
Order getOrder = orderRepository.findOne(orderId);
assertEquals("주문 취소시 상태는 CANCEL이다.", OrderStatus.CANCEL, getOrder.getStatus());
assertEquals("주문이 취소된 상품은 그만큼 재고가 증가해야 한다.", 10, book.getStockQuantity());
}
private Book createBook(int price, int stockQuantity) {
Book book = new Book();
book.setName("시골 JPA");
book.setPrice(price);
book.setStockQuantity(stockQuantity);
entityManager.persist(book);
return book;
}
private Member createMember() {
Member member = new Member();
member.setName("회원1");
member.setAddress(new Address("서울", "강가", "123-123"));
entityManager.persist(member);
return member;
}
}
주문 검색 기능 개발
public List<Order> findAll(OrderSearch orderSearch) {
String jpql = "select o from Order o join o.member m";
boolean isFirstCondition = true;
//주문 상태 검색
if (orderSearch.getOrderStatus() != null) {
if (isFirstCondition) {
jpql += " where";
isFirstCondition = false;
} else {
jpql += " and";
}
jpql += " o.status = :status";
}
//회원 이름 검색
if (StringUtils.hasText(orderSearch.getMemberName())) {
if (isFirstCondition) {
jpql += " where";
isFirstCondition = false;
} else {
jpql += " and";
}
jpql += " m.name like :name";
}
TypedQuery<Order> query = em.createQuery(jpql, Order.class)
.setMaxResults(1000);//최대 1000건 paging하려면 .setFirstResult()
if (orderSearch.getOrderStatus() != null) {
query = query.setParameter("status", orderSearch.getOrderStatus());
}
if (StringUtils.hasText(orderSearch.getMemberName())) { query = query.setParameter("name", orderSearch.getMemberName());
}
return query.getResultList();
}
/**
* JPA Criteria
*/
public List<Order> findAllByCriteria(OrderSearch orderSearch) {
CriteriaBuilder cb = em.getCriteriaBuilder();
CriteriaQuery<Order> cq = cb.createQuery(Order.class);
Root<Order> o = cq.from(Order.class);
Join<Order, Member> m = o.join("member", JoinType.INNER); //회원과 조인
List<Predicate> criteria = new ArrayList<>();
//주문 상태 검색
if (orderSearch.getOrderStatus() != null) {
Predicate status = cb.equal(o.get("status"),
orderSearch.getOrderStatus());
criteria.add(status);
}
//회원 이름 검색
if (StringUtils.hasText(orderSearch.getMemberName())) {
Predicate name =
cb.like(m.<String>get("name"), "%" + orderSearch.getMemberName()
+ "%");
criteria.add(name);
}
cq.where(cb.and(criteria.toArray(new Predicate[criteria.size()])));
TypedQuery<Order> query = em.createQuery(cq).setMaxResults(1000); //최대 1000건
return query.getResultList();
}
위 두가지 방식으로 할 수 있지만, 문제가 많다.
그래서 Querydsl로 작성하는 것이 훨씬 낫다.
728x90
'Spring' 카테고리의 다른 글
[JPA 기본편] SQL 중심적인 개발의 문제 + JPA 소개 (0) | 2024.06.14 |
---|---|
[Springboot & JPA 1] 웹 계층 개발 (2) | 2024.05.15 |
[Springboot & JPA 1] 상품 도메인 개발 (0) | 2024.05.03 |
[Spring boot & JPA 1] 회원 도메인 개발 (0) | 2024.05.01 |
[Spring boot& JPA 1] 도메인 분석 설계 (3) | 2024.04.28 |