rules.pm 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  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. our $lex_vars;
  14. our $lex_actions;
  15. ##################################################
  16. my $rules = {
  17. logged_out => async sub
  18. {
  19. my ($fsa, $line, $info) = @_;
  20. if ($line ne "/start")
  21. {
  22. reply_with($info, { button_menu => 0 }, _("Для начала работы наберите <b>/start</b>"));
  23. return "logged_out";
  24. }
  25. # Поступила команда /start
  26. # У нас есть uid в автомате? Это значит, что мы прочитали состояние из внешней базы
  27. if ($fsa->note("[uid]"))
  28. {
  29. reply_with($info, { button_menu => 1 }, _("Для получения списка доступных команд введите /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_with($info, { button_menu => 1 }, 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_with($info, { button_menu => 1 }, 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. task_needs_descr => async sub
  83. {
  84. my ($fsa, $descr, $info) = @_;
  85. my $uid = $fsa->note("[uid]");
  86. $fsa->note(task_text => $descr);
  87. $fsa->note(if_edited => "$info->{msgid}/task_text");
  88. reply_with($info, {
  89. inline_menu => [[
  90. { text=>_("Отправить"), callback_data=>"\x00/task_post" },
  91. { text=>_("Отменить"), callback_data=>"\x00/task_cancel" },
  92. ]]
  93. },
  94. _("Перед отправкой заявки вы можете ее отредактировать средствами Телеграм")
  95. );
  96. return "command";
  97. },
  98. task_needs_comment => async sub
  99. {
  100. my ($fsa, $comment, $info) = @_;
  101. my $uid = $fsa->note("[uid]");
  102. my $task_id = $fsa->note("task_id");
  103. my $params = {
  104. text => $comment,
  105. "for-client" => $uid,
  106. };
  107. $fsa->state("command"); # на случай die
  108. my $res = await $client->post_p("task", "/task/$task_id/comment", $params);
  109. reply($info, "Ваш ответ добавлен");
  110. return "command";
  111. },
  112. ###################################
  113. needs_input => async sub
  114. {
  115. my ($fsa, $input, $info) = @_;
  116. my $uid = $fsa->note("[uid]");
  117. $fsa->state("command"); # на случай die
  118. my $action = $fsa->note("input_action");
  119. if ($input eq "\x00/cancel_input")
  120. {
  121. reply_with($info, {button_menu=>1}, ucfirst($lex_actions->{$action}->{name}) . " " . $lex_actions->{$action}->{canceled});
  122. return "command";
  123. }
  124. my $var = $fsa->note("input_var");
  125. my $sub = refpath("verify_$var") || sub { undef };
  126. my $error = $sub->($fsa, $input, $info);
  127. $error = await $error if ref $error && $error->can("then");
  128. if ($error)
  129. {
  130. reply_with($info, {
  131. inline_menu => [[
  132. { text=>_("Отмена"), callback_data=>"\x00/cancel_input" },
  133. ]]
  134. }, $error, "", _("Введите заново ") . $lex_vars->{$var} . _(" или нажмите Отмена, чтобы прервать ") . $lex_actions->{$action}->{name});
  135. return "needs_input";
  136. }
  137. else
  138. {
  139. $fsa->note($var => $input);
  140. $sub = refpath("use_$var") || sub { "command" };
  141. my $state = await $sub->($fsa, $input, $info);
  142. return $state;
  143. }
  144. },
  145. };
  146. sub greet($whom)
  147. {
  148. my ($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime(time);
  149. my $greet = $hour>=4 && $hour<=10 ? _("Доброе утро") : $hour>10 && $hour<19 ? _("Добрый день") : _("Добрый вечер");
  150. "$greet, $whom";
  151. }
  152. ######################
  153. sub make_key($id)
  154. {
  155. return "abonbot-$id";
  156. }
  157. sub save_fsa($fsa, $chatid)
  158. {
  159. my $key = make_key($chatid);
  160. say "saving state ", $fsa->state, Dumper $fsa->notes;
  161. my $notes = $fsa->notes;
  162. utf8::encode($notes->{$_}) for keys %$notes;
  163. $redis->del($key);
  164. $redis->hmset($key, %$notes);
  165. $redis->expire($key, 3600*24);
  166. }
  167. sub make_fsa($chatid, $from)
  168. {
  169. my $id = make_key($chatid);
  170. my $notes = { $redis->hgetall($id) };
  171. if (keys %$notes)
  172. {
  173. utf8::decode($notes->{$_}) for keys %$notes;
  174. return fsa->new($rules, $notes->{"[state]"}, $notes);
  175. }
  176. else
  177. {
  178. return fsa->new($rules, "logged_out", {});
  179. }
  180. }
  181. 1;