connection.go 7.5 KB

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