rules.pm 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  1. use Modern::Perl;
  2. use Data::Dumper;
  3. use Mojo::Base -strict, -signatures, -async_await;
  4. use Mojo::Util;
  5. use Scalar::Util;
  6. use experimental qw/switch/;
  7. use fsa;
  8. our $redis;
  9. our $config;
  10. our $client;
  11. our $commands;
  12. our $kb_menu;
  13. our $abon_client;
  14. use constant FAIL_ASK_SERIAL => 3;
  15. ##################################################
  16. my $rules = {
  17. logged_out => async sub
  18. {
  19. my ($fsa, $line, $info) = @_;
  20. if ($line ne "/start")
  21. {
  22. reply($info, _("Для начала работы наберите <b>/start</b>"));
  23. return "logged_out";
  24. }
  25. # Поступила команда /start
  26. # У нас есть uid в автомате? Это значит, что мы прочитали состояние из внешней базы
  27. if ($fsa->note("uid"))
  28. {
  29. reply($info, _("Для получения списка доступных команд введите /help"));
  30. return "command";
  31. }
  32. # Проверяем, логинился ли уже пользователь в нашем боте
  33. my $res = eval { await $client->get_p("client", "/telegram/$info->{id}/client") };
  34. if ($@)
  35. {
  36. if ($@->{code} == 410)
  37. {
  38. reply($info,
  39. _("Вас приветствует провайдер") . " " . $config->{provider},
  40. _("Введите номер учетной записи или логин")
  41. );
  42. return "needs_login";
  43. }
  44. else
  45. {
  46. die $@;
  47. }
  48. }
  49. reply($info, greet($res->{fio}), _("Для получения списка доступных команд введите /help"));
  50. $fsa->note(uid => $res->{uid});
  51. return "command";
  52. },
  53. needs_login => sub($fsa, $line, $info)
  54. {
  55. $fsa->note(login => $line);
  56. reply($info, _("Теперь введите пароль"));
  57. "needs_password";
  58. },
  59. needs_password => async sub
  60. {
  61. my ($fsa, $line, $info) = @_;
  62. request("deleteMessage", { chat_id=>$info->{id}, message_id=>$info->{msgid} });
  63. my $res = eval { await $client->post_p("client", "/telegram/client", {login=>$fsa->note("login"), password=>$line, telegram_id=>$info->{id}}) };
  64. if ($@)
  65. {
  66. $fsa->state("logged_out");
  67. $fsa->delete_note("login");
  68. die $@;
  69. }
  70. reply($info, greet($res->{fio}), _("Для получения списка доступных команд введите /help"));
  71. $fsa->note(uid => $res->{uid});
  72. $fsa->delete_note("login");
  73. "command";
  74. },
  75. command => async sub
  76. {
  77. my ($fsa, $line, $info) = @_;
  78. await do_command($fsa, $line, $info);
  79. $fsa->state; # или все тот же command, или команда установила уже свое состояние
  80. },
  81. #### Перевод денег
  82. xfer_needs_uid => async sub
  83. {
  84. my ($fsa, $target_uid, $info) = @_;
  85. my $uid = $fsa->note("uid");
  86. $fsa->state("command"); # на случай die
  87. unless($target_uid)
  88. {
  89. reply($info, _("Перевод денег прерван"));
  90. }
  91. unless ($target_uid =~ /^\d+$/)
  92. {
  93. reply($info, _("Неправильный номер личного счета"), _("Введите его заново или пустую строку, если передумали пополнять"));
  94. return "xfer_needs_uid";
  95. }
  96. if ($uid==$target_uid)
  97. {
  98. reply($info, _("Нельзя перевести деньги себе самому"));
  99. return "command";
  100. }
  101. my $res = await $abon_client->get_p($info, "client", "/client/$target_uid");
  102. if ($res->{disabled})
  103. {
  104. reply($info, _("Абонент $target_uid отключен"));
  105. return "command";
  106. }
  107. my $nick = $res->{fio};
  108. my @fio = split(/\s+/, $res->{fio});
  109. if (@fio)
  110. {
  111. my $f = shift(@fio);
  112. $nick = join(" ", (substr($f, 0, 1) . ".", @fio));
  113. }
  114. reply(
  115. $info, _("Получатель денег: ") . $nick,
  116. "Теперь введите сумму, которую хотите перевести",
  117. );
  118. $fsa->note(xfer_to => $target_uid);
  119. return "xfer_needs_amount";
  120. },
  121. xfer_needs_amount => async sub
  122. {
  123. my ($fsa, $amount, $info) = @_;
  124. my $uid = $fsa->note("uid");
  125. my $to_uid = $fsa->note("xfer_to");
  126. unless($amount)
  127. {
  128. reply($info, _("Перевод денег прерван"));
  129. }
  130. $amount =~ s/,/./g;
  131. $amount =~ s/[^\d\.]//g;
  132. my $tmp = $amount;
  133. my $count = $amount =~ tr/.//;
  134. if (!$amount || $count>1)
  135. {
  136. reply($info, _("Вы ввели неправильную сумму"), _("Введите ее заново или пустую строку, если передумали пополнять"));
  137. return "xfer_needs_amount";
  138. }
  139. $fsa->note(xfer_amount => $amount);
  140. reply_with($info, {
  141. inline_menu => [[
  142. { text=>_("Подтвердите перевод"), callback_data=>"\x00/transfer" },
  143. ]]
  144. },
  145. sprintf("%.2f %s %s %d", $amount, $config->{currency}->{human}, _(" на счет абонента"), $to_uid),
  146. );
  147. return "command";
  148. },
  149. #### Заявки
  150. task_needs_descr => async sub
  151. {
  152. my ($fsa, $descr, $info) = @_;
  153. my $uid = $fsa->note("uid");
  154. $fsa->note(task_text => $descr);
  155. reply_with($info, {
  156. inline_menu => [[
  157. { text=>_("Отправить"), callback_data=>"\x00/task_post" },
  158. { text=>_("Отменить"), callback_data=>"\x00/task_cancel" },
  159. ]]
  160. },
  161. _("Перед отправкой заявки вы можете ее отредактировать средствами Телеграм")
  162. );
  163. return "command";
  164. },
  165. task_needs_comment => async sub
  166. {
  167. my ($fsa, $comment, $info) = @_;
  168. my $uid = $fsa->note("uid");
  169. my $task_id = $fsa->note("task_id");
  170. my $params = {
  171. text => $comment,
  172. "for-client" => $uid,
  173. };
  174. $fsa->state("command"); # на случай die
  175. my $res = await $client->post_p("task", "/task/$task_id/comment", $params);
  176. reply($info, "Ваш ответ добавлен");
  177. $fsa->delete_note("task_id");
  178. return "command";
  179. },
  180. ##### Карточка пополнения
  181. card_needs_code => sub
  182. {
  183. my ($fsa, $code, $info) = @_;
  184. my $uid = $fsa->note("uid");
  185. $fsa->state("command"); # на случай die
  186. $code =~ s/\D//g;
  187. unless($code)
  188. {
  189. reply($info, _("Пополнение карточкой прервано"));
  190. }
  191. unless ($code =~ /^\d{16}$/)
  192. {
  193. reply($info, _("Код карточки должен состоять из 16 цифр"), _("Введите его заново или пустую строку, если передумали пополнять"));
  194. return "card_needs_code";
  195. }
  196. my $key = "card-guess-$uid";
  197. my $fails = $redis->get($key) || 0;
  198. if ($fails > FAIL_ASK_SERIAL)
  199. {
  200. $fsa->note(card_code => $code);
  201. reply($info, _("Теперь введите номер карточки (7 цифр, можно разделять знаком'-')"));
  202. return "card_needs_serial";
  203. }
  204. else
  205. {
  206. pay_from_card($info, $uid, $code, "");
  207. return "command";
  208. }
  209. },
  210. card_needs_serial => sub
  211. {
  212. my ($fsa, $serial, $info) = @_;
  213. $fsa->state("command"); # на случай die
  214. unless($serial)
  215. {
  216. reply($info, _("Пополнение карточкой прервано"));
  217. }
  218. unless ($serial =~ /^\d{7}$/)
  219. {
  220. reply($info, _("Номер карточки должен состоять из 7 цифр"), _("Введите его заново или пустую строку, если передумали пополнять"));
  221. return "card_needs_serial";
  222. }
  223. pay_from_card($info, $fsa->note("uid"), $fsa->note("card_code"), $serial);
  224. return "command";
  225. },
  226. };
  227. sub greet($whom)
  228. {
  229. my ($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime(time);
  230. my $greet = $hour>=4 && $hour<=10 ? _("Доброе утро") : $hour>10 && $hour<19 ? _("Добрый день") : _("Добрый вечер");
  231. "$greet, $whom";
  232. }
  233. async sub pay_from_card
  234. {
  235. my ($info, $uid, $code, $serial) = @_;
  236. my $args = {
  237. code => $code,
  238. uid => $uid,
  239. domain => $config->{domain},
  240. source => "abonbot",
  241. ip => "0.0.0.0",
  242. };
  243. $args->{serial} = $serial if $serial;
  244. my $res = eval { await $client->post_p("card", "/redemption", $args) };
  245. unless($@)
  246. {
  247. return reply($info, _("Ваш счёт пополнен на " . $res->{amount} . " " . $res->{currency_short_name_ru} . ". " . _("Оплачено до ") . $res->{paid_until}));
  248. };
  249. given($@->{code})
  250. {
  251. when(400)
  252. {
  253. reply($info, _("Вы ввели неправильный номер или код карточки"));
  254. my $key = "card-guess-$uid";
  255. $redis->incr($key);
  256. $redis->expire($key, 24 * 3600);
  257. }
  258. when(409)
  259. {
  260. reply($info, _("Карточка уже использована"));
  261. }
  262. when(406)
  263. {
  264. reply($info, _("Карточка не активирована. Обратитесь в службу поддержки"));
  265. }
  266. default
  267. {
  268. die $@;
  269. }
  270. }
  271. };
  272. ######################
  273. sub make_key($id)
  274. {
  275. return "abonbot-$id";
  276. }
  277. sub save_fsa($fsa, $chatid)
  278. {
  279. my $key = make_key($chatid);
  280. say "saving state ", $fsa->state, Dumper $fsa->notes;
  281. my $notes = $fsa->notes;
  282. utf8::encode($notes->{$_}) for keys %$notes;
  283. $redis->del($key);
  284. $redis->hmset($key, %$notes);
  285. $redis->expire($key, 3600*24);
  286. }
  287. sub make_fsa($chatid, $from)
  288. {
  289. my $id = make_key($chatid);
  290. my $notes = { $redis->hgetall($id) };
  291. if (keys %$notes)
  292. {
  293. utf8::decode($notes->{$_}) for keys %$notes;
  294. return fsa->new($rules, $notes->{_state}, $notes);
  295. }
  296. else
  297. {
  298. return fsa->new($rules, "logged_out", {});
  299. }
  300. }
  301. 1;