djinn.pl 5.4 KB

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