websocket 增加多分组 fork https://github.com/olahol/melody
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

184 lines
4.5 KiB

10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
9 years ago
10 years ago
9 years ago
10 years ago
10 years ago
10 years ago
9 years ago
10 years ago
9 years ago
10 years ago
9 years ago
9 years ago
9 years ago
  1. # melody
  2. [![Build Status](https://travis-ci.org/olahol/melody.svg)](https://travis-ci.org/olahol/melody)
  3. [![Coverage Status](https://img.shields.io/coveralls/olahol/melody.svg?style=flat)](https://coveralls.io/r/olahol/melody)
  4. [![GoDoc](https://godoc.org/github.com/olahol/melody?status.svg)](https://godoc.org/github.com/olahol/melody)
  5. > :notes: Minimalist websocket framework for Go.
  6. Melody is websocket framework based on [github.com/gorilla/websocket](https://github.com/gorilla/websocket)
  7. that abstracts away the tedious parts of handling websockets. It gets out of
  8. your way so you can write real-time apps. Features include:
  9. * [x] Clear and easy interface similar to `net/http` or Gin.
  10. * [x] A simple way to broadcast to all or selected connected sessions.
  11. * [x] Message buffers making concurrent writing safe.
  12. * [x] Automatic handling of ping/pong and session timeouts.
  13. * [x] Store data on sessions.
  14. ## Install
  15. ```bash
  16. go get gopkg.in/olahol/melody.v1
  17. ```
  18. ## [Example: chat](https://github.com/olahol/melody/tree/master/examples/chat)
  19. [![Chat](https://cdn.rawgit.com/olahol/melody/master/examples/chat/demo.gif "Demo")](https://github.com/olahol/melody/tree/master/examples/chat)
  20. Using [Gin](https://github.com/gin-gonic/gin):
  21. ```go
  22. package main
  23. import (
  24. "github.com/gin-gonic/gin"
  25. "gopkg.in/olahol/melody.v1"
  26. "net/http"
  27. )
  28. func main() {
  29. r := gin.Default()
  30. m := melody.New()
  31. r.GET("/", func(c *gin.Context) {
  32. http.ServeFile(c.Writer, c.Request, "index.html")
  33. })
  34. r.GET("/ws", func(c *gin.Context) {
  35. m.HandleRequest(c.Writer, c.Request)
  36. })
  37. m.HandleMessage(func(s *melody.Session, msg []byte) {
  38. m.Broadcast(msg)
  39. })
  40. r.Run(":5000")
  41. }
  42. ```
  43. Using [Echo](https://github.com/labstack/echo):
  44. ```go
  45. package main
  46. import (
  47. "github.com/labstack/echo"
  48. "github.com/labstack/echo/engine/standard"
  49. "github.com/labstack/echo/middleware"
  50. "gopkg.in/olahol/melody.v1"
  51. "net/http"
  52. )
  53. func main() {
  54. e := echo.New()
  55. m := melody.New()
  56. e.Use(middleware.Logger())
  57. e.Use(middleware.Recover())
  58. e.GET("/", func(c echo.Context) error {
  59. http.ServeFile(c.Response().(*standard.Response).ResponseWriter, c.Request().(*standard.Request).Request, "index.html")
  60. return nil
  61. })
  62. e.GET("/ws", func(c echo.Context) error {
  63. m.HandleRequest(c.Response().(*standard.Response).ResponseWriter, c.Request().(*standard.Request).Request)
  64. return nil
  65. })
  66. m.HandleMessage(func(s *melody.Session, msg []byte) {
  67. m.Broadcast(msg)
  68. })
  69. e.Run(standard.New(":5000"))
  70. }
  71. ```
  72. ## [Example: gophers](https://github.com/olahol/melody/tree/master/examples/gophers)
  73. [![Gophers](https://cdn.rawgit.com/olahol/melody/master/examples/gophers/demo.gif "Demo")](https://github.com/olahol/melody/tree/master/examples/gophers)
  74. ```go
  75. package main
  76. import (
  77. "github.com/gin-gonic/gin"
  78. "gopkg.in/olahol/melody.v1"
  79. "net/http"
  80. "strconv"
  81. "strings"
  82. "sync"
  83. )
  84. type GopherInfo struct {
  85. ID, X, Y string
  86. }
  87. func main() {
  88. router := gin.Default()
  89. mrouter := melody.New()
  90. gophers := make(map[*melody.Session]*GopherInfo)
  91. lock := new(sync.Mutex)
  92. counter := 0
  93. router.GET("/", func(c *gin.Context) {
  94. http.ServeFile(c.Writer, c.Request, "index.html")
  95. })
  96. router.GET("/ws", func(c *gin.Context) {
  97. mrouter.HandleRequest(c.Writer, c.Request)
  98. })
  99. mrouter.HandleConnect(func(s *melody.Session) {
  100. lock.Lock()
  101. for _, info := range gophers {
  102. s.Write([]byte("set " + info.ID + " " + info.X + " " + info.Y))
  103. }
  104. gophers[s] = &GopherInfo{strconv.Itoa(counter), "0", "0"}
  105. s.Write([]byte("iam " + gophers[s].ID))
  106. counter += 1
  107. lock.Unlock()
  108. })
  109. mrouter.HandleDisconnect(func(s *melody.Session) {
  110. lock.Lock()
  111. mrouter.BroadcastOthers([]byte("dis "+gophers[s].ID), s)
  112. delete(gophers, s)
  113. lock.Unlock()
  114. })
  115. mrouter.HandleMessage(func(s *melody.Session, msg []byte) {
  116. p := strings.Split(string(msg), " ")
  117. lock.Lock()
  118. info := gophers[s]
  119. if len(p) == 2 {
  120. info.X = p[0]
  121. info.Y = p[1]
  122. mrouter.BroadcastOthers([]byte("set "+info.ID+" "+info.X+" "+info.Y), s)
  123. }
  124. lock.Unlock()
  125. })
  126. router.Run(":5000")
  127. }
  128. ```
  129. ### [More examples](https://github.com/olahol/melody/tree/master/examples)
  130. ## [Documentation](https://godoc.org/github.com/olahol/melody)
  131. ## Contributors
  132. * Ola Holmström (@olahol)
  133. * Shogo Iwano (@shiwano)
  134. * Matt Caldwell (@mattcaldwell)
  135. * Heikki Uljas (@huljas)
  136. * Robbie Trencheny (@robbiet480)
  137. ## FAQ
  138. If you are getting a `403` when trying to connect to your websocket you can [change allow all origin hosts](http://godoc.org/github.com/gorilla/websocket#hdr-Origin_Considerations):
  139. ```go
  140. m := melody.New()
  141. m.Upgrader.CheckOrigin = func(r *http.Request) bool { return true }
  142. ```