ym485.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613
  1. package main
  2. import (
  3. "context"
  4. "errors"
  5. "runtime/debug"
  6. "time"
  7. "github.com/go-redis/redis/v7"
  8. "github.com/sirupsen/logrus"
  9. "lc/common/mqtt"
  10. "lc/common/protocol"
  11. "lc/common/util"
  12. "lc/edge/ipole/ym485"
  13. "strings"
  14. "sync"
  15. )
  16. var YmProtocol = "yuming485"
  17. var mutex sync.Mutex
  18. var oldstate uint8
  19. type YmLampController struct {
  20. devinfo *protocol.DevInfo
  21. model *protocol.IotModel
  22. downQueue *util.MlQueue
  23. mapTopicHandle map[string]func(m mqtt.Message)
  24. mapLamps2OOT []protocol.CHZB_OnOffTime //时控策略
  25. mapTempLampsOOT *LampTimeRange //临时手动控时间段,手动控制开关灯
  26. mapLampAlarm *LampAlarmInfo //告警数据
  27. State []ym485.DeviceState //当前灯控状态 开关灯,电流,电压,电度
  28. ctx context.Context
  29. cancel context.CancelFunc
  30. chanDevInfo chan *protocol.DevInfo //设备管理更新
  31. chanModelInfo chan *ModelInfo //设备管理更新
  32. tswitch time.Time //手动控执行时间
  33. }
  34. func NewYmLampController(info *protocol.DevInfo) Device {
  35. ctx, cancel := context.WithCancel(context.Background())
  36. dev := &YmLampController{
  37. devinfo: info,
  38. downQueue: util.NewQueue(200),
  39. mapTopicHandle: make(map[string]func(m mqtt.Message)),
  40. ctx: ctx,
  41. cancel: cancel,
  42. chanDevInfo: make(chan *protocol.DevInfo),
  43. chanModelInfo: make(chan *ModelInfo),
  44. }
  45. iot, err := loadModel(info.TID)
  46. if err != nil {
  47. logrus.Errorf("NewYmLampController:加载模型[tid=%d]文件发生错误:%s", info.TID, err.Error())
  48. } else {
  49. if iot.Protocol == YmProtocol {
  50. dev.model = iot
  51. } else {
  52. logrus.Error("NewYmLampController:物模型错误,非YmProtocol协议")
  53. }
  54. }
  55. dev.SetTopicHandle()
  56. return dev
  57. }
  58. func (o *YmLampController) SetTopicHandle() {
  59. o.mapTopicHandle[GetTopic(protocol.DT_LAMPCONTROLLER, o.devinfo.DevCode, protocol.TP_YM_SET_SWITCH)] = o.HandleTpYmSetSwitch
  60. o.mapTopicHandle[GetTopic(protocol.DT_LAMPCONTROLLER, o.devinfo.DevCode, protocol.TP_YM_SET_ONOFFTIME)] = o.HandleTpYmSetOnofftime
  61. }
  62. func (o *YmLampController) MQTTSubscribe() {
  63. GetMQTTMgr().Subscribe(GetTopic(protocol.DT_LAMPCONTROLLER, o.devinfo.DevCode, protocol.TP_YM_SET_SWITCH), mqtt.ExactlyOnce, o.HandleCache, ToAll) //开关灯
  64. GetMQTTMgr().Subscribe(GetTopic(protocol.DT_LAMPCONTROLLER, o.devinfo.DevCode, protocol.TP_YM_SET_ONOFFTIME), mqtt.ExactlyOnce, o.HandleCache, ToCloud) //设置时间
  65. }
  66. func (o *YmLampController) HandleCache(m mqtt.Message) {
  67. logrus.Infof("Topic:%v,message:%v\n", m.Topic(), m.PayloadString())
  68. o.downQueue.Put(m)
  69. }
  70. func (o *YmLampController) Start() {
  71. o.MQTTSubscribe()
  72. retry := 3
  73. sleep := time.Duration(2)
  74. for i := 0; i < retry; i++ {
  75. if err := o.ReloadOOTFromRedis(); err == nil {
  76. break
  77. }
  78. time.Sleep(sleep * time.Second)
  79. }
  80. for i := 0; i < retry; i++ {
  81. if err := o.ReloadSwitchOOTFromRedis(); err == nil {
  82. break
  83. }
  84. time.Sleep(sleep * time.Second)
  85. }
  86. for i := 0; i < retry; i++ {
  87. if err := o.ReloadLampAlarmFromRedis(); err == nil {
  88. break
  89. }
  90. time.Sleep(sleep * time.Second)
  91. }
  92. go o.HandleData()
  93. }
  94. func (o *YmLampController) Stop() {
  95. //停止采集和处理
  96. o.cancel()
  97. }
  98. func (o *YmLampController) UpdateInfo(devinfo protocol.DevInfo) {
  99. o.chanDevInfo <- &devinfo
  100. }
  101. func (o *YmLampController) GetDevInfo() *protocol.DevInfo {
  102. return o.devinfo
  103. }
  104. func (o *YmLampController) UpdateModel(tid uint16, flag int) {
  105. if tid > 0 {
  106. mi := ModelInfo{
  107. TID: tid,
  108. Flag: flag,
  109. }
  110. o.chanModelInfo <- &mi
  111. }
  112. }
  113. func (o *YmLampController) UpdateModel2(mi *ModelInfo) {
  114. if o.devinfo.TID != mi.TID {
  115. return
  116. }
  117. if mi.Flag == 0 {
  118. logrus.Errorf("设备[%s]的物模型[tid=%d]模型文件被删除,下次启动即将生效。", o.devinfo.DevCode, mi.TID)
  119. return
  120. }
  121. logrus.Debugf("YmLampController.UpdateModel2:更新设备[%s]的物模型[%d]", o.devinfo.DevCode, mi.TID)
  122. iot, err := loadModel(mi.TID)
  123. if err != nil {
  124. logrus.Errorf("YmLampController.UpdateModel2:加载模型[%d]文件错误:%s", mi.TID, err.Error())
  125. return
  126. }
  127. if iot.Protocol == ModbusRtuProtocol { //合法的物模型
  128. o.model = iot
  129. logrus.Infof("YmLampController.UpdateModel2:更新设备[%s]的物模型[%d]成功", o.devinfo.DevCode, mi.TID)
  130. } else {
  131. logrus.Error("YmLampController.UpdateModel2:物模型错误,TID和文件名tid不一致或协议非ModbusRTU协议")
  132. }
  133. }
  134. func (o *YmLampController) GetDevType() string {
  135. if o.devinfo.DevType == 3 {
  136. return protocol.DT_LAMPCONTROLLER
  137. }
  138. return "unknown"
  139. }
  140. // HandleData 数据处理协程
  141. func (o *YmLampController) HandleData() {
  142. defer func() {
  143. if err := recover(); err != nil {
  144. logrus.Error("YmLampController.HandleData:panic:", err)
  145. logrus.Error("stack:", string(debug.Stack()))
  146. go o.HandleData()
  147. }
  148. }()
  149. o.QueryLampControllerAddr() //线上挂了两个灯控,则不能
  150. logrus.Infof("lamp state:%d", oldstate)
  151. nextFillTime := time.Time{}
  152. for {
  153. select {
  154. case <-o.ctx.Done():
  155. logrus.Errorf("设备[%s]的HandleData退出,原因:%v", o.devinfo.DevCode, o.ctx.Err())
  156. return
  157. case devinfo_ := <-o.chanDevInfo:
  158. o.devinfo = devinfo_
  159. case mi := <-o.chanModelInfo:
  160. o.UpdateModel2(mi)
  161. default:
  162. //从队列钟获取指令执行
  163. if m, ok, _ := o.downQueue.Get(); ok {
  164. if mm, ok := m.(mqtt.Message); ok {
  165. if fn, ok := o.mapTopicHandle[mm.Topic()]; ok {
  166. fn(mm)
  167. } else {
  168. logrus.Errorf("YmLampController.Handle:不支持的主题:%s", mm.Topic())
  169. }
  170. }
  171. } else {
  172. if nextFillTime.IsZero() || nextFillTime.Before(util.MlNow()) { //定时读取和发送
  173. t := util.MlNow()
  174. o.QueryDeviceState() //获取灯的数据
  175. o.ConfirmState(t) //获取设置灯的开/关
  176. nextFillTime = util.MlNow().Add(time.Duration(o.devinfo.SendCloud) * time.Millisecond)
  177. }
  178. time.Sleep(time.Millisecond * 1000)
  179. }
  180. }
  181. }
  182. }
  183. func (o *YmLampController) SendRecvData(aduRequest []byte, retry int) (aduResponse []byte, err error) {
  184. defer mutex.Unlock()
  185. mutex.Lock()
  186. // time.Sleep(time.Millisecond * 10)
  187. serial := GetSerialMgr().GetSerialPort(o.devinfo.Code)
  188. if serial == nil {
  189. return nil, ErrClosedConnection
  190. }
  191. if retry <= 0 {
  192. retry = 1
  193. }
  194. for ; retry > 0; retry-- {
  195. aduResponse, err = serial.SendRecvData(aduRequest, FlagYm485, o.devinfo.WaitTime)
  196. if err == nil {
  197. break
  198. }
  199. }
  200. return aduResponse, err
  201. }
  202. func (o *YmLampController) SendData(aduRequest []byte, retry int) (err error) {
  203. serial := GetSerialMgr().GetSerialPort(o.devinfo.Code)
  204. if serial == nil {
  205. return ErrClosedConnection
  206. }
  207. if retry <= 0 {
  208. retry = 1
  209. }
  210. for ; retry > 0; retry-- {
  211. if err = serial.SendData(aduRequest, FlagYm485, o.devinfo.WaitTime); err == nil {
  212. break
  213. }
  214. }
  215. return err
  216. }
  217. func (o *YmLampController) QueryLampControllerAddr() {
  218. var obj ym485.QueryAddr
  219. buf, err := obj.EnCode()
  220. defer ym485.ReleaseByteBuffer(buf)
  221. if err != nil {
  222. return
  223. }
  224. recvbuf, err := o.SendRecvData(buf.Bytes(), 2)
  225. if err != nil {
  226. return
  227. }
  228. var ret ym485.QueryAddrACK
  229. if err = ret.DeCode(recvbuf); err == nil {
  230. logrus.Infof("灯控地址:%s", ret.Addr)
  231. }
  232. if ret.Addr != o.devinfo.DevCode {
  233. logrus.Errorf("请将DevCode配置为灯控地址!")
  234. }
  235. }
  236. func (o *YmLampController) QuerySoftVer() {
  237. var obj ym485.PackingMark
  238. buf, err := obj.EnCode(o.devinfo.DevCode, 0x1E, [4]byte{0x00, 0x00, 0x60, 0x12})
  239. defer ym485.ReleaseByteBuffer(buf)
  240. if err != nil {
  241. return
  242. }
  243. recvbuf, err := o.SendRecvData(buf.Bytes(), 1)
  244. if err != nil {
  245. return
  246. }
  247. var ret ym485.QuerySoftVerAck
  248. if err = ret.DeCode(recvbuf); err == nil {
  249. logrus.Infof("灯控型号:%s,软件版本:%s", ret.Model, ret.Version)
  250. }
  251. }
  252. func (o *YmLampController) QueryBasicsettings() {
  253. var obj ym485.PackingMark
  254. buf, err := obj.EnCode(o.devinfo.DevCode, 0x1E, [4]byte{0x00, 0x00, 0x60, 0x02})
  255. defer ym485.ReleaseByteBuffer(buf)
  256. if err != nil {
  257. return
  258. }
  259. recvbuf, err := o.SendRecvData(buf.Bytes(), 1)
  260. if err != nil {
  261. return
  262. }
  263. var ret ym485.PackingBasic
  264. if err = ret.DeCode(recvbuf); err == nil {
  265. logrus.Infof("灯控使能状态:%d,是否上电开灯:%d", ret.Enabled, ret.OnOff)
  266. }
  267. }
  268. // QueryDeviceState 查询设备状态
  269. func (o *YmLampController) QueryDeviceState() {
  270. var obj ym485.DataPack //串口协议
  271. buf, err := obj.EnCode(o.devinfo.DevCode, 0x1E, []byte{0x00, 0x00, 0x60, 0x13, 0x01})
  272. defer ym485.ReleaseByteBuffer(buf)
  273. if err != nil {
  274. return
  275. }
  276. recvbuf, err := o.SendRecvData(buf.Bytes(), 1) //根据地址得到数据
  277. if err != nil {
  278. logrus.Error("单灯状态数据查询错误")
  279. return
  280. }
  281. var ret ym485.QueryDeviceStateAck
  282. if err = ret.DeCode(recvbuf); err == nil { //解码串口返回的数据
  283. // if ret.Dp.Addr != o.devinfo.DevCode { //不是本灯数据
  284. if !strings.Contains(o.devinfo.DevCode, ret.Dp.Addr) { //不是本灯数据
  285. o.State = nil
  286. return
  287. }
  288. if len(ret.State) == 0 {
  289. o.State = nil
  290. return
  291. }
  292. if len(o.State) != len(ret.State) {
  293. o.State = make([]ym485.DeviceState, len(ret.State))
  294. }
  295. copy(o.State, ret.State)
  296. //状态改变记录日志
  297. if oldstate != o.State[0].State {
  298. oldstate = o.State[0].State
  299. logrus.Infof("lamp state:%d", oldstate)
  300. }
  301. }
  302. mapData := make(map[string]*protocol.CHZB_LampData)
  303. var data protocol.CHZB_LampData
  304. data.Data = make(map[uint16]float64)
  305. data.SetStateErrorData(err)
  306. if len(o.State) > 0 { //只要第一路
  307. if o.State[0].State == 1 { //开灯
  308. data.Data[1] = float64(o.State[0].BrightnessState)
  309. } else {
  310. data.Data[1] = 0.0
  311. }
  312. data.Data[2] = float64(0)
  313. data.Data[3] = o.State[0].Voltage
  314. data.Data[4] = 0.0
  315. data.Data[5] = 0.0
  316. data.Data[6] = o.State[0].Current
  317. data.Data[7] = 0.0
  318. data.Data[8] = o.State[0].Degree //by hxz 1,2,3,6,8为有效
  319. data.Data[9] = 0.0
  320. }
  321. mapData[o.devinfo.DevCode] = &data
  322. var ret1 protocol.Pack_CHZB_UploadData
  323. if str, err := ret1.EnCode(o.devinfo.DevCode, appConfig.GID, GetNextUint64(), o.devinfo.TID, mapData); err == nil {
  324. GetMQTTMgr().Publish(GetTopic(protocol.DT_LAMPCONTROLLER, o.devinfo.DevCode, protocol.TP_YM_DATA), str, mqtt.AtMostOnce, ToCloud)
  325. }
  326. }
  327. func (o *YmLampController) TurnOnOff(flag uint8) error {
  328. kenabled := uint8(0x00)
  329. if flag > 0 {
  330. kenabled = 0xff
  331. }
  332. var obj ym485.PackingTurnOnOff
  333. buf, err := obj.EnCode(o.devinfo.DevCode, [4]byte{0x00, 0x00, 0x60, 0x07}, 0xff, kenabled)
  334. defer ym485.ReleaseByteBuffer(buf)
  335. if err != nil {
  336. return err
  337. }
  338. recvbuf, err := o.SendRecvData(buf.Bytes(), 3)
  339. if err != nil {
  340. return err
  341. }
  342. var ret ym485.PackingResult
  343. err = ret.DeCode(recvbuf)
  344. if err == nil {
  345. if ret.Result == 89 { //字母'Y'
  346. return nil
  347. }
  348. }
  349. return errors.New(protocol.FAILED_STR)
  350. }
  351. func (o *YmLampController) SetBrightness(brightness uint8) error {
  352. var obj ym485.PackingBrightnessColor
  353. buf, err := obj.EnCode(o.devinfo.DevCode, [4]byte{0x00, 0x00, 0x60, 0x09}, 0xff, []byte{brightness})
  354. defer ym485.ReleaseByteBuffer(buf)
  355. if err != nil {
  356. return err
  357. }
  358. recvbuf, err := o.SendRecvData(buf.Bytes(), 1)
  359. if err != nil {
  360. return err
  361. }
  362. var ret ym485.PackingResult
  363. err = ret.DeCode(recvbuf)
  364. if err == nil {
  365. if ret.Result == 89 { //字母'Y'
  366. return nil
  367. }
  368. }
  369. return errors.New(protocol.FAILED_STR)
  370. }
  371. func (o *YmLampController) SetBasicsettings(enabled, on uint8) {
  372. var obj ym485.PackingBasic
  373. buf, err := obj.EnCode(o.devinfo.DevCode, 0x1E, [4]byte{0x00, 0x00, 0x60, 0x01}, enabled, on, nil)
  374. defer ym485.ReleaseByteBuffer(buf)
  375. if err != nil {
  376. return
  377. }
  378. recvbuf, err := o.SendRecvData(buf.Bytes(), 1)
  379. if err != nil {
  380. return
  381. }
  382. var ret ym485.PackingResult
  383. if err = ret.DeCode(recvbuf); err == nil {
  384. if ret.Result == 89 { //字母'Y'
  385. logrus.Infoln("设置灯控基本信息执行成功")
  386. } else {
  387. logrus.Errorln("设置灯控基本信息执行失败")
  388. }
  389. }
  390. }
  391. func (o *YmLampController) Switch(Switch, Brightness uint8) error {
  392. if Switch == 0 {
  393. return o.TurnOnOff(0)
  394. } else {
  395. err := o.SetBrightness(Brightness)
  396. if err != nil {
  397. logrus.Errorf("调节亮度失败:%s", err.Error())
  398. }
  399. return o.TurnOnOff(Switch)
  400. }
  401. }
  402. func (o *YmLampController) ConfirmState(t time.Time) {
  403. //刚开/关灯,则不判断,避免采集数据误差导致误判
  404. if !o.tswitch.IsZero() {
  405. if t.After(o.tswitch) && t.Sub(o.tswitch).Seconds() < float64(o.devinfo.SendCloud/1000.0) {
  406. return
  407. }
  408. if t.Before(o.tswitch) && o.tswitch.Sub(t).Seconds() < float64(o.devinfo.SendCloud/1000.0) {
  409. return
  410. }
  411. }
  412. if len(o.State) == 0 {
  413. return
  414. }
  415. s := o.State[0]
  416. if o.mapTempLampsOOT != nil { //优先手动控制
  417. if t.After(time.Time(o.mapTempLampsOOT.End)) { //过时,则清除
  418. if err := redisEdgeData.Del(LampSwitchPrefix + o.devinfo.DevCode).Err(); err == nil {
  419. o.mapTempLampsOOT = nil
  420. o.tswitch = time.Time{}
  421. }
  422. } else if o.mapTempLampsOOT.isInTimeRange(t) {
  423. if o.mapTempLampsOOT.Brightness > 0 && s.State == 0 { //应开未开
  424. o.Switch(1, o.mapTempLampsOOT.Brightness) //开灯
  425. } else if o.mapTempLampsOOT.Brightness == 0 && s.State == 1 { //应关闭未关
  426. o.Switch(0, o.mapTempLampsOOT.Brightness) //关灯
  427. }
  428. return
  429. }
  430. }
  431. isInRange := false
  432. for _, v := range o.mapLamps2OOT { //其次时间策略
  433. if v.InTimeRange(t) { //在开灯区间
  434. if v.Brightness > 0 && s.State == 0 { //应开未开
  435. o.Switch(1, v.Brightness) //开灯
  436. } else if v.Brightness == 0 && s.State == 1 { //应关闭未关
  437. o.Switch(0, v.Brightness) //关灯
  438. }
  439. isInRange = true
  440. break
  441. }
  442. }
  443. if isInRange {
  444. return
  445. }
  446. if s.State == 1 {
  447. o.Switch(0, 0)
  448. }
  449. }
  450. func (o *YmLampController) HandleTpYmSetSwitch(m mqtt.Message) {
  451. var obj protocol.Pack_CHZB_Switch
  452. if err := obj.DeCode(m.PayloadString()); err != nil {
  453. logrus.Errorf("协议解析错误:%s,协议主题:%s,协议内容:%s", err.Error(), m.Topic(), m.PayloadString())
  454. return
  455. }
  456. if obj.Id != o.devinfo.DevCode {
  457. return
  458. }
  459. brightness := uint8(0)
  460. if obj.Data.Switch == 1 {
  461. brightness = obj.Data.Brightness
  462. }
  463. err := o.Switch(obj.Data.Switch, obj.Data.Brightness)
  464. if err == nil {
  465. o.tswitch = util.MlNow()
  466. mapRedisTempLampsOOT := make(map[string]interface{}) //临时开关灯记录,用于排除异常亮灯正常亮灯的情况
  467. ltr := LampTimeRange{
  468. Start: util.MLTime(o.tswitch),
  469. End: util.MLTime(o.tswitch.Add(time.Duration(obj.Data.Recovery) * time.Second)), //延迟2分钟,以防指令在队列中未及时执行
  470. Brightness: brightness,
  471. }
  472. ltrstr, _ := json.MarshalToString(ltr)
  473. o.mapTempLampsOOT = &ltr //内存
  474. mapRedisTempLampsOOT[o.devinfo.DevCode] = ltrstr //redis
  475. if err := redisEdgeData.HSet(LampSwitchPrefix+o.devinfo.DevCode, mapRedisTempLampsOOT).Err(); err != nil {
  476. logrus.Errorf("手动开关灯时间设置[内容:%v]缓存到redis失败:%s", mapRedisTempLampsOOT, err.Error())
  477. }
  478. }
  479. var ret protocol.Pack_Ack
  480. if str, err := ret.EnCode(o.devinfo.DevCode, appConfig.GID, obj.Seq, err); err == nil {
  481. GetMQTTMgr().Publish(GetTopic(protocol.DT_LAMPCONTROLLER, o.devinfo.DevCode, protocol.TP_YM_SET_SWITCH_ACK), str, mqtt.AtMostOnce, ToAll)
  482. }
  483. }
  484. func (o *YmLampController) HandleTpYmSetOnofftime(m mqtt.Message) {
  485. var obj protocol.Pack_SetOnOffTime
  486. if err := obj.DeCode(m.PayloadString()); err != nil {
  487. logrus.Errorf("协议解析错误:%s,协议主题:%s,协议内容:%s", err.Error(), m.Topic(), m.PayloadString())
  488. return
  489. }
  490. if obj.Id != o.devinfo.DevCode {
  491. return
  492. }
  493. if len(obj.Data.OnOffTime) == 0 {
  494. logrus.Errorf("Handle_TP_YM_SET_ONOFFTIME:错误,灯控编号[%v],时间段个数:%v", obj.Id, obj.Data.OnOffTime)
  495. return
  496. }
  497. mapRedisOOT := make(map[string]interface{})
  498. datastr, _ := json.MarshalToString(obj.Data.OnOffTime)
  499. mapRedisOOT[obj.Id] = datastr //缓存到redis
  500. o.mapLamps2OOT = obj.Data.OnOffTime //缓存在内存中
  501. //持久缓存到redis,以便于重启后读取进内存中
  502. err := redisEdgeData.HSet(LampOotPrefix+o.devinfo.DevCode, mapRedisOOT).Err()
  503. if err != nil {
  504. logrus.Errorf("灯控时间设置[内容:%v]缓存到redis失败:%s", mapRedisOOT, err.Error())
  505. }
  506. var ret protocol.Pack_Ack
  507. if str, err := ret.EnCode(o.devinfo.DevCode, appConfig.GID, obj.Seq, err); err == nil {
  508. GetMQTTMgr().Publish(GetTopic(protocol.DT_LAMPCONTROLLER, o.devinfo.DevCode, protocol.TP_YM_SET_ONOFFTIME_ACK), str, mqtt.AtMostOnce, ToCloud)
  509. }
  510. }
  511. func (o *YmLampController) ReloadOOTFromRedis() error {
  512. mapdata, err := redisEdgeData.HGetAll(LampOotPrefix + o.devinfo.DevCode).Result()
  513. if err != nil {
  514. if err == redis.Nil {
  515. return nil
  516. }
  517. logrus.Errorf("YmLampController.ReloadOOTFromRedis设备[%s]从redis加载时间策略失败:%s", o.devinfo.DevCode, err.Error())
  518. return err
  519. }
  520. for k, v := range mapdata {
  521. if k == o.devinfo.DevCode {
  522. var oot []protocol.CHZB_OnOffTime
  523. if err := json.UnmarshalFromString(v, &oot); err == nil {
  524. o.mapLamps2OOT = oot
  525. }
  526. break
  527. }
  528. }
  529. return nil
  530. }
  531. func (o *YmLampController) ReloadSwitchOOTFromRedis() error {
  532. mapdata, err := redisEdgeData.HGetAll(LampSwitchPrefix + o.devinfo.DevCode).Result()
  533. if err != nil {
  534. if err == redis.Nil {
  535. return nil
  536. }
  537. logrus.Errorf("YmLampController.ReloadSwitchOOTFromRedis设备[%s]从redis加载时间策略失败:%s", o.devinfo.DevCode, err.Error())
  538. return err
  539. }
  540. for k, v := range mapdata {
  541. if k == o.devinfo.DevCode {
  542. var ltr LampTimeRange
  543. if err := json.UnmarshalFromString(v, &ltr); err == nil {
  544. o.mapTempLampsOOT = &ltr
  545. }
  546. break
  547. }
  548. }
  549. return nil
  550. }
  551. func (o *YmLampController) ReloadLampAlarmFromRedis() error {
  552. mapAlarm, err := redisEdgeData.HGetAll(LampAlarmPrefix + o.devinfo.DevCode).Result()
  553. if err != nil {
  554. if err == redis.Nil {
  555. return nil
  556. }
  557. logrus.Errorf("YmLampController.ReloadLampAlarmFromRedis设备[%s]从redis加载广播恢复截止时间失败:%s", o.devinfo.DevCode, err.Error())
  558. return err
  559. }
  560. for k, v := range mapAlarm {
  561. if k == o.devinfo.DevCode {
  562. var lai LampAlarmInfo
  563. if err := json.UnmarshalFromString(v, &lai); err == nil {
  564. o.mapLampAlarm = &lai
  565. }
  566. break
  567. }
  568. }
  569. return nil
  570. }