myData.go 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. package initialize
  2. import (
  3. "encoding/hex"
  4. "fmt"
  5. "log"
  6. "net"
  7. "os"
  8. "runtime"
  9. "runtime/debug"
  10. "server/dao"
  11. "server/logger"
  12. "server/modbus"
  13. "server/model"
  14. "server/service"
  15. "server/utils"
  16. "strconv"
  17. "strings"
  18. "sync"
  19. "syscall"
  20. "time"
  21. )
  22. type ModbusHandler struct {
  23. queue *modbus.MlQueue
  24. }
  25. var _handlerOnce sync.Once
  26. var _handlerSingle *ModbusHandler
  27. func GetHandler() *ModbusHandler {
  28. _handlerOnce.Do(func() {
  29. _handlerSingle = &ModbusHandler{
  30. queue: modbus.NewQueue(10000),
  31. }
  32. })
  33. return _handlerSingle
  34. }
  35. func InitInductanceTCP() {
  36. lis, err := net.Listen("tcp", ":60001")
  37. if err != nil {
  38. logger.Get().Println(err)
  39. } else {
  40. logger.Get().Println(lis)
  41. model.InductanceTCP = lis
  42. logger.Get().Println("inductanceTCP启动成功")
  43. }
  44. go StartInductanceTCP()
  45. }
  46. func StartInductanceTCP() {
  47. logger.Get().Println("进入StartInductanceTCP")
  48. handler := GetHandler()
  49. for {
  50. logger.Get().Println("进入InductanceTCP循环")
  51. conn, err := model.InductanceTCP.Accept()
  52. if err != nil {
  53. log.Println(err)
  54. }
  55. remoteAddr := conn.RemoteAddr().String()
  56. logger.Get().Printf("lis Accept conn = %s\n", remoteAddr)
  57. model.Mutex.Lock()
  58. deviceId, err := GetDeviceId(conn)
  59. if err != nil {
  60. log.Println("Error getting device ID:", err)
  61. //conn.Close()
  62. model.Mutex.Unlock()
  63. continue
  64. }
  65. model.ConnectionMap1.Store(deviceId, conn)
  66. model.Mutex.Unlock()
  67. // 使用 Load 方法尝试获取连接
  68. if conn1, ok := model.ConnectionMap1.Load(deviceId); ok {
  69. // 成功找到连接
  70. netConn := conn1.(net.Conn)
  71. // 在这里处理 netConn
  72. go handler.ReadAndHandle(netConn, deviceId)
  73. go handler.Handler()
  74. } else {
  75. // 没有找到对应的连接
  76. log.Printf("Connection for key %s not found", deviceId)
  77. }
  78. }
  79. }
  80. func GetDeviceId(conn net.Conn) (id string, err error) {
  81. // 发送 Modbus RTU 帧
  82. //FE 04 03 EE 00 08 85 B2
  83. data, err := utils.WriteAndReadDevice(modbus.ReadDeviceId, conn, 9, 2)
  84. if err != nil {
  85. return
  86. }
  87. return string(data), err
  88. }
  89. func (o *ModbusHandler) ReadAndHandle(conn net.Conn, deviceId string) {
  90. for {
  91. buffer := make([]byte, 1024)
  92. n, err := conn.Read(buffer)
  93. if err != nil {
  94. if isConnReset(err) {
  95. logger.Get().Error("连接被远程主机强制关闭")
  96. } else if os.IsTimeout(err) {
  97. logger.Get().Error("读取操作超时")
  98. } else {
  99. // 处理其他类型的错误
  100. logger.Get().Errorf("读取错误: %s\n", err)
  101. }
  102. return
  103. }
  104. queueData := model.QueueData{
  105. Id: deviceId,
  106. Value: buffer[:n],
  107. }
  108. ok, cnt := o.queue.Put(&queueData)
  109. if ok {
  110. continue
  111. } else {
  112. logger.Get().Errorf("HandlerData:查询队列失败,队列消息数量:%d", cnt)
  113. runtime.Gosched()
  114. }
  115. }
  116. }
  117. func isConnReset(err error) bool {
  118. if opErr, ok := err.(*net.OpError); ok {
  119. if opErr.Err == syscall.ECONNRESET {
  120. return true // Unix-like 系统上的 ECONNRESET
  121. } else if runtime.GOOS == "windows" {
  122. // Windows 上的 WSAECONNRESET 通常是通过错误消息识别的
  123. if se, ok := opErr.Err.(*os.SyscallError); ok {
  124. if errno, ok := se.Err.(syscall.Errno); ok {
  125. if errno == 10054 { // 10054 对应 WSAECONNRESET
  126. return true
  127. }
  128. }
  129. } else if strings.Contains(opErr.Err.Error(), "WSAECONNRESET") {
  130. // 如果错误消息包含 WSAECONNRESET,也认为是连接被重置
  131. return true
  132. }
  133. }
  134. }
  135. return false
  136. }
  137. func (o *ModbusHandler) Handler() interface{} {
  138. defer func() {
  139. if err := recover(); err != nil {
  140. go GetHandler().Handler()
  141. logger.Get().Errorf("MqttHandler.Handler:发生异常:%s", string(debug.Stack()))
  142. }
  143. }()
  144. for {
  145. msg, ok, quantity := o.queue.Get()
  146. if !ok {
  147. time.Sleep(10 * time.Millisecond)
  148. continue
  149. } else if quantity > 1000 {
  150. logger.Get().Errorf("数据队列累积过多,请注意优化,当前队列条数:%d", quantity)
  151. }
  152. queueData, ok := msg.(*model.QueueData)
  153. if !ok {
  154. logger.Get().Errorln("Type assertion failed: msg is not of type model.QueueDat")
  155. return nil
  156. }
  157. // 信息处理返回
  158. parseData(queueData)
  159. // 对数据进行修改
  160. }
  161. }
  162. func parseData(data *model.QueueData) {
  163. reg, dev, err := utils.GetDataByDeviceId(data.Id)
  164. if err != nil {
  165. logger.Get().Errorln("Error getting register and device:", err)
  166. return
  167. }
  168. toString := hex.EncodeToString(data.Value)
  169. logger.Get().Infof("deviceId: %s, value: %s", data.Id, toString)
  170. switch toString[0:2] {
  171. case "fe":
  172. switch toString[2:8] { // 开关灯
  173. case "050000", "050001", "050002", "050003", "050004", "050005", "050006", "050007":
  174. relyId, _ := strconv.Atoi(toString[7:8])
  175. logger.Get().Error(dev.DeviceLoops[relyId])
  176. if toString[8:12] == "0000" {
  177. dev.DeviceLoops[relyId].State = 0
  178. } else if toString[8:12] == "ff00" {
  179. dev.DeviceLoops[relyId].State = 1
  180. }
  181. for i, device := range reg.Devices {
  182. if device.Sn == data.Id {
  183. reg.Devices[i] = dev
  184. }
  185. }
  186. regions, err := utils.SaveRegionOnData(reg)
  187. err = service.SaveData(regions)
  188. if err != nil {
  189. logger.Get().Errorln("设备回路状态" + err.Error())
  190. return
  191. }
  192. case "0f0000":
  193. OperationCommand := hex.EncodeToString(modbus.OperationCommand)
  194. logger.Get().Infof("deviceId: %s, value: %s", data.Id, OperationCommand)
  195. if OperationCommand[14:16] == "00" {
  196. for i, loop := range dev.DeviceLoops {
  197. loop.State = 0
  198. dev.DeviceLoops[i] = loop
  199. }
  200. } else if OperationCommand[14:16] == "ff" {
  201. for i, loop := range dev.DeviceLoops {
  202. loop.State = 1
  203. dev.DeviceLoops[i] = loop
  204. }
  205. }
  206. for i, device := range reg.Devices {
  207. if device.Sn == data.Id {
  208. reg.Devices[i] = dev
  209. }
  210. }
  211. regions, err := utils.SaveRegionOnData(reg)
  212. err = service.SaveData(regions)
  213. if err != nil {
  214. logger.Get().Errorln("设备回路状态" + err.Error())
  215. return
  216. }
  217. }
  218. switch toString[2:6] {
  219. case "0101":
  220. // 将16进制字符串解码为字节切片
  221. bytes, err := hex.DecodeString(toString[6:8])
  222. if err != nil {
  223. fmt.Println("解码失败:", err)
  224. return
  225. }
  226. // 转换为二进制字符串
  227. binStr := ""
  228. for _, b := range bytes {
  229. // 使用fmt.Sprintf将每个字节转换为8位的二进制字符串
  230. binStr += fmt.Sprintf("%08b", b)
  231. }
  232. for i, device := range reg.Devices {
  233. if device.Sn == data.Id {
  234. reg.Devices[i].State = 1
  235. reg.Devices[i].OnlineTime = time.Now()
  236. for j := len(device.DeviceLoops) - 1; j >= 0; j-- {
  237. reg.Devices[i].DeviceLoops[7-j].State = int(binStr[j] - '0')
  238. }
  239. }
  240. }
  241. regions, err := utils.SaveRegionOnData(reg)
  242. err = service.SaveData(regions)
  243. if err != nil {
  244. logger.Get().Errorln("设备回路状态" + err.Error())
  245. return
  246. }
  247. }
  248. case "11":
  249. switch toString[2:6] {
  250. case "0336":
  251. batteryVoltage, _ := strconv.ParseInt(toString[6:10], 16, 64)
  252. batteryCurrent, _ := strconv.ParseInt(toString[10:14], 16, 64)
  253. batteryPlateVoltage, _ := strconv.ParseInt(toString[38:42], 16, 64)
  254. sun := dao.Sun{
  255. DeviceId: dev.Sn,
  256. BatteryVoltage: float64(batteryVoltage) / 100,
  257. BatteryCurrent: int(batteryCurrent),
  258. BatteryPlateVoltage: float64(batteryPlateVoltage) / 100,
  259. }
  260. for i, device := range reg.Devices {
  261. if device.Sn == dev.Sn {
  262. reg.Devices[i].Sun = sun
  263. }
  264. }
  265. regions, err := utils.SaveRegionOnData(reg)
  266. err = service.SaveData(regions)
  267. if err != nil {
  268. logger.Get().Println("太阳能数据保存" + err.Error())
  269. return
  270. }
  271. //电池
  272. if float64(batteryVoltage)/100 < 5 {
  273. data1 := modbus.DeviceSwitch(8, 1)
  274. if conn1, ok := model.ConnectionMap1.Load(data.Id); ok {
  275. // 成功找到连接
  276. netConn := conn1.(net.Conn)
  277. utils.WriteDevice(data1, netConn)
  278. } else {
  279. // 没有找到对应的连接
  280. logger.Get().Printf("Connection for key %s not found", data.Id)
  281. }
  282. } else {
  283. data1 := modbus.DeviceSwitch(8, 0)
  284. if conn1, ok := model.ConnectionMap1.Load(data.Id); ok {
  285. // 成功找到连接
  286. netConn := conn1.(net.Conn)
  287. utils.WriteDevice(data1, netConn)
  288. } else {
  289. // 没有找到对应的连接
  290. logger.Get().Printf("Connection for key %s not found", data.Id)
  291. }
  292. }
  293. }
  294. }
  295. switch toString[0:4] {
  296. case "4c43":
  297. bytes, err := hex.DecodeString(toString[4:])
  298. if err != nil {
  299. logger.Get().Errorln("Error decoding bytes:", err)
  300. return
  301. }
  302. for i, device := range reg.Devices {
  303. if device.Sn == string(bytes) {
  304. reg.Devices[i].State = 1
  305. reg.Devices[i].OnlineTime = time.Now()
  306. }
  307. }
  308. regions, err := utils.SaveRegionOnData(reg)
  309. err = service.SaveData(regions)
  310. if err != nil {
  311. logger.Get().Errorln("心跳" + err.Error())
  312. return
  313. }
  314. }
  315. }