deviceMgr.go 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  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.UpdateScreensStatus(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 data[34:50] != s.info.Sn {
  87. continue
  88. }
  89. s.UpdateInfo(data[82:84], data[86:88], data[90:92])
  90. default:
  91. fmt.Println("读取:", data)
  92. }
  93. }
  94. }
  95. func (s *Device) Handle() {
  96. defer s.conn.Close()
  97. t2 := time.NewTicker(3 * time.Minute) //在线监测
  98. defer t2.Stop()
  99. for {
  100. select {
  101. case <-t2.C:
  102. state := request.DeviceStatus{
  103. Status: 1,
  104. PlayStatus: s.info.PlayStatus,
  105. DisplayStatus: s.info.DisplayStatus,
  106. SourceStatus: s.info.SourceStatus,
  107. }
  108. if time.Now().Add(-2*time.Minute).After(s.lastTime) || s.lastTime.IsZero() {
  109. //离线
  110. state.Status = 0
  111. err := Dev.UpdateScreensStatus(s.info.ScreensCode, state)
  112. if err != nil {
  113. logger.Logger.Error("[Handle] UpdateScreensStatus err", zap.Error(err))
  114. continue
  115. }
  116. s.conn.Close()
  117. close(s.stopChan) // 通知协程停止
  118. delete(devices, s.info.Sn) //从内存中移除连接
  119. return // 结束 Handle 协程
  120. } else {
  121. logger.Logger.Debugf("%s在线", s.info.ScreensCode)
  122. }
  123. //修改数据库状态
  124. err := Dev.UpdateScreensStatus(s.info.ScreensCode, state)
  125. if err != nil {
  126. logger.Logger.Error("[Handle] UpdateScreensStatus err", zap.Error(err))
  127. continue
  128. }
  129. break
  130. default:
  131. continue
  132. }
  133. }
  134. }
  135. func (s *Device) UpdateInfo(playStatus, displayStatus, sourceStatus string) {
  136. play, display, source := 0, 0, 0
  137. if playStatus == "31" {
  138. play = 1
  139. }
  140. if displayStatus == "30" {
  141. display = 1
  142. }
  143. if sourceStatus == "30" {
  144. source = 1
  145. }
  146. s.info.Status = 1
  147. s.info.PlayStatus = play
  148. s.info.DisplayStatus = display
  149. s.info.SourceStatus = source
  150. s.lastTime = time.Now()
  151. }
  152. func (s *Device) SwitchScreen(onOff int) error {
  153. pack := protocol.SwitchDataPack{Type: 0x52} //熄屏
  154. if onOff == 1 {
  155. pack = protocol.SwitchDataPack{Type: 0x51} //亮屏
  156. }
  157. buf := pack.SwitchScreens() //获取开关屏指令
  158. _, err := s.conn.Write(buf.Bytes())
  159. if err != nil {
  160. logger.Logger.Errorf("SwitchScreen write failed, err:%v", err)
  161. return err
  162. }
  163. return nil
  164. }
  165. func (s *Device) SendInternalCode(content []promodel.InternalCodeContent) error {
  166. pack := protocol.InternalCodeDataPack{}
  167. //获取要写入连接的字节数组
  168. buf := pack.SendInternalCode(content)
  169. _, err := s.conn.Write(buf.Bytes())
  170. if err != nil {
  171. logger.Logger.Errorf("SendInternalCode write failed, err:%v", err)
  172. return err
  173. }
  174. return nil
  175. }
  176. func (s *Device) VoiceBroad(broad string) error {
  177. pack := protocol.VoiceBroadDataPack{}
  178. buf := pack.VoiceBroad(broad)
  179. _, err := s.conn.Write(buf.Bytes())
  180. if err != nil {
  181. logger.Logger.Errorf("VoiceBroad write failed, err:%v", err)
  182. return err
  183. }
  184. return nil
  185. }
  186. func (s *Device) SetBrightness(bright byte) error {
  187. pack := protocol.SetBrightnessDataPack{}
  188. buf := pack.SetBrightness(bright)
  189. _, err := s.conn.Write(buf.Bytes())
  190. if err != nil {
  191. logger.Logger.Errorf("SetBrightness write failed, err:%v", err)
  192. return err
  193. }
  194. return nil
  195. }
  196. func SwitchScreen(sn string, onOff int) error {
  197. device := devices[sn]
  198. err := device.SwitchScreen(onOff)
  199. if err != nil {
  200. return err
  201. }
  202. return nil
  203. }
  204. func SendInternalCode(sn string, content []promodel.InternalCodeContent) error {
  205. device := devices[sn]
  206. err := device.SendInternalCode(content)
  207. if err != nil {
  208. return err
  209. }
  210. return nil
  211. }
  212. func VoiceBroad(sn string, broad string) error {
  213. device := devices[sn]
  214. err := device.VoiceBroad(broad)
  215. if err != nil {
  216. return err
  217. }
  218. return nil
  219. }
  220. func SetBrightness(sn string, bright int) error {
  221. device := devices[sn]
  222. err := device.SetBrightness(brightnessMap[bright])
  223. if err != nil {
  224. return err
  225. }
  226. return nil
  227. }
  228. var brightnessMap = map[int]byte{
  229. 0: 0x19,
  230. 10: 0x18,
  231. 20: 0x16,
  232. 30: 0x14,
  233. 40: 0x12,
  234. 50: 0x10,
  235. 60: 0x08,
  236. 70: 0x06,
  237. 80: 0x04,
  238. 90: 0x02,
  239. 100: 0x00,
  240. }