Annotation of ChivanetConvoBot/convobot, revision 1.6
1.1 snw 1: #!/usr/bin/env perl
2:
3: #
4: # ChivaNet Conversation Bot
5: # Copyright (C) 2025 Coherent Logic Development LLC
6: #
7: # Author: Serena Willis <snw@coherent-logic.com>
8: #
9: # Licensed AGPL-3.0
10: #
1.2 snw 11: # $Log: convobot,v $
1.6 ! snw 12: # Revision 1.5 2025/02/03 18:14:15 snw
! 13: # Further work on bot
! 14: #
1.5 snw 15: # Revision 1.4 2025/02/03 17:31:28 snw
16: # Further MySQL work
17: #
1.4 snw 18: # Revision 1.3 2025/02/03 15:38:12 snw
19: # Begin SQL work
20: #
1.3 snw 21: # Revision 1.2 2025/02/03 04:28:34 snw
22: # Fix syntax message
23: #
1.2 snw 24: # Revision 1.1.1.1 2025/02/03 04:22:49 snw
25: # Initial Commit
26: #
1.1 snw 27: #
28: #
29:
30: use Net::OSCAR;
31: use Getopt::Long;
32: use Data::Dumper;
33: use HTML::Strip;
1.3 snw 34: use DBI;
1.1 snw 35:
36: my $idlemax = 1800;
37: my $botsn = '';
38: my $botsrv = '';
39: my $botpw = '';
40: my $rasurl = '';
1.3 snw 41:
42: my $dbhost = '';
43: my $dbname = '';
44: my $dbusername = '';
45: my $dbpw = '';
46: my $dbconn = '';
1.4 snw 47: my $autogreet = 'off';
1.3 snw 48:
1.1 snw 49: my $chatroom = '';
50: my $online = 0;
51: my $chat_idle_seconds = 0;
52: my $last_chat_received = time();
53: my $start_time = time();
1.5 snw 54: my $dbh = '';
55: my $dsn = '';
1.1 snw 56:
57: my @congregants = ();
58:
1.4 snw 59: $oscar = Net::OSCAR->new();
60:
61: sub get_seen_status {
62: my($sn, $chat) = @_;
1.1 snw 63:
1.4 snw 64: my $sth = $dbh->prepare("SELECT * FROM seen WHERE aim_server=? AND aim_sn=? AND aim_chatroom=? AND sn=?");
65: $sth->execute($botsrv, $botsn, $chatroom, $sn);
1.1 snw 66:
1.4 snw 67: if($sth->rows > 0) {
68: my $hashref = $sth->fetchrow_hashref();
69: $chat->chat_send("I last saw <strong>$sn</strong> on $hashref->{seen_time}.");
70: }
71: else {
72: $chat->chat_send("I have never seen <strong>$sn</strong>.");
73: }
74:
75: }
1.3 snw 76:
1.4 snw 77: sub update_seen_status {
1.3 snw 78: my($sn) = @_;
79:
1.4 snw 80: my $del = $dbh->prepare("DELETE FROM seen WHERE aim_server=? AND aim_sn=? AND aim_chatroom=? AND sn=?");
1.5 snw 81: $del->execute($botsrv, $botsn, $chatroom, $sn) or die "error DBI->errstr()";
1.3 snw 82:
1.5 snw 83: my $ins = $dbh->prepare("INSERT INTO seen (aim_server, aim_sn, aim_chatroom, sn, seen_time) VALUES (?, ?, ?, ?, ?)");
84: my $seentime = localtime();
85: $ins->execute($botsrv, $botsn, $chatroom, $sn, $seentime) or die "error DBI->errstr()";
1.3 snw 86: }
1.1 snw 87:
88: sub signon_done {
89: print "[OK]\n";
1.5 snw 90: print "convobot: joining $chatroom...";
91: $oscar->chat_join($chatroom, 5);
1.4 snw 92: print "[OK]\n";
1.1 snw 93: $online = 1;
94: }
95:
1.4 snw 96: sub oscar_error {
97: my($oscar, $connection, $error, $description, $fatal) = @_;
98:
99: if($fatal != 0) {
1.5 snw 100: die "\nconvobot: fatal OSCAR error: $description\n";
1.4 snw 101: }
102: else {
1.5 snw 103: print "\nconvobot: recoverable OSCAR error: $description\n";
1.4 snw 104: }
105:
106: }
107:
1.1 snw 108: sub chat_joined {
109: my($oscar, $chatname, $chat) = @_;
110:
1.5 snw 111: print "bot: chat joined [$chatname]\n";
112:
1.1 snw 113: $room = $chat;
114: bless $room, "Net::OSCAR::Connection::Chat";
1.5 snw 115:
116: print "convobot: connecting to database $dbname\@$dbhost...";
117:
118: $dsn = "DBI:mysql:database=$dbname;host=$dbhost;port=3306;mysql_connect_timeout=5;";
119: $dbh = DBI->connect($dsn, $dbusername, $dbpw, {RaiseError => 1});
120: die "convobot: failed to connect to MySQL database: DBI->errstr()" unless $dbh;
121:
122: print "[OK]\n";
123:
124: $oscar->set_callback_chat_buddy_in(\&chat_buddy_in);
125: $oscar->set_callback_chat_buddy_out(\&chat_buddy_out);
1.1 snw 126: }
127:
128: sub chat_buddy_in {
129: my ($oscar, $who, $chat, $buddy) = @_;
130:
1.4 snw 131: update_seen_status($who);
1.1 snw 132:
133: if($who ne $botsn) {
134: push(@congregants, $who);
1.5 snw 135: print "convobot: [$who] has joined\n";
1.1 snw 136: }
137: else {
1.5 snw 138: print "convobot: [$who] has joined (ignoring bot)\n";
1.1 snw 139: }
140:
1.4 snw 141:
142: if($autogreet eq "on") {
143: if(time() - $start_time > 2) {
144: my @phrases = ('Welcome to [room], [user]! :-)',
145: 'How\'s it going, [user]?',
146: 'Hey [user]! Bring any snacks?',
147: 'Heya [user]! Hope your day is going well!',
148: 'Ooo, [user] has joined [room]! Now the party can start!');
149:
150: my $phrase = $phrases[rand @phrases];
151: $phrase =~ s/\[user\]/$who/g;
152: $phrase =~ s/\[room\]/$chatroom/g;
153: my $phrasefix = "<div id=convobot></div>$phrase";
154: $chat->chat_send($phrasefix);
155: }
156: else {
1.5 snw 157: print "convobot: not sending greeting for 2 seconds after startup\n";
1.4 snw 158: }
1.1 snw 159: }
160: }
161:
162: sub chat_buddy_out {
163: my ($oscar, $who, $chat) = @_;
164: my $index = 0;
165:
166: $index++ until $congregants[$index] eq $who;
167: splice(@congregants, $index, 1);
168:
1.5 snw 169: print "convobot: $who has left\n";
1.1 snw 170: }
171:
172: sub chat_im_in {
173: my($oscar, $who, $chat, $message) = @_;
174:
175: my $hs = HTML::Strip->new();
176: my $rawcmd = $hs->parse($message);
177: my @cmd = split(' ', $rawcmd);
178:
1.4 snw 179: update_seen_status($who);
1.5 snw 180:
181: if($who ne $botsn) {
182: if($cmd[0] eq "!seen") {
183: if(exists($cmd[1])) {
1.6 ! snw 184: my @sna = @cmd[1..$#cmd];
! 185: my $ssn = join(' ', @sna);
! 186: get_seen_status($ssn, $chat);
1.5 snw 187: }
188: else {
189: $chat->chat_send("Syntax: !seen <em>screenname</em>");
190: }
191: }
192: elsif($cmd[0] eq "!speak") {
193: send_idle_message();
194: }
195: elsif($cmd[0] eq "!quote") {
196: my $fortune = `/usr/games/fortune`;
197: $room->chat_send($fortune);
198: }
1.6 ! snw 199: elsif($cmd[0] eq "!invite") {
! 200: if(exists($cmd[1])) {
! 201: my @sna = @cmd[1..$#cmd];
! 202: my $ssn = join(' ', @sna);
! 203: $chat->invite($ssn, "Please join us in $chatroom! <br><em>Requested by $who</em>");
! 204: }
! 205: }
1.5 snw 206: elsif($cmd[0] eq "!help") {
207: $room->chat_send("You can enter the following commands:");
208: $room->chat_send(" <code>!seen <em>screenname</em></code> (find out when <em>screenname</em> was last in the chat)");
1.6 ! snw 209: $room->chat_send(" <code>!invite <em>screenname</em></code> (invite <em>screenname</em> to the chat)");
1.5 snw 210: $room->chat_send(" <code>!speak</code> (send a random message)");
211: $room->chat_send(" <code>!quote</code> (send a quote)");
212: }
1.1 snw 213: }
1.5 snw 214:
215:
1.1 snw 216: $last_chat_received = time();
217:
1.5 snw 218: print "convobot: chat received from $who; resetting idle counter\n";
1.1 snw 219:
220: }
221:
222: sub send_idle_message {
223:
224: my @phrases = ('Hey [user]! How are you today?',
225: 'I think [user] should bring us some pizza!',
226: 'What\'s everyone up to here?',
227: 'My, what a beautiful day for a chat here in [room]!',
228: '[user] always has the most interesting things to say.',
229: 'Remember that time [user] was talking here in [room]?',
230: 'What do all you [room] chatters think about pie?',
231: '[room] seems dead :\'(. That makes me sad! Maybe [user] has something interesting to say?');
232:
233: my $congregant = $congregants[rand @congregants];
234: my $phrase = $phrases[rand @phrases];
235: $phrase =~ s/\[user\]/$congregant/g;
236: $phrase =~ s/\[room\]/$chatroom/g;
237: my $phrasefix = "<div id=convobot></div>$phrase";
238:
239: if(ref($room) eq "Net::OSCAR::Connection::Chat") {
240: $room->chat_send($phrasefix);
241: $last_chat_received = time();
242: }
1.5 snw 243:
1.1 snw 244: }
245:
246: $oscar->set_callback_signon_done(\&signon_done);
247: $oscar->set_callback_chat_joined(\&chat_joined);
248: $oscar->set_callback_chat_im_in(\&chat_im_in);
1.4 snw 249: $oscar->set_callback_error(\&oscar_error);
1.1 snw 250:
251: print "ChivaNet Conversation Bot v0.0.1\n";
252: print " Copyright (C) 2025 Coherent Logic Development LLC\n\n";
253:
1.4 snw 254: GetOptions("aimsn=s" => \$botsn,
255: "aimhost=s" => \$botsrv,
256: "aimpw=s" => \$botpw,
257: "idlemax=s" => \$idlemax,
258: "chatroom=s" => \$chatroom,
259: "dbhost=s" => \$dbhost,
260: "dbname=s" => \$dbname,
261: "dbusername=s" => \$dbusername,
262: "dbpw=s" => \$dbpw,
263: "autogreet=s" => \$autogreet)
264: or die("error in command line arguments");
265:
266: %signon = (
267: screenname => $botsn,
268: password => $botpw,
269: host => $botsrv,
270: );
271:
272: print "AIM Server: $botsrv\n";
273: print "AIM Screen Name: $botsn\n";
274: print "Chat Room: $chatroom\n";
275: print "DB Host: $dbhost\n";
276: print "DB Name: $dbname\n";
277: print "DB Username: $dbusername\n";
278: print "Idle before ping: $idlemax\n";
279: print "Auto-Greet: $autogreet\n\n";
280:
281:
1.5 snw 282: print "convobot: attempting to sign in...";
1.1 snw 283: $oscar->signon(%signon);
284:
285: while(1) {
286: $oscar->do_one_loop();
287: $chat_idle_seconds = time() - $last_chat_received;
288:
289: if($chat_idle_seconds > $idlemax) {
290: send_idle_message();
291: }
292: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>