djinn.pl 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. #!/usr/bin/perl
  2. # Бот-исполнитель запросов техподдержки
  3. # Ю. Жиловец, 11 января 2023 года
  4. use Modern::Perl;
  5. use utf8;
  6. use EV;
  7. use AnyEvent;
  8. use Mojolicious::Lite;
  9. use Mojo::UserAgent;
  10. use Data::Dumper;
  11. use Promises qw/deferred collect/;
  12. use Mojo::JSON qw/j/;
  13. use NetAddr::IP;
  14. use HTML::Restrict;
  15. use FindBin qw/$Bin/;
  16. use lib "$Bin/lib";
  17. use lib "$Bin/modules";
  18. use darsan_auth;
  19. use darsan_client;
  20. use commands;
  21. my $NAME = "djinn";
  22. my $confdir = app->home.'/config/'.app->mode;
  23. use rabbit_async_rec;
  24. plugin yaml_config => {
  25. file => "$confdir/$NAME.cfg",
  26. stash_key => 'config',
  27. };
  28. our $config = app->config;
  29. app->secrets(["Marsz, Marsz, Dabrowski"]);
  30. my $log = new Mojo::Log;
  31. # https://core.telegram.org/bots/api#formatting-options
  32. my $html_strip = HTML::Restrict->new(rules => {
  33. b => [],
  34. strong => [],
  35. i => [],
  36. em => [],
  37. u => [],
  38. ins => [],
  39. s => [],
  40. strike => [],
  41. del => [],
  42. a => [qw/href/],
  43. code => [qw/class/],
  44. pre => [],
  45. });
  46. my $auth = darsan_auth->as_server($config->{darsan}->{auth}, "system", "$confdir/system.private");
  47. our $client = darsan_client->new($auth, $config->{darsan}->{servers});
  48. my $term;
  49. my $int;
  50. my $hup;
  51. Mojo::IOLoop->next_tick(sub
  52. {
  53. $term = AnyEvent->signal(signal => "TERM", cb => \&terminate);
  54. $int = AnyEvent->signal(signal => "INT", cb => \&terminate);
  55. $hup = AnyEvent->signal(signal => "HUP", cb => \&terminate);
  56. });
  57. ##########################
  58. my $ua = new Mojo::UserAgent;
  59. $ua->max_redirects(5);
  60. ##########################
  61. =cut
  62. hook before_dispatch => sub
  63. {
  64. my $c = shift;
  65. say $c->req->to_string;
  66. };
  67. hook after_dispatch => sub
  68. {
  69. my $c = shift;
  70. say $c->res->to_string;
  71. };
  72. =cut
  73. ###############################################
  74. get "/health" => sub
  75. {
  76. shift->render(text => "Djinn OK");
  77. };
  78. post "/:token" => sub
  79. {
  80. my $c = shift;
  81. $c->render(text=>"ok");
  82. unless ($c->param("token") eq $config->{token})
  83. {
  84. return $c->render(status=>401, text=>"Request from unknown URL");
  85. }
  86. my $body = j($c->req->body);
  87. my $m = $body->{message} || $body->{edited_message};
  88. my $chatid = $m->{chat}->{id};
  89. if ($m->{chat}->{type} ne "supergroup")
  90. {
  91. return notify($chatid, "Общение с ботом возможно только в чате");
  92. }
  93. my $from = $m->{from};
  94. my $cmd = $m->{text};
  95. my $msgid = $m->{message_id};
  96. return unless substr($cmd, 0, 1) eq "/"; # Бот не должен мешать общению, даже если его добавили админом
  97. do_command($cmd, $chatid, {msgid=>$msgid, from=>$from});
  98. };
  99. ##################################
  100. sub terminate
  101. {
  102. request("setWebhook", {url=>""})->then(sub
  103. {
  104. exit(0);
  105. })->catch(sub
  106. {
  107. $log->error(Dumper @_);
  108. });
  109. }
  110. sub request
  111. {
  112. my $action = shift;
  113. my $params = shift;
  114. my $deferred = deferred;
  115. $ua->post("https://api.telegram.org/bot$config->{token}/$action" => form => $params => sub
  116. {
  117. my ($ua, $tx) = @_;
  118. my $resp = $tx->result;
  119. if ($resp->is_error)
  120. {
  121. my $err = {};
  122. $err->{code} = $resp->code;
  123. $err->{url} = $tx->req->url->to_string;
  124. $err->{body} = $resp->body;
  125. $err->{body} = j($err->{body}) if $resp->headers->content_type eq "application/json";
  126. $deferred->reject($err);
  127. }
  128. else
  129. {
  130. my $body = $resp->body;
  131. $body = j($body) if $resp->headers->content_type eq "application/json";
  132. $deferred->resolve($body);
  133. }
  134. });
  135. return $deferred->promise;
  136. }
  137. sub notify
  138. {
  139. my $chatid = shift;
  140. my $message = shift;
  141. my $rest = shift || {};
  142. my $params = {
  143. chat_id => $chatid,
  144. text => $message,
  145. disable_web_page_preview => 1,
  146. };
  147. $params->{parse_mode} ||= "HTML";
  148. $params->{reply_to_message_id} = $rest->{reply_to} if $rest->{reply_to};
  149. $params->{disable_notification} = 1 if $rest->{silent};
  150. my $disable_error_handler = delete $params->{disable_error_handler};
  151. if ($params->{parse_mode} eq "HTML")
  152. {
  153. $params->{text} = $html_strip->process($params->{text});
  154. }
  155. my $promise = request("sendMessage", $params);
  156. unless ($disable_error_handler)
  157. {
  158. $promise = $promise->catch(sub
  159. {
  160. $log->error(Dumper $params,@_);
  161. });
  162. }
  163. return $promise;
  164. }
  165. sub do_command
  166. {
  167. my $cmd = shift;
  168. my $chatid = shift;
  169. my $rest = shift;
  170. local($Data::Dumper::Terse) = 1;
  171. my ($c,@args) = split(/\s+/,$cmd);
  172. $c =~ s|^/||;
  173. $c =~ s/\@MolDjinnBot$//;
  174. my $sub = reference("command_$c");
  175. unless ($sub)
  176. {
  177. return notify($chatid, "Неизвестная команда. Введите <b>/help</b>, чтобы увидеть список команд", $rest);
  178. }
  179. eval {
  180. $sub->($c, \@args, $chatid, $rest);
  181. };
  182. if ($@)
  183. {
  184. my $msg = ref $@ eq "HASH" ? Dumper($@) : $@;
  185. $log->error("$cmd from $chatid: $msg");
  186. notify($chatid, "Ошибка при выполнении команды $cmd: $msg");
  187. return;
  188. }
  189. }
  190. sub refpath
  191. {
  192. my $name = shift;
  193. $name =~ tr/.-/_/;
  194. $name =~ s|/|::|g;
  195. return reference($name);
  196. }
  197. sub reference
  198. {
  199. my $name = shift;
  200. return exists(&{$name}) ? \&{$name} : undef;
  201. }
  202. ######################################
  203. $log->info("Started (".app->mode.")");
  204. request("setWebhook",{url=>""})->then(sub
  205. {
  206. $log->info("Webhook to $config->{webhook}");
  207. return request("setWebhook", {url=>"$config->{webhook}/$config->{token}"});
  208. })->catch(sub
  209. {
  210. $log->error(Dumper @_);
  211. });
  212. app->start;