-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathhvzchatclient.pl
executable file
·261 lines (226 loc) · 9.43 KB
/
hvzchatclient.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
#!/usr/bin/env perl
use strict;
use warnings;
use feature ':5.20';
use utf8;
#BEGIN { use File::Spec::Functions qw/rel2abs/; use File::Basename qw/dirname/; }
#use lib dirname rel2abs $0;
## no critic (SubroutinePrototypes)
#use Carp::Always;
use Algorithm::Diff qw/diff/;
use AnyEvent;
use AnyEvent::Strict;
use AnyEvent::Util qw/run_cmd/;
use Class::Struct;
use Data::Dump qw/pp dd/;
use Date::Format;
use DateTime;
use DateTime::Duration;
use DateTime::Format::Strptime;
use File::Which qw/which/;
use HTML::TreeBuilder 5 -weak;
use IO::All;
use List::AllUtils qw/max min notall uniq first shuffle/;
use LWP::UserAgent;
use Term::ReadKey;
use Text::Wrap qw/wrap/;
use URI;
use WWW::Mechanize;
$Text::Wrap::columns = main::min ($ENV{COLUMNS} || 80, 140);
$main::start_time = time;
$main::i_am_dead = 0;
my $strp = DateTime::Format::Strptime->new(
pattern=>"%Y/%m/%d %R",
time_zone=>'America/New_York',
on_error=>'croak'
);
my $quit = AE::cv;
sub concat(@) { map {@$_} @_ }
{
package WWW::Mechanize::GaTechCAS;
use parent qw/WWW::Mechanize/;
sub new {
my $class = shift;
my $self = $class->SUPER::new();
$self->timeout(10);
$self->{CAS} = {username=>"",password=>""};
$self->{hvz_data}->{killboard}->{$_} = [] for qw/human zombie/;
$self->{hvz_data}->{chatlines}->{$_} = [] for qw/all hum zomb/;
$self->{data_file} = $0 =~ s/\.pl$/.data.pl/r;
if (-f $self->{data_file}) {
$self->{hvz_data} = do $self->{data_file};
}
return $self;
}
## no critic (SubroutinePrototypes)
sub ensure_hvz_session($) {
my $self = shift;
$self->get('https://hvz.gatech.edu/killboard/');
if ('login.gatech.edu' eq $self->uri->host) {
$self->submit_form(with_fields=>$self->{CAS});
}
return 'hvz.gatech.edu' eq $self->uri->host;
}
sub stdio_authenticate($) {
my $self = shift;
my $failed_yet = 0;
until ((length $self->{CAS}->{username}) and (length $self->{CAS}->{password}) and $self->ensure_hvz_session) {
print "Authentication failure; please try again.\n" if $failed_yet;
print STDERR "Username for login.gatech.edu: ";
chomp($self->{CAS}->{username} = <STDIN>);
main::ReadMode 'noecho';
print STDERR "Password for login.gatech.edu: ";
chomp($self->{CAS}->{password} = <STDIN>);
main::ReadMode 'restore';
print STDERR "\n";
$failed_yet = 1;
}
}
sub get_killboard($) {
my $self = shift;
$self->ensure_hvz_session;
$self->get('/killboard');
my $tree = HTML::TreeBuilder->new;
$tree->parse_content($self->content);
my $factions = {}; $factions->{$_} = [] for qw/human zombie/;
for my $faction (keys %$factions) {
my @killboard = $tree->look_down(id=>"$faction-killboard");
$factions->{$faction} = [map {$_->as_text} $killboard[0]->look_down(_tag=>'a',href=>qr/\?gtname=/)];
$factions->{$faction} = ['The OZ'] unless scalar @{$factions->{$faction}};
}
my @deaths = map { $_->[2] } grep {$_->[0] eq '+'} main::concat(main::diff($self->{hvz_data}->{killboard}->{zombie}, $factions->{zombie}));
print "$_ is dead.\n" for @deaths;
@deaths = () unless scalar @{$self->{hvz_data}->{killboard}->{zombie}};
for my $nom (@deaths) {
my $exclamation = main::first {1} main::shuffle ("Consarnit.", "Well, drat.", "Argh.", "Dear me.", "Eek!", "Well, I'll be.", "Oh, scrap.", "Hunh.", "Well whaddaya know?", "Are you kidding me?");
my $qualifier = main::first {1} main::shuffle ("Looks like", "I think that", "Intel suggests - ah, nvm that. We are 100% gorram positive that", "Seems that", "It appears as though", "The killboard says that", "Reports indicate that");
my $verbphrase = main::first {1} main::shuffle ("$nom bit the dust", "$nom kicked the bucket", "$nom passed on to the undeath", "$nom died", "$nom was turned", "$nom was nommed", "someone killed $nom", "$nom became an ex-human", "$nom has no longer been with us, starting", "the zeds got $nom", "we lost $nom", "$nom ceased to be human");
main::_groupme_post("hum", "$exclamation $qualifier $verbphrase within the past 3 hours.");
}
$self->{hvz_data}->{killboard} = $factions;
main::_groupme_post("hum", "There are now ".(scalar @{$self->{hvz_data}->{killboard}->{zombie}})." zombies on the killboard.") if @deaths;
$self->back;
return $factions;
}
sub longest_name_length_on_killboard($) {
my $self = shift;
return main::max map { length $_ } main::concat values %{$self->get_killboard};
}
sub whoami($) {
my $self = shift;
return $self->{CAS}->{username};
}
sub make_chattracker($) {
my $self = shift;
my $longest_name_length = $self->longest_name_length_on_killboard;
main::_groupme_post("hum", "Bot reporting for duty.");
ChatLine->print_all($self, $longest_name_length, undef, main::concat values %{$self->{hvz_data}->{chatlines}});
return sub {
use sort 'stable';
my @outlines = ();
$self->ensure_hvz_session or die "auth problem";
$longest_name_length = $self->longest_name_length_on_killboard;
$self->get('/chat/');
for my $faction (sort keys %{$self->{hvz_data}->{chatlines}}) {
$self->post(URI->new_abs('_update.php', $self->uri), {aud=>$faction});
my $tree = HTML::TreeBuilder->new;
$tree->parse_content($self->content);
map { $_->replace_with_content } $tree->look_down(_tag=>'a', href=>qr/\?gtname=/);
my @additions = map { $_->[2] } grep {$_->[0] eq '+'} main::concat(main::diff($self->{hvz_data}->{chatlines}->{$faction}, [map {ChatLine->from_tr($faction, $_)} $tree->look_down(_tag=>'tr',class=>qr/chat_line/)], sub { defined $_ ? $_->hash : "" } ));
push @{$self->{hvz_data}->{chatlines}->{$faction}}, @additions;
push @outlines, @additions;
}
if (@outlines) {
print "\a";
main::pp($self->{hvz_data}) > main::io($self->{data_file});
ChatLine->print_all($self, $longest_name_length, \&main::run_cmd, @outlines);
}
};
}
}
$main::groupme_bots = do ($0 =~ s/hvzchatclient\.pl$/bot_ids.pl/r);
sub _groupme_post($$);
sub _groupme_post($$) {
my ($faction, $text) = @_;
return if $main::groupme_bots->{$faction} =~ /^</;
my $uri = URI->new('https://api.groupme.com/v3/bots/post');
$uri->query_form({bot_id=>$main::groupme_bots->{$faction},text=>$text});
my $poster = LWP::UserAgent->new;
my $response = $poster->post($uri);
if ($response->code eq "400") { sleep 5; return _groupme_post($faction, $text); }
if (!$response->is_success) { die $response->status_line; }
}
{
package ChatLine;
use Class::Struct faction=>'$',sender=>'$',sender_is_admin=>'$',timestamp=>'$',message=>'$';
sub from_tr($$$) {
my $class = shift;
my ($faction, $tr) = @_;
my $ret = $class->new(
faction=>$faction,
sender=>($tr->content_list)[0]->as_trimmed_text,
sender_is_admin=>($tr->attr('class') =~ /admin_line/ || 0),
timestamp=>([$tr->content_list]->[1]->as_trimmed_text =~ s/ ([0-9][^0-9])/ 0$1/r),
message=>[$tr->content_list]->[2]->as_trimmed_text,
);
return $ret;
}
sub is_old($) {
my $self = shift;
# "2016/" is a kludge to avoid error "There is no use providing a month without providing a year."
return (DateTime::Duration->compare(DateTime->now->subtract_datetime($strp->parse_datetime("2016/". $self->timestamp)),DateTime::Duration->new(minutes=>2)) == 1)
}
sub format($$) {
my ($self, $longest_name_length) = @_;
my $header = sprintf("[%s] %-${longest_name_length}s -> %s: ", $self->timestamp, $self->sender, $self->faction);
$header = ($self->is_old ? " " : "!") . $header;
my $subsequent_tab = " " x (length $header);
return main::wrap($header, $subsequent_tab, $self->message) . "\n";
}
sub hash($) { return 0 if not $_; return join "\t", @$_; }
sub alert ($$) {
my $self = shift;
my $run_cmd = shift;
return unless defined main::which 'notify-send';
$run_cmd->([qw/notify-send -a/,'HvZ Chat','--',$self->sender.' -> '.$self->faction,$self->message]);
}
sub groupme_post ($) {
my $self = shift;
return if $self->faction eq "all" and not $self->sender_is_admin;
#print $timestamp; print "\n";
my $message = sprintf("[%s] %s",$self->sender,$self->message);
$message = sprintf("[%s] ",$self->timestamp) . $message if $self->is_old;
main::_groupme_post($self->faction,$message);
}
sub print_all($$$&@) {
my ($class, $mech, $longest_name_length, $should_alert, @lines) = @_;
#open my $null, '>', '/dev/null';
my @oldies = grep { $_->is_old } @lines;
if (main::notall { !(defined $_ ) or defined $main::groupme_bots->{$_->faction} } @lines) {
main::_groupme_post("human","My creator has died. RIP. My genes are at http://gatech.edu/mmirate/gatech_hvz_mechanized if anyone wants to re-clone me. Signing off.");
$main::i_am_dead = 1;
$quit->send;
}
#if (defined $should_alert and scalar @oldies) {
# main::_groupme_post("all","Whoops, looks like I slept through some chat posts. Here we go. Oldies are timestamped.");
#}
print map { $_->alert($should_alert) if defined $should_alert; $_->groupme_post() if defined $should_alert; $_->format($longest_name_length) } sort { $a->timestamp cmp $b->timestamp } @lines;
}
}
my $mech = WWW::Mechanize::GaTechCAS->new();
$mech->stdio_authenticate;
my $tracker = $mech->make_chattracker;
my $timer_repeated = AE::timer 0, 10, $tracker;
my $stdin_ready = AE::io *STDIN, 0, sub {
chomp( my $input = <STDIN> );
$input =~ /(all|hum|zomb|ALL|HUM|ZOMB): *(.+)/ or die 'usage: (*all*|*hum*|*zomb*)*:* message';
if ($1 ne lc $1) {
_groupme_post(lc $1, $2);
} else {
$mech->post(URI->new_abs('_post.php', $mech->uri), {aud=>$1, content=>$2});
}
};
$quit->recv;
END {
main::_groupme_post("hum", '@Milo Mirate Help! I\'ve fallen and I can\'t get up!') unless $main::i_am_dead || ( time() - $main::start_time < 120);
}