down_time.go 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. // package down_time -- счётчик обратного времени в мсек
  2. package down_time
  3. import (
  4. "context"
  5. "fmt"
  6. // "log"
  7. "sync"
  8. "time"
  9. "wartank/pkg/components/parsetime"
  10. "wartank/pkg/components/safebool"
  11. "wartank/pkg/components/safeint"
  12. "wartank/pkg/types"
  13. )
  14. const (
  15. спатьИнтервал = time.Millisecond * 100 // Малый интервал сна в 100 мсек
  16. )
  17. // ВремОбрат -- счётчик обратного времени для игровой зоны (анга, база, битва и т.п.)
  18. type ВремОбрат struct {
  19. сцена types.ИСцена
  20. парсер *parsetime.ParseTime // Парсер значения
  21. времяЗнач *safeint.SafeInt // Фактическое значение счётчика
  22. времПорог *safeint.SafeInt // Целевое время срабатывания
  23. еслиРаботает *safebool.SafeBool // Признак работы
  24. канТик chan int // Канал секундных интервалов сна (для отображения)
  25. канВызов chan int // Канал для отправки сигналов (для верхнего уровня)
  26. кнт context.Context // Контекст для счётчика времени
  27. фнОтмена func() // Функция отмены контекста для счётчика времени
  28. блок sync.RWMutex
  29. }
  30. // НовВремОбрат -- возвращает новый *CountTime
  31. func НовВремОбрат(сцена types.ИСцена, знач int) *ВремОбрат {
  32. if сцена == nil {
  33. panic("НовВремОбрат(): ИСцена == nil")
  34. }
  35. кнт, фнОтмена := context.WithCancel(сцена.Кнт())
  36. сам := &ВремОбрат{
  37. сцена: сцена,
  38. времяЗнач: safeint.NewSafeInt(),
  39. канТик: make(chan int, 5),
  40. канВызов: make(chan int, 2),
  41. еслиРаботает: safebool.NewSafeBool(),
  42. парсер: parsetime.NewParseTime(),
  43. времПорог: safeint.NewSafeInt(),
  44. кнт: кнт,
  45. фнОтмена: фнОтмена,
  46. }
  47. знач = int(time.Now().UTC().Unix()) + знач
  48. сам.времПорог.Уст(знач)
  49. сам.еслиРаботает.Уст()
  50. go сам.сделатьТик()
  51. go сам.пуск()
  52. go сам.закрыть()
  53. return сам
  54. }
  55. // Запускает тикер для секундных интервалов
  56. func (сам *ВремОбрат) сделатьТик() {
  57. спатьСчёт := 0 // Счётчик сна (10 периодов -- 1 секунда)
  58. defer close(сам.канТик)
  59. for {
  60. select {
  61. case <-сам.кнт.Done(): // Отмена контекста бота
  62. // log._rintf("CountTime.makeTick(): отмена контекста бота\n")
  63. return
  64. default:
  65. if !сам.еслиРаботает.Получ() {
  66. сам.фнОтмена()
  67. return
  68. }
  69. if спатьСчёт >= 10 {
  70. сам.канТик <- 1
  71. спатьСчёт = 0
  72. }
  73. time.Sleep(спатьИнтервал)
  74. спатьСчёт++
  75. }
  76. }
  77. }
  78. // Главный цикл обратного отсчёта
  79. func (sf *ВремОбрат) пуск() {
  80. for range sf.канТик {
  81. time.Sleep(time.Millisecond * 100)
  82. timeNow := time.Now().UTC().Unix()
  83. if sf.времПорог.Get() > int(timeNow) {
  84. continue
  85. }
  86. close(sf.канВызов)
  87. sf.фнОтмена()
  88. return
  89. }
  90. }
  91. // Стоп -- останавливает работу cчётчика
  92. func (sf *ВремОбрат) Стоп() {
  93. sf.еслиРаботает.Сброс()
  94. }
  95. // Получ -- возвращает число оставшихся сек
  96. func (sf *ВремОбрат) Получ() int {
  97. return sf.времяЗнач.Get()
  98. }
  99. // устанавливает число оставшихся сек
  100. func (sf *ВремОбрат) parse(val string) error {
  101. sf.блок.Lock()
  102. defer sf.блок.Unlock()
  103. if val == "" {
  104. return fmt.Errorf("CountTime.parse(): val is empty")
  105. }
  106. sf.парсер.Parse(val)
  107. _val := sf.парсер.Hour().Get()*3600 + sf.парсер.Min().Get()*60 + sf.парсер.Min().Get()
  108. sf.времяЗнач.Уст(_val)
  109. _val = int(time.Now().UTC().Unix()) + sf.времяЗнач.Get()
  110. sf.времПорог.Уст(_val)
  111. return nil
  112. }
  113. // устанавливает число оставшихся сек
  114. func (sf *ВремОбрат) set_val(val int) error {
  115. sf.блок.Lock()
  116. defer sf.блок.Unlock()
  117. if val < 0 {
  118. return fmt.Errorf("CountTime.set_val(): val(%v)<0", val)
  119. }
  120. sf.времяЗнач.Уст(val)
  121. { // Обновить локальные счётчики
  122. if val < 60 {
  123. sf.парсер.Hour().Reset()
  124. sf.парсер.Min().Reset()
  125. sf.парсер.Sec().Set(val)
  126. return nil
  127. }
  128. if 60 < val && val < 3600 {
  129. sf.парсер.Hour().Reset()
  130. iMin := val / 60
  131. sf.парсер.Min().Set(iMin)
  132. val -= iMin * 60
  133. sf.парсер.Sec().Set(val)
  134. return nil
  135. }
  136. sf.парсер.Hour().Set(val / 3600)
  137. val -= sf.парсер.Hour().Get() * 3600
  138. sf.парсер.Min().Set(val / 60)
  139. val -= sf.парсер.Min().Get() * 60
  140. sf.парсер.Sec().Set(val)
  141. // val = int(time.Now().UTC().Unix()) + sf.val.Get()
  142. }
  143. return nil
  144. }
  145. // Стр -- возвращает строковое представление оставшихся сек
  146. func (sf *ВремОбрат) Стр() string {
  147. sf.блок.RLock()
  148. defer sf.блок.RUnlock()
  149. timeNow := time.Now().UTC().Unix()
  150. val := sf.времПорог.Get() - int(timeNow)
  151. go sf.set_val(val)
  152. return sf.парсер.String()
  153. }
  154. // КаналСиг -- возвращает канал чтения тиков
  155. func (sf *ВремОбрат) КаналСиг() <-chan int {
  156. return sf.канВызов
  157. }
  158. func (сам *ВремОбрат) закрыть() {
  159. <-сам.кнт.Done()
  160. сам.блок.Lock()
  161. defer сам.блок.Unlock()
  162. if !сам.еслиРаботает.Получ() {
  163. return
  164. }
  165. сам.еслиРаботает.Сброс()
  166. }