Today I Learned

    [TIL] 23/07/20 메뉴 조건에 따라 다르게 장바구니 save 실행 분기 처리 (saveAll, @Transactional)

    ❗️ 메뉴마다 옵션이 있을 수도 없을 수도 있다. 없는 경우에는 바로 cart 엔티티만 생성해서 insert 하면 되지만, 옵션이 있는 경우에는 cart와 그에 대응하는 cart_option을 저장하는 insert문이 n번 이상 실행돼야 한다. 한 개의 메뉴 (한 개의 카트)는 여러 개의 옵션을 가질 수 있다. 즉, 한 개의 카트와 n개의 옵션이 db에 모두 반영되던가, 전혀 반영되지 않아야 하는 원자성을 지녀야 한다. 기존 코드는 이러한 특성이 반영되지 않았기 때문에 CartController addItemToCart를 수정하고 CartService에 카트와 옵션을 함께 저장하는 메서드를 생성했다. Controller if (optionId == null) { cartService.createCart(..

    [TIL] 23/07/18 페이지 리로드 없이 HTML 폼 사용하여 서버로 데이터를 전송하고 응답 처리하기

    카트에 아이템 저장하기 1. html form이 submit 되면 값을 가지고 컨트롤러로 간다. 2. 사용자의 장바구니에 아이템을 넣은 후에 response entity를 리턴한다. message와 httpStatus는 결과에 따라 다르다. 성공하면 message: "item added" status: ok 실패하면 internal server error. 3. 리턴 받은 값을 가지고 다시 프론트에서 처리해준다. 만약에 받은 메세지가 item added라면, 아이템이 잘 담겼다는 팝업을 띄운 후에 현재 페이지에 머무르게 하고 만약 메시지가 item added와 같지 않다면, 팝업으로 에러메시지를 띄워준다. 브라우저는 form의 submit 버튼이 클릭되면 페이지를 새로고침하거나 새 페이지를 열도록 기본 ..

    [TIL] 23/07/17 장바구니 삭제 구현

    1. 장바구니 삭제 HTML form으로 delete 요청 보내기 spring.mvc.hiddenmethod.filter.enabled=true 2. 참조 무결성 제약 조건을 위해 cart_option -> cart 순서로 삭제하기

    [TIL] 23/7/13 장바구니 리스트 쿼리문 성능 개선 고민

    장바구니 리스트 쿼리문 성능 개선 개선 전: 사용자의 아이디를 기준으로 카트에 담은 아이템의 리스트를 가져와야 한다. 먼저 직관적으로... 장바구니에서 보여줘야 할 정보를 담은 UserCartDTO를 만들었다. private int cartId; private int menuId; private String menuName; private int quantity; private Integer optionId; private String optionName; private Double optionPrice; 위 DTO를 프로젝션하는 쿼리문은 다음과 같이 작성했다. select c.cart_id, c.menu_id, m.menu_name_eng, c.quantity, co.option_id, o.option..

    [TIL] 23/7/12 JPQL DTO 프로젝션

    JPQL DTO 프로젝션 하는 법 DTO 프로젝션: 엔티티에서 원하는 필드만 선택하여 쿼리 결과로 반환하는 방법 DTO 클래스 작성: 원하는 필드를 가지고 있는 DTO 클래스를 작성한다. 해당 필드를 담는 생성자 및 Getter 메서드는 필수. JPQL 쿼리 작성: SELECT 절에서는 엔티티 대신 DTO 클래스의 생성자를 사용하여 원하는 필드에 값이 들어가도록 작성한다. FROM절에는 엔티티를 지정하기. @Query("SELECT new com.ohouse.ohouse.domain.UserCartDTO(" + "cart.cartId, cart.user.userId, cart.menu.menuId, cart.quantity, " + "co.cart.cartId, co.cartOptionId, co.opt..

    [TIL] 23/07/11 장바구니 담기 기능 구현 중

    1. 옵션이 있는 메뉴와 없는 메뉴가 있다. 옵션이 없는 경우에는 파라미터가 전달되지 않을 수 있으므로 이를 처리해야 한다. @RequestParam(value = "menuOption", required = false) @RequestParam에 required 파라미터를 false로 설정하면 된다. required가 false로 설정되면 해당 HTTP 요청 파라미터가 반드시 필요하지 않게 되어, 값이 없는 경우에도 에러가 발생하지 않는다. 또한 옵션은 한 개가 들어올 수도 있고 여러 개가 들어올 수 있다. 이 경우를 대비해 List로 optionId를 받았다. cart와 cartOption 객체를 저장하는 코드는 옵션이 한개 이상 들어올 경우에만 수행해야 한다. (옵션이 들어오지 않으면 카트 객체만 ..

    [TIL] 23/07/06 AWS 인스턴스 재부팅 후 502 에러, 대상 그룹 Unhealthy 해결

    문제: AWS EC2 인스턴스를 중지하고 다시 시작한 이후 502 에러가 발생했다. 재부팅 이전에는 로드밸런서를 포함하여 아무 문제 없이 잘 작동했는데 재부팅 이후로 사이트에 접속이 안되기 시작했다. 502 Bad Gateway 에러는 사용자의 요청이 뒷단의 서버까지 제대로 전달되지 않을 때 발생한다. 이를 확인하기 위해 로드밸런서 보안 그룹, IP 연결 등등을 확인했으나 이상이 없었다. 대상 그룹(Target Group)에 들어와 확인해 보니 80 포트가 Unhealthy 상태였다. 조건: 1. 로드밸런서에는 80포트, 8080포트가 등록 2. 대상 그룹에는 80포트만 등록 3. 실제 애플리케이션은 80포트에서 8080포트로 포트 포워딩을 진행하지 않은 상태. 원인: 로드밸런서에는 80포트와 8080포..

    [TIL] 23/07/05 장바구니, 주문 테이블 수정: 연결 테이블 생성, 사용 예시

    현재 장바구니, 주문 테이블은 한개의 옵션만 선택할 수 있다는 가정하에 만들어졌다. 하지만 메뉴와 옵션이 추가되어 여러개의 옵션을 고를 수 있는 메뉴가 생겨났다. 그에 따라 메뉴와 옵션의 다대다 설정이 필요해졌다. 다대다 옵션 설정을 위해 cart_option과 order_item_option 테이블을 생성했다. 두 테이블의 각 컬럼은 다른 테이블의 기본 키를 참조하는 외래 키로만 구성이 되어 있다. 이런 종류의 테이블을 연결 테이블 (결합 테이블, 다대다 관계 테이블)이라고 부른다. 이 테이블들의 각 행은 각각 cart, order와 option 간의 특정 관계를 나타낸다. cart_id, option_id / order_item_id, option_id의 조합이 고유해야 하므로, 데이터 무결성을 위해..

    [TIL] 23/07/04 InnoDB VS MyISAM (엔진 확인하기, 변경하기)

    외래키를 생성하려 했으나 실패하고 index만 생성됐다. 외래키 생성이 제대로 되지 않는다면 여러 이유가 있을 수 있다. 1. 데이터 유형 불일치 2. 참조 무결성 제약 조건 위반 3. 테이블 잠김: MySQL의 경우, MyISAM 엔진은 외래키 지원을 하지 않는다. 확인해 보니, FK 생성이 되지 않는 테이블만 엔진이 MyISAM으로 되어 있었다. InnoDB는 외래키 지원, 트랜잭션 지원, ACID 등의 특징이 있다. 반면, MyISAM은 외래 키 및 트랜잭션 지원을 하지 않는다. 인덱스 테이블을 디스크에 저장하고 테이블 단위의 잠금을 제공하는 특징이 있어서 읽기 전용 (select) 작업이 많은 경우에는 더 빠른 성능을 제공한다. 테이블의 스토리지 엔진을 확인하는 방법: 'SHOW TABLE STA..

    [TIL] 23/07/03 Primary Key의 데이터 타입 INT VS BIGINT

    오늘 한 것 + 오늘 배운 것 int (4바이트, 32비트) : 부호가 있는 경우에는 -2,147,483,648에서 2,147,483,647 사이의 값 (약 21억), 부호가 없는 경우에는 0에서 4,294,967,295 (약 43억)사이의 값을 저장할 수 있다. bigint (8바이트, 64비트) : 부호가 있는 경우에는 -9,223,372,036,854,775,808에서 9,223,372,036,854,775,807 (약 922경) 사이의 값, 부호가 없는 경우에는 0에서 18,446,744,073,709,551,615 사이의 값을 저장할 수 있다. PK의 데이터 타입을 설정할 때는 저장하려는 데이터의 크기와 범위를 고려하여 선택해야 한다. 현재 테이블의 PK는 통일 되어 있지 않고, int, big..