connection.go 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. package mqtt
  2. import (
  3. "encoding/hex"
  4. "errors"
  5. "net"
  6. "sparrow/pkg/rpcs"
  7. "sparrow/pkg/server"
  8. "time"
  9. )
  10. // const def
  11. const (
  12. SendChanLen = 16
  13. defaultKeepAlive = 30
  14. )
  15. // ResponseType response type
  16. type ResponseType struct {
  17. SendTime uint8
  18. PublishType uint8
  19. DataType string
  20. }
  21. // Connection client connection
  22. type Connection struct {
  23. Mgr *Manager
  24. DeviceID uint64
  25. Conn net.Conn
  26. SendChan chan Message
  27. MessageID uint16
  28. MessageWaitChan map[uint16]chan error
  29. KeepAlive uint16
  30. LastHbTime int64
  31. Token []byte
  32. }
  33. // NewConnection create a connection
  34. func NewConnection(conn net.Conn, mgr *Manager) *Connection {
  35. sendchan := make(chan Message, SendChanLen)
  36. c := &Connection{
  37. Conn: conn,
  38. SendChan: sendchan,
  39. Mgr: mgr,
  40. KeepAlive: defaultKeepAlive,
  41. MessageWaitChan: make(map[uint16]chan error),
  42. }
  43. go c.SendMsgToClient()
  44. go c.RcvMsgFromClient()
  45. return c
  46. }
  47. // Submit submit a message to send chan
  48. func (c *Connection) Submit(msg Message) {
  49. if c.Conn != nil {
  50. c.SendChan <- msg
  51. }
  52. }
  53. // Publish will publish a message , and return a chan to wait for completion.
  54. func (c *Connection) Publish(msg Message, timeout time.Duration) error {
  55. server.Log.Debugf("publishing message : %v, timeout %v", msg, timeout)
  56. message := msg.(*Publish)
  57. message.MessageID = c.MessageID
  58. c.MessageID++
  59. c.Submit(message)
  60. ch := make(chan error)
  61. // we don't wait for confirm.
  62. if timeout == 0 {
  63. return nil
  64. }
  65. c.MessageWaitChan[message.MessageID] = ch
  66. // wait for timeout and
  67. go func() {
  68. timer := time.NewTimer(timeout)
  69. <-timer.C
  70. waitCh, exist := c.MessageWaitChan[message.MessageID]
  71. if exist {
  72. waitCh <- errors.New("timeout pushlishing message.")
  73. delete(c.MessageWaitChan, message.MessageID)
  74. close(waitCh)
  75. }
  76. }()
  77. err := <-ch
  78. return err
  79. }
  80. func (c *Connection) confirmPublish(MessageID uint16) {
  81. waitCh, exist := c.MessageWaitChan[MessageID]
  82. if exist {
  83. waitCh <- nil
  84. delete(c.MessageWaitChan, MessageID)
  85. close(waitCh)
  86. }
  87. }
  88. func (c *Connection) ValidateToken(token []byte) error {
  89. err := c.Mgr.Provider.ValidateDeviceToken(c.DeviceID, token)
  90. if err != nil {
  91. return err
  92. }
  93. c.Token = token
  94. return nil
  95. }
  96. func (c *Connection) Close() {
  97. DeviceID := c.DeviceID
  98. server.Log.Infof("closing connection of device %v", DeviceID)
  99. if c.Conn != nil {
  100. c.Conn.Close()
  101. c.Conn = nil
  102. c.Mgr.Provider.OnDeviceOffline(DeviceID)
  103. }
  104. if c.SendChan != nil {
  105. close(c.SendChan)
  106. c.SendChan = nil
  107. }
  108. }
  109. func (c *Connection) RcvMsgFromClient() {
  110. conn := c.Conn
  111. host := conn.RemoteAddr().String()
  112. server.Log.Infof("recieve new connection from %s", host)
  113. for {
  114. msg, err := DecodeOneMessage(conn)
  115. if err != nil {
  116. server.Log.Errorf("read error: %s", err)
  117. c.Close()
  118. return
  119. }
  120. server.Log.Infof("%s, come msg===\n%v\n=====", host, msg)
  121. c.LastHbTime = time.Now().Unix()
  122. switch msg := msg.(type) {
  123. case *Connect:
  124. ret := RetCodeAccepted
  125. if msg.ProtocolVersion == 3 && msg.ProtocolName != "MQIsdp" {
  126. ret = RetCodeUnacceptableProtocolVersion
  127. } else if msg.ProtocolVersion == 4 && msg.ProtocolName != "MQTT" {
  128. ret = RetCodeUnacceptableProtocolVersion
  129. } else if msg.ProtocolVersion > 4 {
  130. ret = RetCodeUnacceptableProtocolVersion
  131. }
  132. if len(msg.ClientID) < 1 || len(msg.ClientID) > 23 {
  133. server.Log.Warn("invalid ClientID length: %d", len(msg.ClientID))
  134. ret = RetCodeIdentifierRejected
  135. c.Close()
  136. return
  137. }
  138. DeviceID, err := ClientIDToDeviceID(msg.ClientID)
  139. if err != nil {
  140. server.Log.Warn("invalid Identify: %d", ret)
  141. c.Close()
  142. return
  143. }
  144. c.DeviceID = DeviceID
  145. token, err := hex.DecodeString(msg.Password)
  146. if err != nil {
  147. server.Log.Warn("token format error : %v", err)
  148. ret = RetCodeNotAuthorized
  149. c.Close()
  150. return
  151. }
  152. err = c.ValidateToken(token)
  153. if err != nil {
  154. server.Log.Warn("validate token error : %v", err)
  155. ret = RetCodeNotAuthorized
  156. c.Close()
  157. return
  158. }
  159. if ret != RetCodeAccepted {
  160. server.Log.Warn("invalid CON: %d", ret)
  161. c.Close()
  162. return
  163. }
  164. args := rpcs.ArgsGetOnline{
  165. Id: c.DeviceID,
  166. ClientIP: host,
  167. AccessRPCHost: server.GetRPCHost(),
  168. HeartbeatInterval: uint32(c.KeepAlive),
  169. }
  170. c.Mgr.AddConn(c.DeviceID, c)
  171. connack := &ConnAck{
  172. ReturnCode: ret,
  173. }
  174. c.Submit(connack)
  175. c.KeepAlive = msg.KeepAliveTimer
  176. err = c.Mgr.Provider.OnDeviceOnline(args)
  177. if err != nil {
  178. server.Log.Warn("device online error : %v", err)
  179. c.Close()
  180. return
  181. }
  182. server.Log.Infof("device %d, connected to server now, host: %s", c.DeviceID, host)
  183. case *Publish:
  184. server.Log.Infof("%s, publish topic: %s", host, msg.TopicName)
  185. c.Mgr.PublishMessage2Server(c.DeviceID, msg)
  186. if msg.QosLevel.IsAtLeastOnce() {
  187. server.Log.Infof("publish ack send now")
  188. publishack := &PubAck{MessageID: msg.MessageID}
  189. c.Submit(publishack)
  190. } else if msg.QosLevel.IsExactlyOnce() {
  191. server.Log.Infof("publish Rec send now")
  192. publishRec := &PubRec{MessageID: msg.MessageID}
  193. c.Submit(publishRec)
  194. }
  195. err := c.Mgr.Provider.OnDeviceHeartBeat(c.DeviceID)
  196. if err != nil {
  197. server.Log.Warnf("%s, heartbeat set error %s, close now...", host, err)
  198. c.Close()
  199. return
  200. }
  201. case *PubAck:
  202. server.Log.Infof("%s, comes publish ack", host)
  203. c.confirmPublish(msg.MessageID)
  204. err := c.Mgr.Provider.OnDeviceHeartBeat(c.DeviceID)
  205. if err != nil {
  206. server.Log.Warnf("%s, heartbeat set error %s, close now...", host, err)
  207. c.Close()
  208. return
  209. }
  210. case *PubRec:
  211. server.Log.Infof("%s, comes publish rec", host)
  212. publishRel := &PubRel{MessageID: msg.MessageID}
  213. c.Submit(publishRel)
  214. case *PubRel:
  215. server.Log.Infof("%s, comes publish rel", host)
  216. publishCom := &PubComp{MessageID: msg.MessageID}
  217. c.Submit(publishCom)
  218. case *PubComp:
  219. server.Log.Infof("%s, comes publish comp", host)
  220. c.confirmPublish(msg.MessageID)
  221. err := c.Mgr.Provider.OnDeviceHeartBeat(c.DeviceID)
  222. if err != nil {
  223. server.Log.Warnf("%s, heartbeat set error %s, close now...", host, err)
  224. c.Close()
  225. return
  226. }
  227. case *PingReq:
  228. server.Log.Infof("%s, ping req comes", host)
  229. pingrsp := &PingResp{}
  230. err := c.Mgr.Provider.OnDeviceHeartBeat(c.DeviceID)
  231. if err != nil {
  232. server.Log.Warnf("%s, heartbeat set error %s, close now...", host, err)
  233. c.Close()
  234. return
  235. }
  236. c.Submit(pingrsp)
  237. case *Subscribe:
  238. server.Log.Infof("%s, subscribe topic: %v", host, msg.Topics)
  239. case *Unsubscribe:
  240. server.Log.Infof("%s, unsubscribe topic: %v", host, msg.Topics)
  241. case *Disconnect:
  242. server.Log.Infof("%s, disconnect now, exit...", host)
  243. c.Close()
  244. return
  245. default:
  246. server.Log.Errorf("unknown msg type %T", msg)
  247. c.Close()
  248. return
  249. }
  250. }
  251. }
  252. func (c *Connection) SendMsgToClient() {
  253. host := c.Conn.RemoteAddr()
  254. for {
  255. msg, ok := <-c.SendChan
  256. if !ok {
  257. server.Log.Errorf("%s is end now", host)
  258. return
  259. }
  260. server.Log.Debugf("send msg to %s=======\n%v\n=========", host, msg)
  261. err := msg.Encode(c.Conn)
  262. if err != nil {
  263. server.Log.Errorf("send msg err: %s=====\n%v\n=====", err, msg)
  264. continue
  265. }
  266. }
  267. }