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. for {
  59. select {
  60. case <-сам.кнт.Done(): // Отмена контекста бота
  61. // log._rintf("CountTime.makeTick(): отмена контекста бота\n")
  62. return
  63. default:
  64. if !сам.еслиРаботает.Получ() {
  65. сам.фнОтмена()
  66. return
  67. }
  68. if спатьСчёт >= 10 {
  69. сам.канТик <- 1
  70. спатьСчёт = 0
  71. }
  72. time.Sleep(спатьИнтервал)
  73. спатьСчёт++
  74. }
  75. }
  76. }
  77. // Главный цикл обратного отсчёта
  78. func (sf *ВремОбрат) пуск() {
  79. for range sf.канТик {
  80. time.Sleep(time.Millisecond * 100)
  81. timeNow := time.Now().UTC().Unix()
  82. if sf.времПорог.Get() > int(timeNow) {
  83. continue
  84. }
  85. close(sf.канВызов)
  86. sf.фнОтмена()
  87. return
  88. }
  89. }
  90. // Стоп -- останавливает работу cчётчика
  91. func (sf *ВремОбрат) Стоп() {
  92. sf.еслиРаботает.Сброс()
  93. }
  94. // Получ -- возвращает число оставшихся сек
  95. func (sf *ВремОбрат) Получ() int {
  96. return sf.времяЗнач.Get()
  97. }
  98. // устанавливает число оставшихся сек
  99. func (sf *ВремОбрат) parse(val string) error {
  100. sf.блок.Lock()
  101. defer sf.блок.Unlock()
  102. if val == "" {
  103. return fmt.Errorf("CountTime.parse(): val is empty")
  104. }
  105. sf.парсер.Parse(val)
  106. _val := sf.парсер.Hour().Get()*3600 + sf.парсер.Min().Get()*60 + sf.парсер.Min().Get()
  107. sf.времяЗнач.Уст(_val)
  108. _val = int(time.Now().UTC().Unix()) + sf.времяЗнач.Get()
  109. sf.времПорог.Уст(_val)
  110. return nil
  111. }
  112. // устанавливает число оставшихся сек
  113. func (sf *ВремОбрат) set_val(val int) error {
  114. sf.блок.Lock()
  115. defer sf.блок.Unlock()
  116. if val < 0 {
  117. return fmt.Errorf("CountTime.set_val(): val(%v)<0", val)
  118. }
  119. sf.времяЗнач.Уст(val)
  120. { // Обновить локальные счётчики
  121. if val < 60 {
  122. sf.парсер.Hour().Reset()
  123. sf.парсер.Min().Reset()
  124. sf.парсер.Sec().Set(val)
  125. return nil
  126. }
  127. if 60 < val && val < 3600 {
  128. sf.парсер.Hour().Reset()
  129. iMin := val / 60
  130. sf.парсер.Min().Set(iMin)
  131. val -= iMin * 60
  132. sf.парсер.Sec().Set(val)
  133. return nil
  134. }
  135. sf.парсер.Hour().Set(val / 3600)
  136. val -= sf.парсер.Hour().Get() * 3600
  137. sf.парсер.Min().Set(val / 60)
  138. val -= sf.парсер.Min().Get() * 60
  139. sf.парсер.Sec().Set(val)
  140. // val = int(time.Now().UTC().Unix()) + sf.val.Get()
  141. }
  142. return nil
  143. }
  144. // Стр -- возвращает строковое представление оставшихся сек
  145. func (sf *ВремОбрат) Стр() string {
  146. sf.блок.RLock()
  147. defer sf.блок.RUnlock()
  148. timeNow := time.Now().UTC().Unix()
  149. val := sf.времПорог.Get() - int(timeNow)
  150. go sf.set_val(val)
  151. return sf.парсер.String()
  152. }
  153. // КаналСиг -- возвращает канал чтения тиков
  154. func (sf *ВремОбрат) КаналСиг() <-chan int {
  155. return sf.канВызов
  156. }
  157. func (сам *ВремОбрат) закрыть() {
  158. <-сам.кнт.Done()
  159. сам.блок.Lock()
  160. defer сам.блок.Unlock()
  161. if !сам.еслиРаботает.Получ() {
  162. return
  163. }
  164. сам.еслиРаботает.Сброс()
  165. close(сам.канТик)
  166. }