rules.pm 9.6 KB

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