bot.go 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. // package bot -- бот для игры в wartank
  2. package bot
  3. import (
  4. "fmt"
  5. "log"
  6. "strings"
  7. "time"
  8. . "gitp78su.ipnodns.ru/svi/kern"
  9. . "gitp78su.ipnodns.ru/svi/kern/kc/helpers"
  10. . "gitp78su.ipnodns.ru/svi/kern/krn/ktypes"
  11. . "wartank/app/lev0/alias"
  12. "wartank/app/lev0/bfunc/bf_ammo_make"
  13. "wartank/app/lev0/bfunc/bf_ammo_stat"
  14. "wartank/app/lev0/bfunc/bf_fuel_attack"
  15. "wartank/app/lev0/bfunc/bf_fuel_find"
  16. "wartank/app/lev0/bfunc/bf_glory_make"
  17. "wartank/app/lev0/bfunc/bf_gold_find"
  18. "wartank/app/lev0/bfunc/bf_mission_simple"
  19. "wartank/app/lev0/bfunc/bf_polygon_activate"
  20. "wartank/app/lev0/bfunc/bf_silver_find"
  21. "wartank/app/lev0/bfunc/bf_silver_get"
  22. "wartank/app/lev0/bfunc/bf_silver_prod"
  23. "wartank/app/lev0/bfunc/bf_tank_stat"
  24. . "wartank/app/lev0/types"
  25. "wartank/app/lev2"
  26. "wartank/app/lev3/bot/bot_config"
  27. "wartank/app/lev3/bot/bot_net"
  28. "wartank/app/lev3/bot/bot_stat/tank_stat"
  29. )
  30. // ВарБот -- бот для игры в вартанк
  31. type Бот struct {
  32. прилож ИПриложение
  33. хран IKernelStoreKv
  34. стата ИТанкСтат
  35. errFinal error // Финальная ошибка работы, если была
  36. сеть ИБотСеть
  37. ангар ИАренаАнгар
  38. еслиРаботает ISafeBool
  39. еслиАвтозапуск ISafeBoolReact
  40. лог ILogBuf
  41. конт ILocalCtx
  42. конфиг *bot_config.БотКонфиг
  43. }
  44. // ЗагрузитьВарБот -- загружает бота из базы
  45. func ЗагрузитьВарБот(номер АБотНомер) *Бот {
  46. лог := NewLogBuf()
  47. лог.Info("ЗагрузитьВарБот()\n")
  48. Hassert(номер != 0, "ЗагрузитьВарБот(): номер пустой")
  49. стрНомер := fmt.Sprint(номер)
  50. лог.Info("ЗагрузитьВарБот(): номер=%q\n", стрНомер)
  51. конт := GetKernelCtx()
  52. хран := конт.Get("kernStoreKV").Val().(IKernelStoreKv)
  53. binData, err := хран.Get("/bots/" + стрНомер)
  54. if err != nil {
  55. if !strings.Contains(err.Error(), "not found") {
  56. Hassert(false, "ЗагрузитьВарБот(): in load bot '%v' from store, err=\n\t%v\n", номер, err)
  57. }
  58. Hassert(err == nil, "ЗагрузитьВарБот(): in load bot '%v' from store, err=\n\t%v\n", номер, err)
  59. }
  60. конфиг := &bot_config.БотКонфиг{}
  61. конфиг.Unmarshal(binData)
  62. сам := создатьЯдроВарБот(конфиг)
  63. go сам.рестарт()
  64. _ = ИБот(сам)
  65. return сам
  66. }
  67. // Каждые два часа перезапускает себя
  68. func (сам *Бот) рестарт() {
  69. time.Sleep(time.Hour * 2)
  70. сам.конт.Cancel()
  71. сам.лог.Info("рестарт(): бот %q перезагружен\n", сам.конфиг.Логин_)
  72. }
  73. // НовВарБот -- возвращает новый WarBot
  74. func НовВарБот(конт IKernelCtx, номер АБотНомер, логин, пароль string, еслиАвто bool) *Бот {
  75. лог := NewLogBuf()
  76. лог.Info("НовВарБот()\n")
  77. Hassert(логин != "", "НовВарБот(): логин пустой")
  78. Hassert(пароль != "", "НовВарБот(): пароль пустой")
  79. лог.Info("НовВарБот(): name=%q\n", логин)
  80. config := &bot_config.БотКонфиг{
  81. ЕслиАвтозапуск_: еслиАвто,
  82. Логин_: логин,
  83. Пароль_: пароль,
  84. Номер_: номер,
  85. }
  86. сам := создатьЯдроВарБот(config)
  87. сам.сохрКонфиг()
  88. _ = ИБот(сам)
  89. return сам
  90. }
  91. // Создаёт ядро бота
  92. func создатьЯдроВарБот(конфиг *bot_config.БотКонфиг) *Бот {
  93. Hassert(конфиг != nil, "создатьЯдроВарБот(): ВарБотКонфиг==nil")
  94. лог := NewLogBuf()
  95. лог.Info("создатьЯдроВарБот()\n")
  96. конт := GetKernelCtx()
  97. приложение := конт.Get("мод_сервер").Val().(ИПриложение)
  98. контБот := NewLocalCtx(конт.Ctx())
  99. сам := &Бот{
  100. конт: контБот,
  101. прилож: приложение,
  102. хран: конт.Get("kernStoreKV").Val().(IKernelStoreKv),
  103. стата: tank_stat.НовТанкСтат(контБот),
  104. еслиРаботает: NewSafeBool(),
  105. конфиг: конфиг,
  106. лог: лог,
  107. }
  108. сам.еслиАвтозапуск = NewSafeBoolReact()
  109. сам.еслиАвтозапуск.Add(сам.Имя(), сам.автозапускИзм)
  110. сам.конт.Set("бот", сам, "создание ядра")
  111. сам.конт.Set("приложение", приложение, "Приложение WarBot")
  112. сам.сеть = bot_net.НовБотСеть(сам.конт)
  113. if сам.конфиг.ЕслиАвтозапуск_ {
  114. сам.еслиАвтозапуск.Set()
  115. сам.Пуск()
  116. }
  117. сам.ангар = lev2.НовАнгар(сам.конт)
  118. _ = lev2.НовКонвой(сам.конт)
  119. _ = lev2.НовБанк(сам.конт)
  120. _ = lev2.НовМиссииПростые(сам.конт)
  121. _ = lev2.НовПолигон(сам.конт)
  122. _ = lev2.НовБаза(сам.конт)
  123. _ = lev2.НовШахта(сам.конт)
  124. _ = lev2.НовАрсенал(сам.конт)
  125. _ = lev2.НовАренаТопливо(сам.конт)
  126. return сам
  127. }
  128. // ЕслиРабота -- возвращает признак, что бот подключен
  129. func (сам *Бот) ЕслиРабота() bool {
  130. return сам.еслиРаботает.Get()
  131. }
  132. // Номер -- возвращает номер бота
  133. func (сам *Бот) Номер() АБотНомер {
  134. return сам.конфиг.Номер()
  135. }
  136. // Имя -- возвращает имя бота
  137. func (сам *Бот) Имя() string {
  138. return сам.конфиг.Логин()
  139. }
  140. // Пароль -- возвращает пароль бота
  141. func (сам *Бот) Пароль() string {
  142. return сам.конфиг.Пароль_
  143. }
  144. // Пуск -- запускает бот в работу
  145. func (сам *Бот) Пуск() {
  146. if сам.еслиРаботает.Get() {
  147. return
  148. }
  149. go сам.пуск()
  150. сам.еслиРаботает.Set()
  151. }
  152. // Работает в отдельном потоке ,пока сервер не даст команду остановки
  153. func (сам *Бот) пуск() {
  154. for {
  155. select {
  156. case <-сам.конт.Ctx().Done():
  157. return
  158. default:
  159. time.Sleep(time.Second * 5)
  160. сам.ангар.Обновить()
  161. bf_gold_find.ЗолотоНайти(сам.конт)
  162. bf_silver_find.СереброНайти(сам.конт)
  163. bf_fuel_find.ТопливоНайти(сам.конт)
  164. bf_fuel_attack.ТопливоАтаковать(сам.конт)
  165. bf_mission_simple.МиссииПростыеЗабрать(сам.конт)
  166. bf_silver_get.СереброЗабрать(сам.конт)
  167. bf_silver_prod.СереброПроизводить(сам.конт)
  168. bf_polygon_activate.ПолигонАктивировать(сам.конт)
  169. bf_tank_stat.ТанкСтатПолучить(сам.конт)
  170. bf_ammo_stat.СнарядыСтат(сам.конт)
  171. bf_ammo_make.СнарядыСделать(сам.конт)
  172. bf_glory_make.СлаваБой(сам.конт)
  173. }
  174. }
  175. }
  176. // Error -- возвращает финальную ошибку работы, если была
  177. func (сам *Бот) Error() error {
  178. return сам.errFinal
  179. }
  180. // Стата -- возвращает статистику танка
  181. func (сам *Бот) Стата() ИТанкСтат {
  182. return сам.стата
  183. }
  184. // Сеть -- возвращает ссылку на свой сетевой клиент
  185. func (сам *Бот) Сеть() ИБотСеть {
  186. return сам.сеть
  187. }
  188. // Автозапуск -- возвращает признак автоматического запуска бота
  189. func (сам *Бот) Автозапуск() ISafeBoolReact {
  190. return сам.еслиАвтозапуск
  191. }
  192. // Обратный вызов автоматического запуска бота
  193. func (сам *Бот) автозапускИзм(знач bool) {
  194. сам.лог.Debug("Бот.автозапускИзм()")
  195. // сам.еслиАвтозапуск.Уст()
  196. сам.конфиг.ЕслиАвтозапуск_ = знач
  197. сам.сохрКонфиг()
  198. }
  199. // Сохраняет конфиг бота
  200. func (сам *Бот) сохрКонфиг() {
  201. log.Printf("ВарБот.сохрКонфиг()")
  202. strConf := сам.конфиг.Marshall()
  203. стрНомер := fmt.Sprint(сам.Номер())
  204. err := сам.хран.Set("/bots/"+стрНомер, strConf)
  205. Hassert(err == nil, "ВарБот.сохрКонфиг(): err=\n\t%v\n", err)
  206. }
  207. // КонтБот -- возвращает контекст бота
  208. func (сам *Бот) КонтБот() ILocalCtx {
  209. return сам.конт
  210. }