Created
May 28, 2021 04:55
-
-
Save mmitou/41816ffa66e0deb6b23e4b9f2281e789 to your computer and use it in GitHub Desktop.
部屋分けwebsocket
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package main | |
import ( | |
"context" | |
"fmt" | |
"time" | |
"github.com/gorilla/websocket" | |
"github.com/labstack/echo/v4" | |
"github.com/rs/zerolog" | |
"github.com/rs/zerolog/log" | |
"github.com/rs/zerolog/pkgerrors" | |
) | |
const ( | |
// Time allowed to write a message to the peer. | |
writeWait = 10 * time.Second | |
// Time allowed to read the next pong message from the peer. | |
pongWait = 60 * time.Second | |
// Send pings to peer with this period. Must be less than pongWait. | |
pingPeriod = (pongWait * 9) / 10 | |
// Maximum message size allowed from peer. | |
maxMessageSize = 512 | |
) | |
var upgrader = websocket.Upgrader{ | |
ReadBufferSize: 1024, | |
WriteBufferSize: 1024, | |
} | |
type wsclient struct { | |
roomID string | |
id string | |
conn *websocket.Conn | |
msg chan<- message | |
} | |
type wsMessage struct { | |
messageType int | |
payload []byte | |
} | |
type message struct { | |
roomID string | |
clientID string | |
wsMessage | |
err error | |
} | |
func (c wsclient) reciever() { | |
defer c.conn.Close() | |
c.conn.SetReadLimit(maxMessageSize) | |
c.conn.SetReadDeadline(time.Now().Add(pongWait)) | |
c.conn.SetPongHandler(func(string) error { c.conn.SetReadDeadline(time.Now().Add(pongWait)); return nil }) | |
for { | |
mt, p, err := c.conn.ReadMessage() | |
if err != nil { | |
c.msg <- message{roomID: c.roomID, clientID: c.id, err: err} | |
return | |
} | |
c.msg <- message{roomID: c.roomID, clientID: c.id, wsMessage: wsMessage{messageType: mt, payload: p}, err: nil} | |
} | |
} | |
func (c wsclient) runSender() chan<- wsMessage { | |
snd := make(chan wsMessage) | |
go func() { | |
ticker := time.NewTicker(pingPeriod) | |
defer func() { | |
c.conn.Close() | |
ticker.Stop() | |
}() | |
for { | |
select { | |
case <-ticker.C: | |
c.conn.SetWriteDeadline(time.Now().Add(writeWait)) | |
if err := c.conn.WriteMessage(websocket.PingMessage, nil); err != nil { | |
c.msg <- message{clientID: c.id, err: err} | |
return | |
} | |
case m, ok := <-snd: | |
if !ok { | |
return | |
} | |
log.Debug().Str("clientID", c.id).Str("message", string(m.payload)).Msg("write") | |
if err := c.conn.WriteMessage(m.messageType, m.payload); err != nil { | |
c.msg <- message{roomID: c.roomID, clientID: c.id, err: err} | |
return | |
} | |
} | |
} | |
}() | |
return snd | |
} | |
func (c wsclient) connect(msg chan<- message) chan<- wsMessage { | |
c.msg = msg | |
go c.reciever() | |
snd := c.runSender() | |
return snd | |
} | |
func runEchoServer(ctx context.Context) chan<- wsclient { | |
register := make(chan wsclient) | |
go func() { | |
snds := make(map[string]map[string]chan<- wsMessage) | |
msg := make(chan message) | |
for { | |
select { | |
case <-ctx.Done(): | |
return | |
case client := <-register: | |
log.Debug().Str("roomID", client.roomID).Str("clientID", client.id).Msg("register") | |
snd := client.connect(msg) | |
if _, ok := snds[client.roomID]; !ok { | |
snds[client.roomID] = make(map[string]chan<- wsMessage) | |
} | |
snds[client.roomID][client.id] = snd | |
case m := <-msg: | |
log.Debug().Msg(string(m.payload)) | |
if m.err != nil { | |
log.Debug().Err(m.err).Msg("") | |
close(snds[m.roomID][m.clientID]) | |
delete(snds[m.roomID], m.clientID) | |
if len(snds[m.roomID]) == 0 { | |
delete(snds, m.roomID) | |
} | |
continue | |
} | |
for id, snd := range snds[m.roomID] { | |
log.Debug().Str("clientID", id).Str("message", string(m.payload)).Msg("send") | |
snd <- wsMessage{messageType: m.messageType, payload: m.payload} | |
} | |
} | |
} | |
}() | |
return register | |
} | |
func hub(registrar chan<- wsclient) func(c echo.Context) error { | |
i := 0 | |
return func(c echo.Context) error { | |
i++ | |
w := c.Response() | |
r := c.Request() | |
roomID := c.Param("id") | |
conn, err := upgrader.Upgrade(w, r, nil) | |
if err != nil { | |
return err | |
} | |
registrar <- wsclient{roomID: roomID, id: fmt.Sprintf("hello%d", i), conn: conn} | |
return nil | |
} | |
} | |
func main() { | |
zerolog.TimeFieldFormat = time.RFC3339Nano | |
zerolog.ErrorStackMarshaler = pkgerrors.MarshalStack | |
zerolog.SetGlobalLevel(zerolog.DebugLevel) | |
e := echo.New() | |
ctx, cancel := context.WithCancel(context.Background()) | |
defer cancel() | |
registrar := runEchoServer(ctx) | |
e.HTTPErrorHandler = func(err error, c echo.Context) { | |
log.Debug().Err(err).Msg(fmt.Sprintf("%+v", err)) | |
e.DefaultHTTPErrorHandler(err, c) | |
} | |
e.GET("/rooms/:id/ws", hub(registrar)) | |
e.Static("/", "./web") | |
e.Logger.Fatal(e.Start(":8080")) | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment