[플레이 데이터] 한화시스템 BEYOND SW캠프
🥪팀 5VEN
팀장: 곽효림 | 팀원: 김재구 | 팀원: 임찬근 | 팀장: 최인욱 | 팀원: 최원익 |
현대인들은 책을 통해 지식을 얻고 감동을 경험하지만 더 이상 필요하지 않은 책이 쌓여가는 문제를 자주 겪습니다. 중고서적 거래 플랫폼은 개인과 개인(P2P) 또는 개인과 업자 간의 중고 책 거래를 손쉽게 연결하여 책의 가치를 지속적으로 나눌 수 있는 공간을 제공합니다.
- DB 클러스터 (Active-Active)
- 채팅 기능 등 비교적 쓰기 작업이 많은 서비스인 만큼 여러대의 쓰기 서버 구성
- 일부 서버 재난 발생 시에도 계속 문제없이 사용하기 좋으며 이후 동기화도 비교적 간단한 구성
- 이후 추가 서버 구성이 필요할 시 서버 추가하기 쉬운 구조
- HAProxy
- DB 클러스터의 부하 분산
- SPOF에 따른 문제를 방지하기 위해 2대를 구성
DB 클러스터
- 1대 장애 시
- HAProxy에서 wsrep_local_state 값을 확인하여 장애 확인하고 해당 노드 차단
- 노드 복구 후 MySQL을 다시 실행하여 클러스터에 재참여
- 복구된 노드는 클러스터로 자동 동기화 됨
- 이후 HAProxy에서 다시 해당 노드 복구
- 2대 장애 시
- HAProxy에서 wsrep_local_state 값을 확인하여 장애 확인하고 해당 노드들 차단
- 남아있는 노드에서 Primary Component를 수동으로 복구
- 복구된 노드 하나를 Primary Component에 합류시킴
- 이후 HAProxy에서 다시 해당 노드들 복구
- 3대 장애 시
- 가장 최근에 종료되었거나 상태가 최신인 노드를 찾아 복구.
- 해당 노드를 기반으로 클러스터를 부트스트랩
- 부트스트랩된 노드가 정상 동작하면 다른 노드들을 클러스터에 다시 추가
- 최신 백업을 사용해 클러스터를 초기화
- HAProxy에서 다시 세 노드들 연결
- 추가적으로 데이터 유실을 최대한 방지하기 위해 주기적인 클러스터 백업 및 자동화된 복구 스크립트 준비
- 전체 장애 발생을 최대한 방지하기 위해 노드들을 서로 다른 데이터센터에 분산 배치
HAProxy
- keepalived를 사용하여 Active-standby 상태로 공유된 가상 IP를 이용하여 접속
- 주 HAProxy가 응답하지 않을 경우 예비 HAProxy로 VIP를 자동 전환
- 장애 복구 후 VIP가 다시 Primary HAProxy로 돌아오도록 설정
- 추가로 HAProxy 설정 파일을 주기적으로 동기화 하여 동일한 환경 유지
- 혹은 글로벌 서버 로드밸런싱을 사용하여 다중 지역 HAProxy를 사용
DDL
MEMBER
CHAT
PRODUCT
PAY
- sql 쿼리
-- 채팅방 조회(개선전) ---- SELECT cr.id AS chat_room_id, -- 채팅방 ID cr.identifier AS book_title, -- 책 제목 b.id AS book_id, -- 책 ID p.id AS product_id, -- 판매 게시글 ID p.member_id AS seller_id, -- 판매자 ID cr.last_chat, -- 마지막 메시지 cr.created_at -- 생성일시 FROM chatting_room cr JOIN product p ON cr.identifier = ( SELECT b.title -- 책 제목과 identifier 매칭 FROM book b WHERE b.id = p.book_id ) JOIN book b ON b.id = p.book_id -- 책 ID를 추가로 가져오기 위해 조인 LIMIT 0, 1000; -- 채팅방 조회(개선후) ---- SELECT cr.id AS chat_room_id, -- 채팅방 ID b.title AS book_title, -- 책 제목 (JOIN에서 가져옴) b.id AS book_id, -- 책 ID p.id AS product_id, -- 판매 게시글 ID p.member_id AS seller_id, -- 판매자 ID cr.last_chat, -- 마지막 메시지 cr.created_at -- 생성일시 FROM chatting_room cr JOIN book b ON cr.identifier = b.title -- identifier와 title 매칭 JOIN product p ON b.id = p.book_id -- 책 ID를 가져오기 위한 조인 LIMIT 1000; -- LIMIT 범위 적용
- 핵심 기능인 채팅방 조회의 쿼리가 실행하는데 상당 시간 걸림을 확인하여 성능 개선 시도
- 채팅방 조회 기능 쿼리 성능 개선을 위해 연산이 비교적 더 많이 사용되는 서브 쿼리에서 비교적 연산이 적은 다중 조인으로 쿼리 개선
- set profiling을 통해 쿼리 실행시간 약 1/3으로 축소 확인
- grafana를 활용한 모니터링으로 쿼리 실행시 DB 서버의 CPU 사용량 약 1/2로 축소 확인
- 다중 조인을 사용함으로써 한 번의 연산으로 데이터를 가져올 수 있고, chatting_room과 book을 먼저 JOIN하고, 그 후에 product를 조인함으로써 불필요한 연산을 줄임으로써 시간 개선 및 서버의 CPU 사용량 감소
성능 테스트 전/후 비교