| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229 |
- // package warbot -- бот для игры в wartank
- package warbot
- import (
- "context"
- "fmt"
- "log"
- "strings"
- "time"
- . "wartank/app/lev0/alias"
- . "wartank/app/lev0/types"
- "wartank/app/lev2/serv_bots/warbot/angar"
- "wartank/app/lev2/serv_bots/warbot/bot_stat/tank_stat"
- "wartank/app/lev2/serv_bots/warbot/warbot_config"
- "wartank/app/lev2/serv_bots/warbot/warbot_net"
- "wartank/pkg/kernel/kernel_dict"
- "wartank/pkg/kernel/kernel_log"
- . "wartank/pkg/kernel/kernel_types"
- "wartank/pkg/kernel/safe_bool"
- )
- // ВарБот -- бот для игры в вартанк
- type ВарБот struct {
- сервер ИПриложение
- store IKernelStore
- стата ИТанкСтат
- errFinal error // Финальная ошибка работы, если была
- ангар ИАнгар
- сеть ИБотСеть
- еслиРаботает *safe_bool.БезопБул
- еслиАвтозапуск *safe_bool.БезопБул
- конфиг *warbot_config.ВарБотКонфиг // Конфиг бота для хранения в базе
- кнт context.Context // Контекст бота
- фтОтмена func() // Функция отменя контекста бота
- лог ИЯдроЛог
- конт ИБотКонтекст
- }
- // ЗагрузитьВарБот -- загружает бота из базы
- func ЗагрузитьВарБот(конт ИЯдроКонтекст, сервер ИПриложение, номер АБотНомер) *ВарБот {
- лог := kernel_log.НовЛоггер("ВарБот")
- лог.Инфо("ЗагрузитьВарБот()\n")
- лог.Паника(сервер == nil, "ЗагрузитьВарБот(): ИСервер ==nil")
- лог.Паника(номер == 0, "ЗагрузитьВарБот(): номер пустой")
- стрНомер := fmt.Sprint(номер)
- log.Printf("ЗагрузитьВарБот(): номер=%q\n", стрНомер)
- store := сервер.Получ("kernStore").(IKernelStore)
- binData, err := store.Get("/bots/" + стрНомер)
- if err != nil {
- if !strings.Contains(err.Error(), "not found") {
- лог.Паника(true, "ЗагрузитьВарБот(): in load bot '%v' from store, err=\n\t%v\n", номер, err)
- }
- лог.Паника(err != nil, "ЗагрузитьВарБот(): in load bot '%v' from store, err=\n\t%v\n", номер, err)
- }
- конфиг := &warbot_config.ВарБотКонфиг{}
- конфиг.Unmarshal(binData)
- сам := создатьЯдроВарБот(сервер, конфиг)
- go сам.рестарт()
- _ = ИБот(сам)
- return сам
- }
- // Каждые два часа перезапускает себя
- func (сам *ВарБот) рестарт() {
- time.Sleep(time.Hour * 2)
- сам.сервер.Отменить()
- log.Printf("ВарБот.рестарт(): бот %q перезагружен\n", сам.конфиг.Логин_)
- }
- // НовВарБот -- возвращает новый WarBot
- func НовВарБот(сервер ИПриложение, номер АБотНомер, логин, пароль string, еслиАвто bool) *ВарБот {
- лог := kernel_log.НовЛоггер("ВарБот")
- лог.Инфо("НовВарБот()\n")
- лог.Паника(сервер == nil, "НовВарБот(): ИСервер == nil")
- лог.Паника(логин == "", "НовВарБот(): логин пустой")
- лог.Паника(пароль == "", "НовВарБот(): пароль пустой")
- log.Printf("НовВарБот(): name=%q\n", логин)
- config := &warbot_config.ВарБотКонфиг{
- ЕслиАвтозапуск_: еслиАвто,
- Логин_: логин,
- Пароль_: пароль,
- Номер_: номер,
- }
- сам := создатьЯдроВарБот(сервер, config)
- сам.сохр()
- _ = ИБот(сам)
- return сам
- }
- // Создаёт ядро бота
- func создатьЯдроВарБот(серв ИПриложение, конфиг *warbot_config.ВарБотКонфиг) *ВарБот {
- лог := kernel_log.НовЛоггер("ВарБот")
- лог.Инфо("создатьЯдроВарБот()\n")
- лог.Паника(серв == nil, "создатьЯдроВарБот(): ИСервер == nil")
- лог.Паника(конфиг == nil, "создатьЯдроВарБот(): ВарБотКонфиг==nil")
- ctx, fnCancel := context.WithCancel(серв.Конт())
- конт := kernel_dict.НовЯдроСловарь()
- сам := &ВарБот{
- конт: конт,
- сервер: серв,
- store: серв.Получ("kernStore").(IKernelStore),
- стата: tank_stat.НовТанкСтат(конт),
- еслиРаботает: safe_bool.НовБезопБул(),
- еслиАвтозапуск: safe_bool.НовБезопБул(),
- конфиг: конфиг,
- кнт: ctx,
- фтОтмена: fnCancel,
- лог: лог,
- }
- сам.конт.Уст("бот", сам)
- var err error
- // WarBotNet
- сам.сеть = warbot_net.НовВарБотСеть(конт, сам)
- сам.ангар, err = angar.НовАнгар(конт, сам)
- сам.лог.Паника(err != nil, "НовВарБот(): in make IAngar, err=\n\t%w", err)
- if сам.конфиг.ЕслиАвтозапуск_ {
- сам.еслиАвтозапуск.Уст()
- сам.Пуск()
- }
- return сам
- }
- // Сервер -- возвращает ссылку на объект сервера
- func (сам *ВарБот) Сервер() ИПриложение {
- return сам.сервер
- }
- // ЕслиПуск -- возвращает признак, что бот подключен
- func (сам *ВарБот) ЕслиПуск() bool {
- return сам.еслиРаботает.Получ()
- }
- // Номер -- возвращает номер бота
- func (сам *ВарБот) Номер() АБотНомер {
- return сам.конфиг.Номер()
- }
- // Имя -- возвращает имя бота
- func (сам *ВарБот) Имя() string {
- return сам.конфиг.Логин()
- }
- // Пароль -- возвращает пароль бота
- func (сам *ВарБот) Пароль() string {
- return сам.конфиг.Пароль_
- }
- // Пуск -- запускает бот в работу
- func (сам *ВарБот) Пуск() {
- if сам.еслиРаботает.Получ() {
- return
- }
- сам.ангар.Пуск()
- сам.еслиРаботает.Уст()
- }
- // Error -- возвращает финальную ошибку работы, если была
- func (сам *ВарБот) Error() error {
- return сам.errFinal
- }
- // Ангар -- возвращает ангар игры
- func (сам *ВарБот) Ангар() ИАнгар {
- return сам.ангар
- }
- // Стата -- возвращает статистику танка
- func (сам *ВарБот) Стата() ИТанкСтат {
- return сам.стата
- }
- // Сеть -- возвращает ссылку на свой сетевой клиент
- func (сам *ВарБот) Сеть() ИБотСеть {
- return сам.сеть
- }
- // АвтоИграЕсли -- возвращает признак автоматического запуска бота
- func (сам *ВарБот) АвтоИграЕсли() bool {
- return сам.еслиАвтозапуск.Получ()
- }
- // АвтоИграУст -- устанавливает признак автоматического запуска бота
- func (сам *ВарБот) АвтоИграУст() {
- log.Printf("WarBot.SetAutoGame()")
- сам.еслиАвтозапуск.Уст()
- сам.конфиг.ЕслиАвтозапуск_ = true
- сам.сохрКонфиг()
- }
- // АвтоИграСброс -- сбрасывает признак автоматического запуска бота
- func (сам *ВарБот) АвтоИграСброс() {
- log.Printf("WarBot.ResetAutoGame()")
- сам.еслиАвтозапуск.Уст()
- сам.конфиг.ЕслиАвтозапуск_ = false
- сам.сохрКонфиг()
- }
- // Сохраняет конфиг бота
- func (сам *ВарБот) сохрКонфиг() {
- log.Printf("ВарБот.сохрКонфиг()")
- strConf := сам.конфиг.Marshall()
- err := сам.store.Set("/bots/"+сам.Имя(), strConf)
- if err != nil {
- log.Printf("ВарБот.сохрКонфиг(): err=\n\t%v\n", err)
- }
- }
- // Контекст -- возвращает контекст бота
- func (сам *ВарБот) Контекст() context.Context {
- return сам.кнт
- }
- // Закончить -- отменяет контекст бота
- func (сам *ВарБот) Закончить() {
- сам.фтОтмена()
- }
- // Сохраняет себя в базу
- func (сам *ВарБот) сохр() {
- стрНомер := fmt.Sprint(сам.Номер())
- err := сам.store.Set("/bots/"+стрНомер, сам.конфиг.Marshall())
- сам.лог.Паника(err != nil, "ВарБот.сохр(): err=\n\t%v\n", err)
- }
- // КонтБот -- возвращает контекст бота
- func (сам *ВарБот) КонтБот() ИБотКонтекст {
- return сам.конт
- }
|