동기화란 ?
프로세스 또는 스레드들이 수행되는 시점을 조절하여 서로가 알고 있는 정보가 일치하는 것을 의미
우리가 알다시피 컴퓨터는 프로세스 또는 스레드로 작업을 실행합니다.
예를 들어 a = 2 라는 자원이 있다고 해보겠습니다.
P1이라는 프로세스는 a라는 값을 이용해서 어떠한 로직을 수행합니다.
그런데 만약 그 사이 P2라는 프로세스가 a의 값을 3으로 바꾸게 된다면 어떻게 될까요?
아마 프로그램은 우리가 원하는 형태의 값을 반환하지 않을 것입니다.
이때 우리는 a를 공유 자원이라고 말하고,
이렇게 두개 이상의 프로세스 혹은 스레드가 동기화 없이 접근하려는 현상을 race condition(경쟁 상태)라고 합니다.
그리고 이러한 문제를 해결하는 것. 즉 서로가 알고있는 정보를 일치시키는 것을 synchronization(동기화)라고 합니다.
참고 자료 : https://ooeunz.tistory.com/94
백엔드팀은 WebRTC를 위한 시그널링 서버 부분으로 요청타입에 따라 분기 처리하는 SignalHandler 클래스를 구현했다.
그런데, 시그널링 과정에서 비정상적으로 웹소켓 연결이 끊기는 이슈가 발생했다. 기술 멘토링에서 시니어 개발자 님도 피드백을 주셨던 부분인데, 근본적으로 웹소켓이 끊기는 이유를 찾아야 한다고 하셨다. 그런데, 도무지 모르겠다. ㅠ
왜 끊기는지 우리도 몰루 ... ㅠ ㅠ알려주세요 ... !
그래서 구글링을 통해 임시 방편으로 동기화 키워드인 synchronized에 대해 알게 되었다.
아래 첨부된 코드와 같이 sendMessege 메소드 자체에 스레드를 동기화 시켜주는 synchronized를 걸어주었다.
이 방법은 성능을 많이 저하 시키기 때문에 최대한 지양해야 하는 방법이다. 그렇지만 다른 방법 모두 실패했기 때문에, 일단 사용!
// 기능 : WebRTC를 위한 시그널링 서버 부분으로 요청타입에 따라 분기 처리
@Slf4j
@Component
public class SignalHandler extends TextWebSocketHandler {
@Autowired
private GameRoomService gameRoomService;
@Autowired
private RepositoryService repositoryService;
private final SessionRepository sessionRepository = SessionRepository.getInstance(); // 세션 데이터 저장소
private final ObjectMapper objectMapper = new ObjectMapper();
private static final String MSG_TYPE_JOIN_ROOM = "join_room";
private static final String MSG_TYPE_OFFER = "offer";
private static final String MSG_TYPE_ANSWER = "answer";
private static final String MSG_TYPE_CANDIDATE = "candidate";
@Override
public void afterConnectionEstablished(final WebSocketSession session) {
// 웹소켓이 연결되면 실행되는 메소드
}
// 시그널링 처리 메소드
@Override
protected void handleTextMessage(final WebSocketSession session, final TextMessage textMessage) {
try {
WebSocketMessage message = objectMapper.readValue(textMessage.getPayload(), WebSocketMessage.class);
String userName = message.getSender();
Long roomId = message.getRoomId();
switch (message.getType()) {
// 처음 입장
case MSG_TYPE_JOIN_ROOM:
if (sessionRepository.hasRoom(roomId)) {
// 해당 챗룸이 존재하면
// 세션 저장 1) : 게임방 안의 session List에 새로운 Client session정보를 저장
sessionRepository.addClient(roomId, session);
} else {
// 해당 챗룸이 존재하지 않으면
// 세션 저장 1) : 새로운 게임방 정보와 새로운 Client session정보를 저장
sessionRepository.addClientInNewRoom(roomId, session);
}
// 세션 저장 2) : 이 세션이 어느 방에 들어가 있는지 저장
sessionRepository.saveRoomIdToSession(session, roomId);
// 세션 저장 3) : 방 안에 닉네임들 저장
sessionRepository.addNicknameInRoom(session.getId(), message.getNickname());
Map<String, WebSocketSession> joinClientList = sessionRepository.getClientList(roomId);
// 방안 참가자 중 자신을 제외한 나머지 사람들의 Session ID를 List로 저장
List<String> exportSessionList = new ArrayList<>();
for (Map.Entry<String, WebSocketSession> entry : joinClientList.entrySet()) {
if (entry.getValue() != session) {
exportSessionList.add(entry.getKey());
}
}
Map<String, String> exportNicknameList = new HashMap<>();
for (Map.Entry<String, WebSocketSession> entry : joinClientList.entrySet()) {
if (entry.getValue() != session) {
exportNicknameList.put(entry.getKey(), sessionRepository.getNicknameInRoom(entry.getKey()));
}
}
// 접속한 본인에게 방안 참가자들 정보를 전송
sendMessage(session,
new WebSocketResponseMessage().builder()
.type("all_users")
.sender(userName)
.data(message.getData())
.allUsers(exportSessionList)
.allUsersNickNames(exportNicknameList)
.candidate(message.getCandidate())
.sdp(message.getSdp())
.build());
break;
case MSG_TYPE_OFFER:
case MSG_TYPE_ANSWER:
case MSG_TYPE_CANDIDATE:
if (sessionRepository.hasRoom(roomId)) {
Map<String, WebSocketSession> oacClientList = sessionRepository.getClientList(roomId);
if (oacClientList.containsKey(message.getReceiver())) {
WebSocketSession ws = oacClientList.get(message.getReceiver());
sendMessage(ws,
new WebSocketResponseMessage().builder()
.type(message.getType())
.sender(session.getId()) // 보낸사람 session Id
.senderNickName(message.getNickname())
.receiver(message.getReceiver()) // 받을사람 session Id
.data(message.getData())
.offer(message.getOffer())
.answer(message.getAnswer())
.candidate(message.getCandidate())
.sdp(message.getSdp())
.build());
}
} else {
throw new CustomException(CHAT_ROOM_NOT_FOUND);
}
break;
default:
log.info("======================================== DEFAULT");
log.info("============== 들어온 타입 : " + message.getType());
}
} catch (JsonProcessingException e) {
log.info("=================== SignalHandler Json처리 에러 : {} ", e.getMessage());
}
}
// 웹소켓 연결이 끊어지면 실행되는 메소드
@Override
public void afterConnectionClosed(final WebSocketSession session, final CloseStatus status) {
String nickname = sessionRepository.getNicknameInRoom(session.getId());
// 끊어진 세션이 어느방에 있었는지 조회
Long roomId = sessionRepository.getRoomId(session);
// 1) 방 참가자들 세션 정보들 사이에서 삭제
sessionRepository.deleteClient(roomId, session);
// 2) 별도 해당 참가자 세션 정보도 삭제
sessionRepository.deleteRoomIdToSession(session);
// 3) 별도 해당 닉네임 리스트에서도 삭제
sessionRepository.deleteNicknameInRoom(session.getId());
// 본인 제외 모두에게 전달
for(Map.Entry<String, WebSocketSession> oneClient : sessionRepository.getClientList(roomId).entrySet()){
sendMessage(oneClient.getValue(),
new WebSocketResponseMessage().builder()
.type("leave")
.sender(session.getId())
.receiver(oneClient.getKey())
.build());
}
Member member = repositoryService.findMemberByNickname(nickname);
List<GameRoomAttendee> gameRoomAttendeeList = repositoryService.findAttendeeByRoomId(roomId);
for(GameRoomAttendee gameRoomAttendee : gameRoomAttendeeList) {
if(nickname.equals(gameRoomAttendee.getMemberNickname())){
gameRoomService.roomExit(roomId, member);
}
}
}
// 메세지 발송
private synchronized void sendMessage(WebSocketSession session, WebSocketResponseMessage message) {
try {
String json = objectMapper.writeValueAsString(message);
session.sendMessage(new TextMessage(json));
} catch (IOException e) {
log.info("============== 발생한 에러 메세지: {}", e.getMessage());
}
}
}