deviceMgr.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. package tcp
  2. import (
  3. "encoding/hex"
  4. "errors"
  5. "fmt"
  6. "go.uber.org/zap"
  7. "gorm.io/gorm"
  8. "net"
  9. Dev "server/dao/devices"
  10. deviceDao "server/dao/devices"
  11. promodel "server/model/common/devices"
  12. "server/model/common/request"
  13. "server/protocol"
  14. "server/utils/logger"
  15. "strings"
  16. "time"
  17. )
  18. var (
  19. devices = make(map[string]Device) // 存储连接的全局 map
  20. )
  21. type Device struct {
  22. info deviceDao.Screens
  23. conn net.Conn
  24. lastTime time.Time
  25. stopChan chan struct{} // 添加停止通道
  26. }
  27. func (s *Device) Start(conn net.Conn) {
  28. s.conn = conn
  29. s.stopChan = make(chan struct{})
  30. go s.Process()
  31. go s.Handle()
  32. }
  33. func (s *Device) Process() {
  34. // 函数执行完之后关闭连接
  35. defer s.conn.Close()
  36. for {
  37. buf := make([]byte, 256)
  38. // 将tcp连接读取到的数据读取到byte数组中, 返回读取到的byte的数目
  39. n, err := s.conn.Read(buf)
  40. if err != nil {
  41. // 从客户端读取数据的过程中发生错误 这里如果没读到可以视为设备离线了
  42. logger.Logger.Errorf("read from client failed [离线了], err:%v", err)
  43. break
  44. }
  45. data := hex.EncodeToString(buf[:n])
  46. if !strings.Contains(strings.ToLower(data), "fe5c4b89") {
  47. continue
  48. }
  49. switch data[16:18] {
  50. case "61":
  51. logger.Logger.Debug("登录")
  52. // fe5c4b89 2a000000 62 00000000 17000000 31 23 32303233(年) 3035(月) 3135(日) 3031(星期) 3038(时) 3334(分) 3135(秒) 23 303630(心跳包时间) 23 ffff
  53. buffer := protocol.AuthDataPack{}.AuthLogin()
  54. // 通过sn 查设备是否存在 不存在return 存在就保存con 和 info
  55. screens, err := deviceDao.QueryScreenBySn(data[34:50])
  56. if errors.Is(err, gorm.ErrRecordNotFound) {
  57. logger.Logger.Errorf("Process[case '61'] SN not found \n")
  58. break
  59. }
  60. err = Dev.UpdateScreensStatusAndOnline(screens.ScreensCode, request.DeviceStatus{
  61. Status: 1,
  62. PlayStatus: 1,
  63. DisplayStatus: 1,
  64. SourceStatus: 1,
  65. })
  66. if err != nil {
  67. logger.Logger.Error("[Handle] UpdateScreensStatus err", zap.Error(err))
  68. continue
  69. }
  70. s.info = screens
  71. devices[s.info.Sn] = *s
  72. s.conn.Write(buffer.Bytes())
  73. case "91":
  74. logger.Logger.Debug("心跳")
  75. //判断内存devices中是否存储了设备(连接)信息
  76. if _, exists := devices[data[34:50]]; !exists {
  77. //不存在
  78. screens, err := deviceDao.QueryScreenBySn(data[34:50])
  79. if errors.Is(err, gorm.ErrRecordNotFound) {
  80. logger.Logger.Errorf("Process[case '91'] SN not found \n")
  81. break
  82. }
  83. s.info = screens
  84. devices[s.info.Sn] = *s
  85. //如果是未登录状态,则登录并更新在线时间
  86. if screens.IsLogin == 0 {
  87. _ = Dev.UpdateScreensStatusAndOnline(screens.ScreensCode, request.DeviceStatus{
  88. Status: s.info.Status,
  89. PlayStatus: s.info.PlayStatus,
  90. DisplayStatus: s.info.DisplayStatus,
  91. SourceStatus: s.info.SourceStatus,
  92. })
  93. }
  94. }
  95. if data[34:50] != s.info.Sn {
  96. continue
  97. }
  98. s.UpdateInfo(data[82:84], data[86:88], data[90:92])
  99. default:
  100. fmt.Println("读取:", data)
  101. }
  102. }
  103. }
  104. func (s *Device) Handle() {
  105. defer s.conn.Close()
  106. t2 := time.NewTicker(3 * time.Minute) //在线监测
  107. defer t2.Stop()
  108. for {
  109. select {
  110. case <-t2.C:
  111. state := request.DeviceStatus{
  112. Status: 1,
  113. PlayStatus: s.info.PlayStatus,
  114. DisplayStatus: s.info.DisplayStatus,
  115. SourceStatus: s.info.SourceStatus,
  116. }
  117. if time.Now().Add(-2*time.Minute).After(s.lastTime) || s.lastTime.IsZero() {
  118. //离线
  119. state.Status = 0
  120. err := Dev.UpdateScreensStatusAndOffline(s.info.ScreensCode, state)
  121. if err != nil {
  122. logger.Logger.Error("[Handle] UpdateScreensStatus err", zap.Error(err))
  123. continue
  124. }
  125. s.conn.Close()
  126. close(s.stopChan) // 通知协程停止
  127. delete(devices, s.info.Sn) //从内存中移除连接
  128. return // 结束 Handle 协程
  129. } else {
  130. logger.Logger.Debugf("%s在线", s.info.ScreensCode)
  131. }
  132. //修改数据库状态
  133. err := Dev.UpdateScreensStatus(s.info.ScreensCode, state)
  134. if err != nil {
  135. logger.Logger.Error("[Handle] UpdateScreensStatus err", zap.Error(err))
  136. continue
  137. }
  138. break
  139. default:
  140. continue
  141. }
  142. }
  143. }
  144. func (s *Device) UpdateInfo(playStatus, displayStatus, sourceStatus string) {
  145. play, display, source := 0, 0, 0
  146. if playStatus == "31" {
  147. play = 1
  148. }
  149. if displayStatus == "30" {
  150. display = 1
  151. }
  152. if sourceStatus == "30" {
  153. source = 1
  154. }
  155. s.info.Status = 1
  156. s.info.PlayStatus = play
  157. s.info.DisplayStatus = display
  158. s.info.SourceStatus = source
  159. s.lastTime = time.Now()
  160. }
  161. func (s *Device) SwitchScreen(onOff int) error {
  162. pack := protocol.SwitchDataPack{Type: 0x52} //熄屏
  163. if onOff == 1 {
  164. pack = protocol.SwitchDataPack{Type: 0x51} //亮屏
  165. }
  166. buf := pack.SwitchScreens() //获取开关屏指令
  167. _, err := s.conn.Write(buf.Bytes())
  168. if err != nil {
  169. logger.Logger.Errorf("SwitchScreen write failed, err:%v", err)
  170. return err
  171. }
  172. return nil
  173. }
  174. func (s *Device) SendInternalCode(content []promodel.InternalCodeContent) error {
  175. pack := protocol.InternalCodeDataPack{}
  176. //获取要写入连接的字节数组
  177. buf := pack.SendInternalCode(content)
  178. _, err := s.conn.Write(buf.Bytes())
  179. if err != nil {
  180. logger.Logger.Errorf("SendInternalCode write failed, err:%v", err)
  181. return err
  182. }
  183. return nil
  184. }
  185. func (s *Device) VoiceBroad(broad string) error {
  186. pack := protocol.VoiceBroadDataPack{}
  187. buf := pack.VoiceBroad(broad)
  188. _, err := s.conn.Write(buf.Bytes())
  189. if err != nil {
  190. logger.Logger.Errorf("VoiceBroad write failed, err:%v", err)
  191. return err
  192. }
  193. return nil
  194. }
  195. func (s *Device) SetBrightness(bright byte) error {
  196. pack := protocol.SetBrightnessDataPack{}
  197. buf := pack.SetBrightness(bright)
  198. _, err := s.conn.Write(buf.Bytes())
  199. if err != nil {
  200. logger.Logger.Errorf("SetBrightness write failed, err:%v", err)
  201. return err
  202. }
  203. return nil
  204. }
  205. func SwitchScreen(sn string, onOff int) error {
  206. device := devices[sn]
  207. err := device.SwitchScreen(onOff)
  208. if err != nil {
  209. return err
  210. }
  211. return nil
  212. }
  213. func SendInternalCode(sn string, content []promodel.InternalCodeContent) error {
  214. device := devices[sn]
  215. err := device.SendInternalCode(content)
  216. if err != nil {
  217. return err
  218. }
  219. return nil
  220. }
  221. func VoiceBroad(sn string, broad string) error {
  222. device := devices[sn]
  223. err := device.VoiceBroad(broad)
  224. if err != nil {
  225. return err
  226. }
  227. return nil
  228. }
  229. func SetBrightness(sn string, bright int) error {
  230. device := devices[sn]
  231. err := device.SetBrightness(brightnessMap[bright])
  232. if err != nil {
  233. return err
  234. }
  235. return nil
  236. }
  237. var brightnessMap = map[int]byte{
  238. 0: 0x19,
  239. 10: 0x18,
  240. 20: 0x16,
  241. 30: 0x14,
  242. 40: 0x12,
  243. 50: 0x10,
  244. 60: 0x08,
  245. 70: 0x06,
  246. 80: 0x04,
  247. 90: 0x02,
  248. 100: 0x00,
  249. }