cron.go 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. package service
  2. import (
  3. "server/logger"
  4. "server/modbus"
  5. "server/model"
  6. "server/utils"
  7. "sort"
  8. "time"
  9. )
  10. type Cron struct {
  11. }
  12. // RelayOnOffTimeTask 定时开关 前提是 设备在线 否则失效
  13. func (c Cron) RelayOnOffTimeTask() {
  14. current := time.Now().Format("15:04")
  15. devices, err := utils.GetOnlineDevices()
  16. if err != nil {
  17. logger.Get().Errorf("LoadData err = %s", err.Error())
  18. return
  19. }
  20. relayMap := make(map[string]map[int]int)
  21. for _, d := range devices {
  22. //if d.State == 0 {
  23. // continue
  24. //}
  25. tmp, isExist := relayMap[d.Sn]
  26. if !isExist { // 回路id 回路状态
  27. tmp = make(map[int]int)
  28. }
  29. relays := d.DeviceLoops
  30. rcTime, rlTime, err := utils.SunriseSunsetForChina(28.23, 113.05)
  31. if err != nil {
  32. logger.Get().Errorf("SunriseSunsetForChina err = %s", err.Error())
  33. rcTime = "06:00"
  34. rlTime = "18:00"
  35. }
  36. for _, r := range relays {
  37. //日出日落时间
  38. if r.TimeCondition1OnTime == "日出" {
  39. r.TimeCondition1OnTime = rcTime
  40. } else if r.TimeCondition1OnTime == "日落" {
  41. r.TimeCondition1OnTime = rlTime
  42. }
  43. if r.TimeCondition1OffTime == "日出" {
  44. r.TimeCondition1OffTime = rcTime
  45. } else if r.TimeCondition1OffTime == "日落" {
  46. r.TimeCondition1OffTime = rlTime
  47. }
  48. if r.TimeCondition2OnTime == "日出" {
  49. r.TimeCondition2OnTime = rcTime
  50. } else if r.TimeCondition2OnTime == "日落" {
  51. r.TimeCondition2OnTime = rlTime
  52. }
  53. if r.TimeCondition2OffTime == "日出" {
  54. r.TimeCondition2OffTime = rcTime
  55. } else if r.TimeCondition2OffTime == "日落" {
  56. r.TimeCondition2OffTime = rlTime
  57. }
  58. isTiming := 0 // 用来判断该设备的回路 是否定时
  59. //控制回路 时间判断 是否关闭 当前时间是否等与规定时间 规定时间到规定时间+10分钟
  60. //因为1分钟执行一次,所以会多次执行,则新加判断 如果数据库中状态和回路状态相同则不执行下去
  61. layout := "15:04"
  62. rTimeCondition1OnTime, _ := time.Parse(layout, r.TimeCondition1OnTime)
  63. rTimeCondition1OffTime, _ := time.Parse(layout, r.TimeCondition1OffTime)
  64. rTimeCondition2OnTime, _ := time.Parse(layout, r.TimeCondition2OnTime)
  65. rTimeCondition2OffTime, _ := time.Parse(layout, r.TimeCondition2OffTime)
  66. rCurrent, _ := time.Parse(layout, current)
  67. if r.TimeCondition1OnTime != "关闭" &&
  68. (r.TimeCondition1OnTime == current ||
  69. (rCurrent.Before(rTimeCondition1OnTime.Add(11*time.Minute)) && rCurrent.After(rTimeCondition1OnTime))) {
  70. tmp[r.ID] = 1
  71. isTiming = 1
  72. }
  73. if r.TimeCondition1OffTime != "关闭" && (r.TimeCondition1OffTime == current ||
  74. (rCurrent.Before(rTimeCondition1OffTime.Add(11*time.Minute)) && rCurrent.After(rTimeCondition1OffTime))) {
  75. tmp[r.ID] = 0
  76. isTiming = 1
  77. }
  78. if r.TimeCondition2OnTime != "关闭" && (r.TimeCondition2OnTime == current ||
  79. (rCurrent.Before(rTimeCondition2OnTime.Add(11*time.Minute)) && rCurrent.After(rTimeCondition2OnTime))) {
  80. tmp[r.ID] = 1
  81. isTiming = 1
  82. }
  83. if r.TimeCondition2OffTime != "关闭" && (r.TimeCondition2OffTime == current ||
  84. (rCurrent.Before(rTimeCondition2OffTime.Add(11*time.Minute)) && rCurrent.After(rTimeCondition2OffTime))) {
  85. tmp[r.ID] = 0
  86. isTiming = 1
  87. }
  88. if isTiming == 1 {
  89. relayMap[d.Sn] = tmp
  90. }
  91. }
  92. }
  93. for key, value := range relayMap {
  94. reg, dev, err := utils.GetDataByDeviceId(key)
  95. if err != nil {
  96. logger.Get().Errorf("GetDataByDeviceId err = %s\n", err.Error())
  97. }
  98. for i, i2 := range value {
  99. data := modbus.DeviceLoopSwitch(i, i2)
  100. if model.ConnectionMap[key] == nil {
  101. logger.Get().Errorf("设备连接丢失")
  102. continue
  103. }
  104. err := utils.WriteDevice(dev.Sn, data, model.ConnectionMap[key])
  105. time.Sleep(100 * time.Millisecond)
  106. if err != nil {
  107. logger.Get().Errorf("WriteDevice err = %s\n", err.Error())
  108. }
  109. for j, loop := range dev.DeviceLoops {
  110. if loop.ID == i {
  111. dev.DeviceLoops[j].State = i2
  112. }
  113. }
  114. }
  115. for i, device := range reg.Devices {
  116. if device.Sn == dev.Sn {
  117. reg.Devices[i] = dev
  118. }
  119. }
  120. data, err := utils.SaveRegionOnData(reg)
  121. if err != nil {
  122. logger.Get().Errorf("SaveRegionOnData err = %s\n", err.Error())
  123. }
  124. err = SaveData(data)
  125. if err != nil {
  126. logger.Get().Errorf("SaveData err = %s\n", err.Error())
  127. }
  128. }
  129. }
  130. func (c Cron) RelayOnOffTimeTaskSn(sn string) {
  131. _, dev, err := utils.GetDataByDeviceId(sn)
  132. if err != nil {
  133. logger.Get().Errorf("LoadData err = %s", err.Error())
  134. return
  135. }
  136. relays := dev.DeviceLoops
  137. rcTime, rlTime, err := utils.SunriseSunsetForChina(28.23, 113.05)
  138. tmp := make(map[int]int)
  139. for _, relay := range relays {
  140. //日出日落时间
  141. if relay.TimeCondition1OnTime == "日出" {
  142. relay.TimeCondition1OnTime = rcTime
  143. } else if relay.TimeCondition1OnTime == "日落" {
  144. relay.TimeCondition1OnTime = rlTime
  145. }
  146. if relay.TimeCondition1OffTime == "日出" {
  147. relay.TimeCondition1OffTime = rcTime
  148. } else if relay.TimeCondition1OffTime == "日落" {
  149. relay.TimeCondition1OffTime = rlTime
  150. }
  151. if relay.TimeCondition2OnTime == "日出" {
  152. relay.TimeCondition2OnTime = rcTime
  153. } else if relay.TimeCondition2OnTime == "日落" {
  154. relay.TimeCondition2OnTime = rlTime
  155. }
  156. if relay.TimeCondition2OffTime == "日出" {
  157. relay.TimeCondition2OffTime = rcTime
  158. } else if relay.TimeCondition2OffTime == "日落" {
  159. relay.TimeCondition2OffTime = rlTime
  160. }
  161. if relay.TimeCondition1OffTime == "关闭" && relay.TimeCondition1OnTime == "关闭" && relay.TimeCondition2OffTime == "关闭" && relay.TimeCondition2OnTime == "关闭" {
  162. continue
  163. }
  164. state, _ := ss(relay.TimeCondition1OnTime, relay.TimeCondition1OffTime, relay.TimeCondition2OnTime, relay.TimeCondition2OffTime)
  165. tmp[relay.ID] = state
  166. }
  167. for loop1, state := range tmp {
  168. reg, dev, err := utils.GetDataByDeviceId(sn)
  169. if err != nil {
  170. logger.Get().Errorf("GetDataByDeviceId err = %s\n", err.Error())
  171. }
  172. data := modbus.DeviceLoopSwitch(loop1, state)
  173. if model.ConnectionMap[sn] == nil {
  174. logger.Get().Errorf("设备连接丢失")
  175. continue
  176. }
  177. err = utils.WriteDevice(dev.Sn, data, model.ConnectionMap[sn])
  178. time.Sleep(100 * time.Millisecond)
  179. if err != nil {
  180. logger.Get().Errorf("WriteDevice err = %s\n", err.Error())
  181. }
  182. for j, loop := range dev.DeviceLoops {
  183. if loop.ID == loop1 {
  184. dev.DeviceLoops[j].State = state
  185. }
  186. }
  187. for i, device := range reg.Devices {
  188. if device.Sn == dev.Sn {
  189. reg.Devices[i] = dev
  190. }
  191. }
  192. data1, err := utils.SaveRegionOnData(reg)
  193. if err != nil {
  194. logger.Get().Errorf("SaveRegionOnData err = %s\n", err.Error())
  195. }
  196. err = SaveData(data1)
  197. if err != nil {
  198. logger.Get().Errorf("SaveData err = %s\n", err.Error())
  199. }
  200. }
  201. }
  202. const (
  203. Off = iota
  204. On
  205. )
  206. type TimeControl struct {
  207. start time.Time
  208. end time.Time
  209. state int
  210. }
  211. func getDeviceState(schedules []TimeControl, currentTime time.Time) int {
  212. for _, period := range schedules {
  213. now := time.Date(0, time.January, 1, currentTime.Hour(), currentTime.Minute(), currentTime.Second(), currentTime.Nanosecond(), time.UTC)
  214. if period.start.After(period.end) {
  215. if now.After(period.start) || now.Before(period.end) {
  216. return period.state
  217. }
  218. } else {
  219. if now.After(period.start) && now.Before(period.end) {
  220. return period.state
  221. }
  222. }
  223. }
  224. return Off
  225. }
  226. func stateTransition(str string) int {
  227. if str == "on1" || str == "on2" {
  228. return On
  229. } else {
  230. return Off
  231. }
  232. }
  233. func ss(on1, off1, on2, off2 string) (int, error) {
  234. deviceRecoveryTime := time.Now()
  235. // 定义时间映射
  236. deviceTime := map[string]string{
  237. "on1": on1,
  238. "off1": off1,
  239. "on2": on2,
  240. "off2": off2,
  241. }
  242. // 创建一个切片存储时间
  243. var times []struct {
  244. key string
  245. value time.Time
  246. }
  247. // 将时间字符串解析为 time.Time 对象,并存入切片
  248. for key, value := range deviceTime {
  249. if value == "关闭" {
  250. continue
  251. }
  252. t, err := time.Parse("15:04", value)
  253. if err != nil {
  254. logger.Get().Error("时间转换失败!" + err.Error())
  255. return 0, err
  256. }
  257. times = append(times, struct {
  258. key string
  259. value time.Time
  260. }{key, t})
  261. }
  262. // 对时间进行排序
  263. sort.Slice(times, func(i, j int) bool {
  264. return times[i].value.Before(times[j].value)
  265. })
  266. var timeControls []TimeControl
  267. for i, _ := range times {
  268. if len(times) == i+1 {
  269. timeControls = append(timeControls, TimeControl{start: times[i].value, end: times[0].value.Add(24 * time.Hour), state: stateTransition(times[i].key)})
  270. continue
  271. }
  272. timeControls = append(timeControls, TimeControl{start: times[i].value, end: times[i+1].value, state: stateTransition(times[i].key)})
  273. }
  274. //for _, control := range timeControls {
  275. // fmt.Printf("%s : %s : %d\n", control.start, control.end, control.state)
  276. //}
  277. // 在恢复时检查状态
  278. state := getDeviceState(timeControls, deviceRecoveryTime)
  279. return state, nil
  280. }