2017년 3월 5일 일요일

Meteor에서 접속(onConnection)/이탈(onClose) 관리에 대해 알아보자.

Meteor에서 로그인과는 상관없이 어떤 유저가 접속해 있고 이탈했는지를 추적하고 싶어서 정리,
서버사이드에서 아래와 같이 작성했다.
Meteor.startup ->
  console.log "server initiated"
  ConnectedUsers.remove {}
Meteor.onConnection (o)->
  console.log "session [#{o.id}] is connected"
  ConnectedUsers.insert userId: o.id
  o.onClose (p)->
    ConnectedUsers.remove userId: o.id
Meteor.methods
  "getSessionId": ->
    @connection.id
Meteor.publish "list.connectedUsers", ->
  ConnectedUsers.find()
결론은 이정도이지 않나 싶다.
mongodb를 사용하지 않고도 해보았는데 지나치게 publish쪽과 onConnection쪽이 복잡하고 일반화하기 좋지 않아 일단 보류.
https://gist.github.com/makrem025/35543cd60aa88ca49fa0 의 예에서 응용하여 onConnection 일때 publish 의 added, remove로 쏘게 하면 되긴 함. 메모리 디비는 사랑입니다 <3

접속자 관리 관련한 기존 package(aka. mizzao:user-status)들은 지나치게 User collection을 괴롭혀서 성능을 떨어뜨린다.
생선을 낚으라고 준 낚시대인데 잡아야 하는 게 상어라면 분질러 버리고 작살을 만드는 게 맞다. (뭐? 상어도 낚시로 잡는다고? 낚알못)
우리는 뾰족하게 DDP를 통해 접근한 소켓(사용자가 아니다. 이름도 몰라요 성도 몰라)에 대해 파악하고 이를 확인하는 것이 전부. 실전은 언제나 case-by-case니까.
Meteor.startup ->
  console.log "server initiated"
  ConnectedUsers.remove {}
먼저 서버 시작시 접속 유저 컬렉션을 깨끗이 비운다. 메모리 디비라면 불필요.
Meteor.onConnection (o)->
  console.log "session [#{o.id}] is connected"
  ConnectedUsers.insert userId: o.id
  o.onClose (p)->
    ConnectedUsers.remove userId: o.id
https://docs.meteor.com/api/connections.html#Meteor-onConnection 말하고 싶은 것은 여기 전부 나와있다.
서버에서 누군가 접속시 Meteor객체의 onConnection 이벤트가 발생한다.
해당 이벤트의 인자안엔 접근 소켓의 UUID를 반환하는 id와 onClose이벤트가 있다.
onConnection에 insert를 구현하고 onClose 시에 remove 하게 하자.

필요하다면 피아구분을 위해 자신의 id를 알아야할 수도 있는데
Meteor.methods
  "getSessionId": ->
    @connection.id
명시적으로 아이디가 궁금하다면 이렇게 하자. 사실 웹에선 Meteor.connection._lastSessionId를 통해 동기화하지만 외부 정적 페이지나 비Meteor환경에선 이게 오히려 간편하다.
method 안에서 this는 connection 객체를 가지고 있고 이는 해당 method를 call한 사용자의 정보와 일치한다.
만일 websocket을 직접 다룰 수 있다면 서버 응답 중
"{"msg":"connected","session":"yECjz3DgK5nyhNMWJ"}"
connected일때 session값을 가로채자.

이게 전부다.
남은 건 ConnectedUsers를 정밀하게 publish해서 권한에 따라 접속여부를 감시하게 하면 된다.
Meteor.publish "list.connectedUsers", ->
  ConnectedUsers.find() # 여기서 조건을 정밀하게!
이상이다.
기회가 되면 mongodb를 사용하지 않고 하는 방법에 대해서도 언급하려고 한다. 끗.