arena_market.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  1. // package arena_market -- объект рынка
  2. package arena_market
  3. import (
  4. "log"
  5. "strconv"
  6. "strings"
  7. "time"
  8. . "gitp78su.ipnodns.ru/svi/kern/krn/ktypes"
  9. . "wartank/app/lev0/alias"
  10. . "wartank/app/lev0/types"
  11. "wartank/app/lev2/arena"
  12. )
  13. // АренаРынок -- объект рынка
  14. type АренаРынок struct {
  15. ИАрена
  16. конт ILocalCtx
  17. бот ИБот
  18. }
  19. // НовРынок -- возвращает новый рынок
  20. func НовРынок(конт ILocalCtx) ИАренаРынок {
  21. сам := &АренаРынок{
  22. бот: конт.Get("бот").(ИБот),
  23. }
  24. аренаКонфиг := arena.АренаКонфиг{
  25. Бот_: сам.бот,
  26. АренаИмя_: "Рынок",
  27. СтрКонтроль_: `<title>Рынок</title>`,
  28. ФнПуск_: сам.пуск,
  29. СтрУрл_: "https://wartank.ru/market",
  30. }
  31. сам.ИАрена = arena.НовАрена(конт, аренаКонфиг)
  32. return сам
  33. }
  34. // Пуск -- запускает всю работу рынка в отдельном потоке
  35. func (сам *АренаРынок) Пуск() {
  36. go сам.пуск()
  37. }
  38. // выполняет опрос рынка базы, должен работать как горутина
  39. func (сам *АренаРынок) пуск() {
  40. time.Sleep(time.Second * 7)
  41. фнРабота := func() {
  42. defer time.Sleep(time.Minute * 30)
  43. for !сам.уровеньОбновить() {
  44. }
  45. сам.ускорениеПровер()
  46. сам.проверОжидание()
  47. for сам.купитьЗолото() {
  48. }
  49. }
  50. for {
  51. select {
  52. case <-сам.бот.КонтБот().Ctx().Done():
  53. return
  54. default:
  55. фнРабота()
  56. }
  57. }
  58. }
  59. // Проверяет ускорение строительства
  60. func (сам *АренаРынок) ускорениеПровер() {
  61. списСтр := сам.Сеть().ВебВоркер().Получ("http://wartank.ru/buildings")
  62. // <span class="green2">Склад топлива -
  63. var (
  64. еслиНайти = false
  65. стр string
  66. )
  67. for _, стр = range списСтр {
  68. if strings.Contains(стр, `<span class="green2">Склад топлива - `) {
  69. еслиНайти = true
  70. break
  71. }
  72. }
  73. if !еслиНайти {
  74. return
  75. }
  76. }
  77. // Обновляет текущий уровень рынка (может быть не построен)
  78. func (сам *АренаРынок) уровеньОбновить() bool {
  79. списСтр := сам.Сеть().ВебВоркер().Получ("http://wartank.ru/buildings")
  80. // <span class="green2">Рынок -
  81. var (
  82. еслиНайти = false
  83. стр = ""
  84. )
  85. for _, стр = range списСтр {
  86. if strings.Contains(стр, `<span class="green2">Рынок -`) {
  87. еслиНайти = true
  88. break
  89. }
  90. }
  91. if !еслиНайти {
  92. return false
  93. }
  94. // <span class="green2">Рынок - 0</span><br/>
  95. _стр := strings.TrimPrefix(стр, `<span class="green2">Рынок - `)
  96. _стр = strings.TrimSuffix(_стр, `</span><br/>`)
  97. иУровень, ош := strconv.Atoi(_стр)
  98. if ош != nil {
  99. log.Printf("Рынок.уровеньОбновить(): строка уровня сбойная, стр=%q, ош=\n\t%v\n", стр, ош)
  100. return false
  101. }
  102. сам.Уровень().Уст(иУровень)
  103. switch иУровень {
  104. case 0: // рынок надо построить
  105. for !сам.построить() {
  106. }
  107. default: // Может можно проапгрейдить
  108. счёт := 5
  109. for !сам.проапгрейдить() {
  110. счёт--
  111. if счёт >= 0 {
  112. break
  113. }
  114. }
  115. }
  116. return true
  117. }
  118. // Строит шахту при нулевом уровне
  119. func (сам *АренаРынок) построить() bool {
  120. time.Sleep(time.Millisecond * 1000)
  121. // <td style="width:50%;padding-left:1px;"><a class="simple-but border mb5" href="building-upgrade/Market"><span><span>Построить</span></span></a></td>
  122. var (
  123. еслиНайти = false
  124. списСтр []string
  125. стр = ""
  126. )
  127. фнКупить := func() bool {
  128. defer time.Sleep(time.Millisecond * 1000)
  129. списСтр = сам.Сеть().ВебВоркер().Получ("https://wartank.ru/building-upgrade/Market")
  130. for _, стр = range списСтр {
  131. // <a class="simple-but border mb5" href="Market?19-1.ILinkListener-upgradeLink-link">
  132. if strings.Contains(стр, `ILinkListener-upgradeLink-link`) {
  133. еслиНайти = true
  134. break
  135. }
  136. }
  137. if !еслиНайти {
  138. return true
  139. }
  140. // Пробуем построить шахту
  141. _стр := strings.TrimPrefix(стр, "<a class=\"simple-but border mb5\" href=\"")
  142. _стр = strings.TrimSuffix(_стр, "\">")
  143. // https://wartank.ru/building-upgrade/Market?18-1.ILinkListener-upgradeLink-link
  144. // <a class="simple-but border mb5" href="Market?19-1.ILinkListener-upgradeLink-link">
  145. ссылка := "https://wartank.ru/building-upgrade/" + _стр
  146. списСтр = сам.Сеть().ВебВоркер().Получ(ссылка)
  147. // Проверить, что постройка состоялась
  148. for _, стр := range списСтр {
  149. if strings.Contains(стр, "ILinkListener-upgradeLink-link") {
  150. log.Printf("Рынок.построить().фнКупить(): покупка склада топлива не прошла\n\tlink=%v\n\tстр=\n\t%v\n", ссылка, стр)
  151. return false // Покупка не оплачена
  152. }
  153. }
  154. log.Printf("+++++Рынок.построить().фнКупить(): покупка склада топлива прошла\n")
  155. return true
  156. }
  157. фнПодтверждение := func() bool {
  158. for _, стр = range списСтр {
  159. // <a class="simple-but border w50 mXa mb10" w:id="confirmLink" href="../wicket/page?21-1.ILinkListener-confirmLink"><span><span>да, подтверждаю</span></span></a>
  160. if strings.Contains(стр, `ILinkListener-confirmLink`) {
  161. еслиНайти = true
  162. break
  163. }
  164. }
  165. if !еслиНайти {
  166. return true
  167. }
  168. // Пробуем построить шахту
  169. _стр := strings.TrimPrefix(стр, `<a class="simple-but border w50 mXa mb10" w:id="confirmLink" href="..`)
  170. _стр = strings.TrimSuffix(_стр, `"><span><span>да, подтверждаю</span></span></a>`)
  171. // https://wartank.ru/wicket/page?20-1.ILinkListener-confirmLink
  172. ссылка := "https://wartank.ru" + _стр
  173. списСтр = сам.Сеть().ВебВоркер().Получ(ссылка)
  174. // Проверить, что постройка состоялась
  175. for _, стр := range списСтр {
  176. if strings.Contains(стр, "<title>Вы сделали слишком большую паузу</title>") {
  177. log.Printf("Рынок.построить().фнПодтверждение(): подтверждение покупка склада топлива не прошла\n\tlink=%v\n\tстр=\n\t%v\n", ссылка, стр)
  178. return false // Покупка не оплачена
  179. }
  180. }
  181. log.Printf("+++++Рынок.построить().фнПодтверждение(): подтверждение покупка склада топлива прошла\n")
  182. return true
  183. }
  184. фнКомплекс := func() {
  185. for {
  186. if фнКупить() {
  187. if фнПодтверждение() {
  188. break
  189. }
  190. }
  191. }
  192. }
  193. фнКомплекс()
  194. return true
  195. }
  196. // Пытается проапгрейдить топливный склад
  197. func (сам *АренаРынок) проапгрейдить() bool {
  198. time.Sleep(time.Millisecond * 1000)
  199. var (
  200. еслиНайти = false
  201. списСтр []string
  202. стр = ""
  203. )
  204. фнКупить := func() bool {
  205. defer time.Sleep(time.Millisecond * 1000)
  206. списСтр = сам.Сеть().ВебВоркер().Получ("https://wartank.ru/building-upgrade/Market")
  207. for _, стр = range списСтр {
  208. // <a class="simple-but border mb5" href="Market?5-1.ILinkListener-upgradeLink-link">
  209. if strings.Contains(стр, `ILinkListener-upgradeLink-link`) {
  210. еслиНайти = true
  211. break
  212. }
  213. }
  214. if !еслиНайти {
  215. return true
  216. }
  217. // Пробуем улучшить шахту
  218. _стр := strings.TrimPrefix(стр, "<a class=\"simple-but border mb5\" href=\"")
  219. _стр = strings.TrimSuffix(_стр, "\">")
  220. // https://wartank.ru/building-upgrade/Market?4-1.ILinkListener-upgradeLink-link
  221. // <a class="simple-but border mb5" href="Market?50-1.ILinkListener-upgradeLink-link">
  222. ссылка := "https://wartank.ru/building-upgrade/" + _стр
  223. списСтр = сам.Сеть().ВебВоркер().Получ(ссылка)
  224. // Проверить, что постройка состоялась
  225. for _, стр := range списСтр {
  226. if strings.Contains(стр, "ILinkListener-upgradeLink-link") {
  227. log.Printf("Рынок.проапгрейдить().фнКупить(): покупка рынка не прошла\n\tlink=%v\n\tстр=\n\t%v\n", ссылка, стр)
  228. return false // Покупка не оплачена
  229. }
  230. }
  231. log.Printf("+++++Рынок.проапгрейдить().фнКупить(): покупка рынка прошла\n")
  232. return true
  233. }
  234. фнПодтверждение := func() bool {
  235. for _, стр = range списСтр {
  236. // <a class="simple-but border w50 mXa mb10" w:id="confirmLink" href="../wicket/page?7-1.ILinkListener-confirmLink"><span><span>да, подтверждаю</span></span></a>
  237. if strings.Contains(стр, `ILinkListener-confirmLink`) {
  238. еслиНайти = true
  239. break
  240. }
  241. }
  242. if !еслиНайти {
  243. return true
  244. }
  245. // Пробуем построить шахту
  246. _стр := strings.TrimPrefix(стр, `<a class="simple-but border w50 mXa mb10" w:id="confirmLink" href="..`)
  247. _стр = strings.TrimSuffix(_стр, `"><span><span>да, подтверждаю</span></span></a>`)
  248. // https://wartank.ru/wicket/page?6-1.ILinkListener-confirmLink
  249. ссылка := "https://wartank.ru" + _стр
  250. списСтр = сам.Сеть().ВебВоркер().Получ(ссылка)
  251. // Проверить, что постройка состоялась
  252. for _, стр := range списСтр {
  253. if strings.Contains(стр, "<title>Вы сделали слишком большую паузу</title>") {
  254. log.Printf("Рынок.проапгрейдить().фнПодтверждение(): подтверждение покупка рынка не прошла\n\tlink=%v\n\tстр=\n\t%v\n", ссылка, стр)
  255. return false // Покупка не оплачена
  256. }
  257. }
  258. log.Printf("+++++Рынок.проапгрейдить().фнПодтверждение(): подтверждение покупка склада топлива прошла\n")
  259. return true
  260. }
  261. фнКомплекс := func() {
  262. count := 5
  263. for count > 0 {
  264. if фнКупить() {
  265. if фнПодтверждение() {
  266. break
  267. }
  268. }
  269. count--
  270. }
  271. }
  272. фнКомплекс()
  273. return true
  274. }
  275. // Проверяет время ожидания рынка
  276. func (сам *АренаРынок) проверОжидание() {
  277. var (
  278. strOut string
  279. еслиНайдено bool
  280. )
  281. // countDown := сам.CountDown().Get()
  282. фнЕслиСеребро := func() bool { // Найти счётчик цены серебра
  283. сам.Обновить()
  284. еслиНайдено := false
  285. lstMarket := сам.СписПолучить()
  286. for _, strOut = range lstMarket {
  287. if strings.Contains(strOut, `alt="Серебро" title="Серебро"> `) {
  288. еслиНайдено = true
  289. break
  290. }
  291. }
  292. if еслиНайдено {
  293. lstSilver := strings.Split(strOut, `<img class="ico vm" src="/images/icons/silver.png?2" alt="Серебро" title="Серебро"> `)
  294. strSilver := lstSilver[1]
  295. switch strSilver {
  296. case "10", "50", "100", "500":
  297. return true
  298. default:
  299. серебро := сам.конт.Get("серебро").Val().(int)
  300. if серебро > 1_000_000 {
  301. return true
  302. }
  303. return false
  304. }
  305. }
  306. return false
  307. }
  308. fnGetCountDown := func() { // Искать счётчик времени
  309. lstMarket := сам.СписПолучить()
  310. // Найти счётчик времени
  311. for _, strOut = range lstMarket {
  312. if strings.Contains(strOut, `Минимальная цена через `) {
  313. еслиНайдено = true
  314. break
  315. }
  316. }
  317. if !еслиНайдено {
  318. return // Минимальная цена
  319. }
  320. lstTime := strings.Split(strOut, `Минимальная цена через `)
  321. strTime := lstTime[1]
  322. if err := сам.ОбратВремяУст(АВремя(strTime)); err != nil {
  323. // log._rintf("ERRO Market.checkTime(): при установке времени ожидания рынка(%v)\n\terr=%v\n", strTime, err)
  324. return // Возможно минимальная цена
  325. }
  326. }
  327. if фнЕслиСеребро() {
  328. return
  329. }
  330. fnGetCountDown()
  331. }
  332. // Проверяет рынок на режим покупки
  333. func (сам *АренаРынок) купитьЗолото() bool {
  334. var (
  335. ind int
  336. еслиНайдено bool
  337. strOut string
  338. lstMarket = сам.СписПолучить()
  339. strSilver string
  340. )
  341. for ind, strOut = range lstMarket {
  342. if strings.Contains(strOut, `alt="Серебро" title="Серебро"> `) {
  343. еслиНайдено = true
  344. break
  345. }
  346. }
  347. if !еслиНайдено { // Не найдена продажа золота за серебро
  348. return false
  349. }
  350. lstSilver := strings.Split(strOut, `<img class="ico vm" src="/images/icons/silver.png?2" alt="Серебро" title="Серебро"> `)
  351. strSilver = lstSilver[1]
  352. серебро := сам.конт.Get("серебро").Val().(int)
  353. еслиКупить := false
  354. switch strSilver {
  355. case "10", "50", "100", "500": // Допустимые суммы трат
  356. еслиКупить = true
  357. case "1000": // Если стоит тысяча серебра
  358. if серебро > 500_000 { // Если серебра больше полумиллиона -- покупаем
  359. еслиКупить = true
  360. }
  361. case "5000", "10000": // Если большая сумма -- можно купить и больше
  362. if серебро > 1_000_000 {
  363. еслиКупить = true
  364. }
  365. }
  366. if !еслиКупить {
  367. return false
  368. }
  369. ind -= 15
  370. strOut = lstMarket[ind]
  371. lstLink := strings.Split(strOut, `<a class="simple-but border mb5" href="`)
  372. if len(lstLink) < 2 {
  373. return false
  374. }
  375. strLink := lstLink[1]
  376. lstLink = strings.Split(strLink, `"><span><span>Получить `)
  377. strLink = "https://wartank.ru/" + lstLink[0]
  378. lstMarket, err := сам.Сеть().Get(strLink)
  379. if err != nil {
  380. // log._rintf("ERRO Market.buyGold(): при выполнении GET-команды на покупку золота, err=\n\t%v\n", err)
  381. return true
  382. }
  383. for _, strOut = range lstMarket {
  384. if strings.Contains(strOut, `Ошибка на сервере. Сообщение админу уже отправлено.`) {
  385. // log._rintf("ERRO Market.buyGold(): при получении lstMarket, strHTML=%v, err=\nt%v\n", strOut, err)
  386. return false
  387. }
  388. }
  389. if err = сам.СтрОбновить(lstMarket); err != nil {
  390. // log._rintf("Market.buyGold(): при обновлении lstMarket, err=\n\t%v\n", err)
  391. return true
  392. }
  393. return true
  394. }