commands.pm 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556
  1. use Modern::Perl;
  2. use utf8;
  3. use Mojo::Base -strict, -async_await, -signatures;
  4. use Data::Dumper;
  5. use experimental qw/switch/;
  6. our $client;
  7. our $abon_client;
  8. our $config;
  9. our $redis;
  10. sub __
  11. {
  12. @_;
  13. }
  14. ##############################################
  15. our $commands = [
  16. {command=>"help", name=>__("Помощь"), description=>__("Список доступных команд")},
  17. {command=>"info", name=>__("Информация"), description=>__("Информация о пользователе")},
  18. {command=>"balance", name=>__("Баланс"), description=>__("Проверка баланса"), main=>1, icon=>"\x{1F45B}"},
  19. {command=>"service", name=>__("Сервисы"), description=>__("Подключенные сервисы")},
  20. {command=>"credit", name=>__("Кредит"), description=>__("Установка кредита"), main=>1, icon=>"\x{1FAF0}"},
  21. {command=>"card", name=>__("Карта пополнения"), description=>__("Оплата карточкой пополнения"), main=>1, icon=>"\x{1F4B3}"},
  22. {command=>"payberry", name=>__("Payberry"), description=>__("Оплата через Payberry"), main=>1, icon=>"\x{1F17F}\x{FE0F}"},
  23. {command=>"transfer", name=>__("Перевод"), description=>__("Перевод денег")},
  24. {command=>"new_task", name=>__("Новая заявка"), description=>__("Новая заявка")},
  25. {command=>"tasks", name=>__("Заявки"), description=>__("Открытые заявки"), main=>1, icon=>"\x{2692}\x{FE0F}"},
  26. {command=>"support", name=>__("Техподдержка"), description=>__("Связь с техподдержкой")},
  27. {command=>"logout", name=>__("Выход"), description=>__("Выход")},
  28. ];
  29. #############################
  30. our $lex_actions = {
  31. xfer => {
  32. name => __("перевод денег"),
  33. canceled => __("отменён"),
  34. },
  35. card => {
  36. name => __("пополнение карточкой"),
  37. canceled => __("отмененo"),
  38. },
  39. };
  40. our $lex_vars = {
  41. xfer_to => __("номер личного счета"),
  42. xfer_amount => __("сумму"),
  43. card_code => __("код карточки"),
  44. card_serial => __("номер карточки"),
  45. };
  46. ##############################################
  47. sub command_help($fsa, $info)
  48. {
  49. my @list = map { "<b>/$_->{command}</b> " . _($_->{description}) } @$commands;
  50. reply($info, join("\n", @list));
  51. }
  52. async sub command_logout
  53. {
  54. my ($fsa, $info) = @_;
  55. my $uid = $fsa->note("[uid]");
  56. await $client->delete_p("client", "/client/$uid/telegram");
  57. reply_with($info, {button_menu => 0}, _("Благодарим за использование нашего бота"));
  58. $fsa->delete_note("[uid]");
  59. $fsa->state("logged_out");
  60. }
  61. async sub command_balance
  62. {
  63. my ($fsa, $info) = @_;
  64. my $uid = $fsa->note("[uid]");
  65. my $money = await $abon_client->get_p($info, "client", "/client/$uid/money?human=1");
  66. my $cur = $money->{human};
  67. my @lines = (
  68. sprintf("<u>%s:</u> <b>%.2f $cur</b> (%s %.2f $cur + %s %.2f $cur) ",
  69. _("Ваш баланс"), $money->{balance}, ("депозит"), $money->{deposit}, _("кредит"), $money->{credit}),
  70. );
  71. push @lines, sprintf("<u>%s:</u> %s", _("Оплачено включительно до"), format_date($money->{last_day})) if $money->{last_day} ne "-";
  72. push @lines, sprintf("<u>%s:</u> %d%%", _("Скидка"), $money->{reduction}) if $money->{reduction};
  73. push @lines, sprintf("<u>%s:</u> <b>%.2f $cur</b> %s (%s)",
  74. _("Последнее снятие"), $money->{last_withdrawal}->{sum}, format_time($money->{last_withdrawal}->{date}), $money->{last_withdrawal}->{comment})
  75. if $money->{last_withdrawal}->{sum};
  76. push @lines, sprintf("<u>%s:</u> <b>%.2f $cur</b> %s", _("Последний платеж"), $money->{last_payment}->{sum}, format_time($money->{last_payment}->{date}))
  77. if $money->{last_payment}->{sum};
  78. for (keys %{ $money->{accounts} })
  79. {
  80. push @lines, sprintf("<u>%s:</u> <b>%.2f $cur</b>", $_, $money->{accounts}->{$_});
  81. }
  82. reply($info, @lines);
  83. }
  84. async sub command_info
  85. {
  86. my ($fsa, $info) = @_;
  87. my $uid = $fsa->note("[uid]");
  88. my $client = await $abon_client->get_p($info, "client", "/client/$uid");
  89. reply($info,
  90. sprintf("<u>%s</u>: %d", _("Номер учетной записи"), $client->{uid}),
  91. sprintf("<u>%s</u>: %s", _("Логин"), $client->{login}),
  92. sprintf("<u>%s</u>: %s", _("ФИО"), $client->{fio}),
  93. sprintf("<u>%s</u>: %s", _("Адрес"), $client->{address}),
  94. sprintf("<u>%s</u>: %s", _("Телефон"), $client->{phone}),
  95. );
  96. }
  97. async sub command_credit
  98. {
  99. my ($fsa, $info) = @_;
  100. my $uid = $fsa->note("[uid]");
  101. my $money = await $abon_client->get_p($info, "client", "/client/$uid/money?human=1");
  102. if ($money->{credit} > 0)
  103. {
  104. return reply($info, sprintf("%s <b>%.2f %s</b>. %s", _("У вас уже установлен кредит"), $money->{credit}, $money->{human},
  105. _("Продлевать кредит повторно до оплаты нельзя. При следующей оплате кредит будет погашен")));
  106. }
  107. reply_with($info, {
  108. inline_menu => [[
  109. { text=>_("Я согласен с условиями"), callback_data=>"\x00/set-credit" },
  110. ]],
  111. },
  112. _("Вы можете самостоятельно установить кредит на два дня"),
  113. _("<b>Ограничения</b>: только для физических лиц, продлевать кредит повторно до оплаты нельзя. При следующей оплате кредит будет погашен"),
  114. );
  115. }
  116. async sub callback_set_credit
  117. {
  118. my ($fsa, $info) = @_;
  119. my $uid = $fsa->note("[uid]");
  120. my $res = await $abon_client->post_p($info, "client", "/client/$uid/credit", {human=>1});
  121. return reply($info, _("Кредит не имеет смысла для бесплатных тарифных планов")) if $res->{credit} == 0;
  122. await reply_with($info, {button_menu=>1},
  123. sprintf("%s <b>%.2f %s</b>", _("Установлен кредит "), $res->{credit}, $res->{human}),
  124. "",
  125. );
  126. command_balance($fsa, $info);
  127. }
  128. async sub command_service
  129. {
  130. my ($fsa, $info) = @_;
  131. my $uid = $fsa->note("[uid]");
  132. my $res = await $abon_client->get_p($info, "client", "/client/$uid/service?human=1&as-array=1");
  133. my @list = map { sprintf("<u>%s:</u> %s (%s '%s')", $_->{name_ru}, format_wd($_->{tariff}, $_->{human}), _("тариф"), $_->{tariff}->{name_ru}) }
  134. grep { !$_->{disabled} } @$res;
  135. reply($info, @list);
  136. };
  137. ######################################
  138. async sub command_transfer
  139. {
  140. my ($fsa, $info) = @_;
  141. reply($info, _("Введите номер личного счета пользователя, которому вы хотите перевести деньги со своего собственного счета"));
  142. return needs_input($fsa, "xfer", "xfer_to");
  143. }
  144. async sub verify_xfer_to
  145. {
  146. my ($fsa, $target_uid, $info) = @_;
  147. my $uid = $fsa->note("[uid]");
  148. return _("Номер личного счета должен состоять из цифр") unless $target_uid =~ /^\d+$/;
  149. return _("Нельзя перевести деньги себе самому") if $uid==$target_uid;
  150. my $res = eval { await $client->get_p("client", "/client/$target_uid") };
  151. if ($@)
  152. {
  153. die $@ unless $@->{code} == 404;
  154. return _("Абонент") . " $target_uid " . _("не найден");
  155. }
  156. return _("Абонент") . " $target_uid " . _("отключен") if $res->{disabled};
  157. $fsa->note("xfer_fio" => $res->{fio});
  158. return undef;
  159. }
  160. async sub use_xfer_to
  161. {
  162. my ($fsa, $target_uid, $info) = @_;
  163. my $uid = $fsa->note("[uid]");
  164. my $nick = $fsa->note("xfer_fio");
  165. my @fio = split(/\s+/, $nick);
  166. if (@fio)
  167. {
  168. my $f = shift(@fio);
  169. $nick = join(" ", (substr($f, 0, 1) . ".", @fio));
  170. }
  171. $fsa->note(xfer_nick => $nick);
  172. reply(
  173. $info, _("Получатель денег: ") . $nick,
  174. _("Теперь введите сумму, которую хотите перевести"),
  175. );
  176. return needs_input($fsa, "xfer", "xfer_amount");
  177. }
  178. sub verify_xfer_amount
  179. {
  180. my ($fsa, $amount, $info) = @_;
  181. my $uid = $fsa->note("[uid]");
  182. $amount =~ s/,/./g;
  183. $amount =~ s/[^\d\.]//g;
  184. my $tmp = $amount;
  185. my $count = $amount =~ tr/.//;
  186. return _("Вы ввели неправильную сумму") if !$amount || $count>1;
  187. }
  188. async sub use_xfer_amount
  189. {
  190. my ($fsa, $amount, $info) = @_;
  191. my $uid = $fsa->note("[uid]");
  192. my $to_uid = $fsa->note("xfer_to");
  193. my $nick = $fsa->note("xfer_nick");
  194. reply_with($info, {
  195. inline_menu => [[
  196. { text=>_("Подтвердите перевод"), callback_data=>"\x00/transfer" },
  197. ]]
  198. },
  199. sprintf("%.2f %s %s %d (%s)", $amount, $config->{currency}->{human}, _(" на счет абонента"), $to_uid, $nick)
  200. );
  201. return "command";
  202. }
  203. async sub callback_transfer
  204. {
  205. my ($fsa, $info) = @_;
  206. my $uid = $fsa->note("[uid]");
  207. my $to_uid = $fsa->note("xfer_to");
  208. my $amount = $fsa->note("xfer_amount");
  209. return unless $to_uid && $amount;
  210. my $res = await $abon_client->post_p($info, "client", "/client/$uid/money/to/$to_uid", {
  211. amount => $amount,
  212. ip => "0.0.0.0",
  213. via => "abonbot",
  214. currency => $config->{currency}->{name},
  215. });
  216. reply_with($info, {button_menu=>1}, _("Деньги успешно переведены"));
  217. command_balance($fsa, $info);
  218. }
  219. ##########################################
  220. async sub command_new_task
  221. {
  222. my ($fsa, $info) = @_;
  223. my $uid = $fsa->note("[uid]");
  224. my $res = await $abon_client->get_p($info, "task", "/task?created_by_client=$uid&list=new,work");
  225. if ($res->{total})
  226. {
  227. return reply($info, _("Создание новой заявки невозможно, пока не будут решены уже открытые"));
  228. }
  229. reply($info, _("Изложите вашу проблему"));
  230. $fsa->state("task_needs_descr");
  231. };
  232. async sub callback_task_post
  233. {
  234. my ($fsa, $info) = @_;
  235. my $uid = $fsa->note("[uid]");
  236. my $descr = $fsa->note("task_text");
  237. return unless $descr;
  238. my $params = {
  239. description => $descr,
  240. "for-client" => $uid,
  241. client => $uid,
  242. type => "client-issue",
  243. list => "new",
  244. task_attr => {
  245. source => "telegram"
  246. },
  247. };
  248. my $res = await $client->post_json_p("task", "/task", $params);
  249. reply_with($info, {button_menu=>1}, _("Ваша заявка размещена"));
  250. command_tasks($fsa, $info);
  251. }
  252. async sub callback_task_cancel
  253. {
  254. my ($fsa, $info) = @_;
  255. $fsa->delete_note("task_text");
  256. $fsa->delete_note("if_edited");
  257. reply_with($info, {button_menu=>1}, _("Заявка отменена"));
  258. }
  259. async sub command_tasks
  260. {
  261. my ($fsa, $info) = @_;
  262. my $uid = $fsa->note("[uid]");
  263. my $res = await $abon_client->get_p($info, "task", "/task?created_by_client=$uid&list=new,work&sort=id&with_comments_for_client=1");
  264. return reply($info, _("Открытых заявок нет")) unless $res->{total};
  265. my @str = map {
  266. _("Заявка") . " <b>$_->{number}</b>\n$_->{description}\n<i>" . _("Комментариев") . " " . $_->{total_comments} . ", " . _(" не прочитано") . " " . $_->{unread_comments} . "</i>"
  267. } @{$res->{data}};
  268. my $menu = [[
  269. map {
  270. { text => _("Комментарии к заявке") . " " . $_->{number}, callback_data => "\x00/task $_->{entity}" }
  271. } @{$res->{data}}
  272. ]];
  273. reply_with($info, {inline_menu=>$menu}, @str);
  274. };
  275. async sub callback_task
  276. {
  277. my ($fsa, $info, $task_id) = @_;
  278. my $uid = $fsa->note("[uid]");
  279. my $res = await $client->get_p("task", "/task/$task_id?with_comments_for_client=1");
  280. my @str = map {
  281. format_timestamp($_->{created}) . " " . "<b>" . ($_->{created_by}->[0] eq "worker" ? _("Оператор") : _("Вы")) . ":</b>\n"
  282. . $_->{text}
  283. } @{ $res->{comments} };
  284. my $menu = [[
  285. {text => _("Добавить комментарий"), callback_data => "\x00comment $task_id"},
  286. ]];
  287. reply_with($info, {inline_menu=>$menu}, @str);
  288. $client->post_p("task", "/task/$task_id/comment/read", {for_client=>$uid});
  289. }
  290. async sub callback_comment
  291. {
  292. my ($fsa, $info, $task_id) = @_;
  293. $fsa->note(task_id => $task_id);
  294. $fsa->state("task_needs_comment");
  295. reply($info, _("Введите текст вашего ответа"));
  296. }
  297. ##################################################
  298. async sub command_payberry
  299. {
  300. my ($fsa, $info) = @_;
  301. my $uid = $fsa->note("[uid]");
  302. my $menu = [[
  303. {text => _("Payberry"), url => $config->{pay}->{payberry_url} . "?acc=$uid"}
  304. ]];
  305. reply_with($info, {inline_menu=>$menu}, _("Для оплаты перейдите по ссылке"));
  306. }
  307. #####################################
  308. use constant FAIL_BLOCK => 10;
  309. use constant FAIL_ASK_SERIAL => 3;
  310. async sub command_card
  311. {
  312. my ($fsa, $info) = @_;
  313. my $uid = $fsa->note("[uid]");
  314. my $key = "card-guess-$uid";
  315. my $failed = $redis->get($key) || 0;
  316. if ($failed > FAIL_BLOCK)
  317. {
  318. return reply($info, _("Вы ввели неправильный код слишком много раз. Пополнение карточкой заблокировано на сутки"));
  319. }
  320. my $res = await $client->get_p("client", "/client/$uid");
  321. if ($res->{disabled})
  322. {
  323. return reply($info, _("Пополнение счета недоступно отключенным пользователям"));
  324. }
  325. reply($info, _("Введите код карточки пополнения (16 цифр, можно разделять их знаком '-')"));
  326. return needs_input($fsa, "card", "card_code");
  327. }
  328. sub verify_card_code($fsa, $code, $info)
  329. {
  330. $code =~ s/\D//g;
  331. return _("Код карточки должен состоять из 16 цифр") unless $code =~ /^\d{16}$/;
  332. return undef;
  333. }
  334. async sub use_card_code
  335. {
  336. my ($fsa, $code, $info) = @_;
  337. my $uid = $fsa->note("[uid]");
  338. my $key = "card-guess-$uid";
  339. my $fails = $redis->get($key) || 0;
  340. if ($fails > FAIL_ASK_SERIAL)
  341. {
  342. reply($info, _("Теперь введите номер карточки (7 цифр, можно разделять знаком'-')"));
  343. return needs_input($fsa, "card", "card_serial");
  344. }
  345. else
  346. {
  347. pay_from_card($info, $uid, $code, "");
  348. return "command";
  349. }
  350. }
  351. sub verify_card_serial($fsa, $serial, $info)
  352. {
  353. $serial =~ s/\D//g;
  354. return _("Номер карточки должен состоять из 7 цифр") unless $serial =~ /^\d{7}$/;
  355. return undef;
  356. }
  357. async sub use_card_serial
  358. {
  359. my ($fsa, $serial, $info) = @_;
  360. pay_from_card($info, $fsa->note("[uid]"), $fsa->note("card_code"), $serial);
  361. }
  362. async sub pay_from_card
  363. {
  364. my ($info, $uid, $code, $serial) = @_;
  365. my $args = {
  366. code => $code =~ s/\D//gr,
  367. uid => $uid,
  368. domain => $config->{domain},
  369. source => "abonbot",
  370. ip => "0.0.0.0",
  371. };
  372. $args->{serial} = $serial =~ s/\D//gr if $serial;
  373. my $res = eval { await $client->post_p("card", "/redemption", $args) };
  374. unless($@)
  375. {
  376. return reply_with($info, {button_menu=>1}, _("Ваш счёт пополнен на " . $res->{amount} . " " . $res->{currency_short_name_ru} . ". " . _("Оплачено включительно до ") . $res->{paid_until}));
  377. };
  378. given($@->{code})
  379. {
  380. when(400)
  381. {
  382. reply_with($info, {button_menu=>1}, _("Вы ввели неправильный номер или код карточки."), _("Пополнение прервано"));
  383. my $key = "card-guess-$uid";
  384. $redis->incr($key);
  385. $redis->expire($key, 24 * 3600);
  386. }
  387. when(409)
  388. {
  389. reply_with($info, {button_menu=>1}, _("Карточка уже использована."), _("Пополнение прервано"));
  390. }
  391. when(406)
  392. {
  393. reply_with($info, {button_menu=>1}, _("Карточка не активирована. Обратитесь в службу поддержки."), _("Пополнение прервано"));
  394. }
  395. default
  396. {
  397. die $@;
  398. }
  399. }
  400. };
  401. #####################################################
  402. sub command_support
  403. {
  404. my ($fsa, $info) = @_;
  405. return reply($info, _("Телефоны техподдержки:"), @{$config->{support_phones}});
  406. }
  407. ##################################################
  408. sub needs_input($fsa, $action, $var)
  409. {
  410. $fsa->delete_note($var);
  411. $fsa->note(input_var => $var);
  412. $fsa->note(input_action => $action);
  413. $fsa->state("needs_input");
  414. return "needs_input";
  415. }
  416. ##################################################
  417. sub format_wd($rec, $cur)
  418. {
  419. return _("бесплатно") if $rec->{dayly} == 0 && $rec->{monthly} == 0;
  420. my $m = sprintf("<b>%.2f $cur</b> %s", $rec->{monthly}, _("в месяц")) if $rec->{monthly} != 0;
  421. my $d = sprintf("<b>%.2f $cur</b> %s", $rec->{dayly}, _("в день")) if $rec->{dayly} != 0;
  422. return ("$m + $d") if $m && $d;
  423. return $m if $m;
  424. return $d if $d;
  425. }
  426. sub parse_error
  427. {
  428. my $e = shift;
  429. return $e unless ref $e;
  430. return "$e->{code} $e->{message} $e->{body}";
  431. }
  432. 1;
  433. # локализация