connection.go 7.6 KB

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