1 ###RUN_PERL: #!/usr/bin/perl
4 # chat.pl is generated from chat.1.pl.
6 # The coincidence interface
8 # Copyright (C) 2016, 2017, 2023, 2024 Balthasar SzczepaĆski
10 # This program is free software: you can redistribute it and/or modify
11 # it under the terms of the GNU Affero General Public License as
12 # published by the Free Software Foundation, either version 3 of the
13 # License, or (at your option) any later version.
15 # This program is distributed in the hope that it will be useful,
16 # but WITHOUT ANY WARRANTY; without even the implied warranty of
17 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 # GNU Affero General Public License for more details.
20 # You should have received a copy of the GNU Affero General Public License
21 # along with this program. If not, see <http://www.gnu.org/licenses/>.
25 ###PERL_LIB: use lib /botm/lib/bsta
26 # use Encode::Locale ('decode_argv');
27 use Encode ('encode', 'decode');
29 ###PERL_LIB: use lib /botm/lib/bsta
31 'read_data_file', 'write_data_file',
33 'url_query_decode', 'url_query_encode',
35 'html_entity_encode_dec',
39 'STATE', 'CHAT_STATE', 'CHAT_ACTION',
40 'fail_method', 'fail_content_type',
41 'get_remote_addr', 'get_id', 'get_password',
42 'print_html_start', 'print_html_end',
43 'print_html_head_start', 'print_html_head_end',
44 'print_html_body_start', 'print_html_body_end',
49 ###PERL_CGI_PATH: CGI_PATH = /bsta/
50 ###PERL_CGI_COIN_PATH: CGI_COIN_PATH = /bsta/coin
52 ###PERL_DATA_CHAT_PATH: DATA_CHAT_PATH = /botm/data/bsta/chat
53 ###PERL_DATA_COIN_PATH: DATA_COIN_PATH = /botm/data/bsta/coincidence
54 ###PERL_DATA_SETTINGS_PATH: DATA_SETTINGS_PATH = /botm/data/bsta/settings
55 ###PERL_DATA_STATE_PATH: DATA_STATE_PATH = /botm/data/bsta/state
57 ###PERL_WEBSITE_NAME: WEBSITE_NAME = Bicycles on the Moon
59 binmode STDIN, ':encoding(UTF-8)';
60 binmode STDOUT, ':encoding(UTF-8)';
61 binmode STDERR, ':encoding(UTF-8)';
79 my $action = CHAT_ACTION->{'none'};
90 delete @ENV{qw(IFS CDPATH ENV BASH_ENV)};
91 ###PERL_SET_PATH: $ENV{'PATH'} = /usr/local/bin:/usr/bin:/bin;
93 if ($ENV{'REQUEST_METHOD'} =~ /^(HEAD|GET|POST)$/) {
97 exit fail_method($ENV{'REQUEST_METHOD'}, 'GET, POST, HEAD');
100 %http = read_header_env(\%ENV);
101 %cgi = url_query_decode($ENV{'QUERY_STRING'});
103 if ($method eq 'POST') {
104 if ($http{'content-type'} eq 'application/x-www-form-urlencoded') {
105 my %cgi_post = url_query_decode( <STDIN> );
106 %cgi = merge_settings(\%cgi, \%cgi_post);
108 # multipart not supported
110 exit fail_content_type($method, $http{'content-type'});
114 $IP = get_remote_addr();
115 $page = get_id(\%cgi, -1);
116 $password = get_password(\%cgi);
118 %coin = read_data_file(DATA_COIN_PATH());
119 %settings = read_data_file(DATA_SETTINGS_PATH());
120 %state = read_data_file(DATA_STATE_PATH());
122 $password_ok = ($password eq $settings{'password'});
124 if ($cgi{'words'} ne '') {
125 $words = $cgi{'words'};
127 if ($password_ok && ($cgi{'username'} ne '')) {
128 $username = $cgi{'username'};
130 foreach my $action_id ('join', 'leave', 'nopost', 'file') {
131 if ($cgi{$action_id} ne '') {
132 $action = CHAT_ACTION->{$action_id};
139 if (open_encoded($fh, "+<", DATA_CHAT_PATH())) {
141 %chat = read_data_file($fh);
143 $chat_state = int($chat{'state'});
144 $chat_id = int($chat{'id'});
147 if (($action == CHAT_ACTION->{'none'}) && ($words ne '')) {
148 if (($chat_state < CHAT_STATE->{'ready'}) && !$password_ok) {
149 $message = 'Not connected.';
152 if ($words !~ /[\r\n]/) {
153 if ($username =~ /^[A-Za-z]*$/) {
154 $chat{'content'} .= $username.': '.$words."\n";
155 if ($chat_state < CHAT_STATE->{'active'}) {
156 $chat_state = CHAT_STATE->{'active'};
157 $chat{'state'} = $chat_state;
159 write_data_file($fh, \%chat);
162 $message = 'Invalid username.';
166 $message = 'Invalid text.';
171 elsif ($action == CHAT_ACTION->{'join'}) {
172 if (($chat_state > CHAT_STATE->{'disconnected'}) && !$password_ok) {
173 $message = 'Already connected.';
176 if ($username =~ /^[A-Za-z]*$/) {
177 if ($password_ok || $words eq $coin{'server'}) {
178 $chat{'content'} .= 'join@'.$username.': '.$words."\n";
179 if ($chat_state < CHAT_STATE->{'ready'}) {
180 $chat_state = CHAT_STATE->{'ready'};
181 $chat{'state'} = $chat_state;
183 write_data_file($fh, \%chat);
185 elsif ($words eq '') {
186 $message = 'Server ID missing.';
188 elsif ($words !~ /^[0-9]+$/) {
189 $message = 'Invalid server ID.';
192 $message = 'No active Coincidence server with this ID.';
196 $message = 'Invalid username.';
201 elsif ($action == CHAT_ACTION->{'leave'}) {
202 if (($chat_state < CHAT_STATE->{'ready'}) && !$password_ok) {
203 $message = 'Already disconnected.';
206 if ($username =~ /^[A-Za-z]*$/) {
207 $chat{'content'} .= 'leave@'.$username.': '.$words."\n";
208 if ($username ne '') {
209 write_data_file($fh, \%chat);
213 if ($chat_state > 1) {
214 write_data_file(DATA_CHAT_PATH.$chat_id, \%chat);
215 $new_chat{'id'} = $chat_id+1;
218 $new_chat{'id'} = $chat_id;
220 $new_chat{'state'} = CHAT_STATE->{'disconnected'};
221 $new_chat{'content'} = '';
222 write_data_file($fh, \%new_chat);
226 $message = 'Invalid username.';
232 ($action == CHAT_ACTION->{'file'}) &&
233 ($cgi{'file'} ne '') &&
237 if ($words !~ /[\r\n]/) {
238 if ($username =~ /^[A-Za-z]*$/) {
239 $chat{'content'} .= 'file@'.$username.': '.$words."\n";
240 if ($chat_state < CHAT_STATE->{'active'}) {
241 $chat_state = CHAT_STATE->{'active'};
242 $chat{'state'} = $chat_state;
244 write_data_file($fh, \%chat);
247 $message = 'Invalid username.';
251 $message = 'Invalid text.';
254 @chat_lines = split(/\r?\n/, $chat{'content'});
257 $chat_state = CHAT_STATE->{'disconnected'};
258 $message = 'Can\'t lock data file!';
264 $chat_state = CHAT_STATE->{'disconnected'};
265 $message='Can\'t open data file!';
271 %chat = read_data_file(DATA_CHAT_PATH());
272 $last_id = int($chat{'id'});
273 if ($chat_id < $last_id) {
274 %chat = read_data_file(DATA_CHAT_PATH.$page);
275 $chat_state = int($chat{'state'});
276 @chat_lines = split(/\r?\n/, $chat{'content'});
280 print "Content-type: text/html\n\n";
281 if($method eq 'HEAD') {
285 if ($username eq '') {
286 $username = $coin{'name'};
289 my $base_url = CGI_PATH();
290 my $coin_url = CGI_COIN_PATH();
291 my $form_url = $coin_url;
292 my $oldest_url = merge_url(
293 {'path' => $coin_url},
296 my $older_url = merge_url(
297 {'path' => $coin_url},
298 {'path' => $chat_id -1}
300 my $newer_url = ($chat_id < ($last_id -1)) ?
302 {'path' => $coin_url},
303 {'path' => $chat_id +1}
307 my $password_query = url_query_encode({'p', $settings{'password'}});
308 $coin_url = merge_url($coin_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
309 $oldest_url = merge_url($oldest_url, {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
310 $older_url = merge_url($older_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
311 $newer_url = merge_url($newer_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
314 my $abbr = abbr_name($username);
315 my $_website_name = html_entity_encode_dec(WEBSITE_NAME() , 1);
316 my $_server = html_entity_encode_dec($coin {'server'} , 1);
317 my $_key = html_entity_encode_dec($coin {'key'} , 1);
318 my $_password = html_entity_encode_dec($settings{'password'}, 1);
319 my $_cgi_username = html_entity_encode_dec($cgi {'username'}, 1);
320 my $_username = html_entity_encode_dec($username , 1);
321 my $_abbr = html_entity_encode_dec($abbr , 1);
322 my $_message = html_entity_encode_dec($message , 1);
323 my $_base_url = html_entity_encode_dec($base_url , 1);
324 my $_coin_url = html_entity_encode_dec($coin_url , 1);
325 my $_form_url = html_entity_encode_dec($form_url , 1);
326 my $_oldest_url = html_entity_encode_dec($oldest_url, 1);
327 my $_older_url = html_entity_encode_dec($older_url , 1);
328 my $_newer_url = html_entity_encode_dec($newer_url , 1);
330 print_html_start(\*STDOUT);
331 print_html_head_start(\*STDOUT);
333 print ' <title>Coincidence • '.$_website_name.'</title>'."\n";
335 print_html_head_end(\*STDOUT);
336 print_html_body_start(\*STDOUT);
338 print ' <div id="inst" class="ins">'."\n";
340 print ' <div id="title">'."\n";
341 print ' <H1 id="titletext">Coincidence</H1>'."\n";
342 print ' </div>'."\n";
344 print ' <div id="storypuzzle">'."\n";
346 print ' Before: '.$chat_id."\n";
348 elsif ($chat_state > CHAT_STATE->{'disconnected'}) {
349 print ' Connected to server <span class="br">'.$_server.'</span> as user <span class="ni">'.$_username.'</span> (<span class="ni">'.$_abbr.'</span>), public key <span class="br">'.$_key.'</span>.'."\n";
352 print ' Not connected.';
354 print ' </div>'."\n";
356 print ' <div id="command">'."\n";
357 if ($message ne '') {
358 print ' <span class="br">'.$_message.'</span>'."\n";
361 print ' <form method="post" action="'.$_form_url.'">'."\n";
363 print ' <input class="intxc" type="text" name="words">'."\n";
364 print ' <input class="inbt" type="submit" value="Send">'."\n";
366 print ' <input class="intx" type="text" name="username" value="'.$_cgi_username.'">'."\n";
367 print ' <input class="inbt" type="submit" name="nopost" value="Refresh">'."\n";
368 print ' <input class="inbt" type="submit" name="join" value="Connect">'."\n";
369 print ' <input class="inbt" type="submit" name="leave" value="Disconnect">'."\n";
370 print ' <input class="inbt" type="submit" name="file" value="Send file">'."\n";
371 print ' <input type="hidden" name="p" value="'.$_password.'">'."\n";
373 elsif ($chat_state > CHAT_STATE->{'disconnected'}) {
374 print ' <input class="intxc" type="text" name="words">'."\n";
375 print ' <input class="inbt" type="submit" value="Send">'."\n";
377 print ' <input class="inbt" type="submit" name="nopost" value="Refresh">'."\n";
378 print ' <input class="inbt" type="submit" name="leave" value="Disconnect">'."\n";
381 print ' <input class="intx" type="text" name="words">'."\n";
382 print ' <input class="inbt" type="submit" name="join" value="Connect">'."\n";
384 print ' </form>'."\n";
386 print ' </div>'."\n";
388 print ' </div>'."\n";
389 print ' <div id="insb" class="ins">'."\n";
391 print ' <div id="chat">'."\n";
393 for (my $i = @chat_lines-1; $i>=0; --$i) {
394 print ' '.chat_line($chat_lines[$i])."<br>\n";
398 for (my $i = 0; $i<@chat_lines; ++$i) {
399 print ' '.chat_line($chat_lines[$i])."<br>\n";
402 print ' </div>'."\n";
404 print ' <div id="underlinks">'."\n";
405 print ' <a href="'.$_base_url.'">BSTA</a> | <a href="'.$_coin_url.'">Once again</a>';
407 print ' | <a href="'.$_older_url.'">Before</a>';
409 if ($chat_id < $last_id) {
410 print ' | <a href="'.$_newer_url.'">Unbefore</a>';
413 print ' | <a href="'.$_oldest_url.'">Initially</a>';
415 print ' | (This interface is only a demo, a proof of concept. It is very limited. No autorefresh, no private chat, etc. For full functionality use the actual Coincidence client.)'."\n";
416 print ' </div>'."\n";
418 print ' </div>'."\n";
420 print_html_body_end(\*STDOUT, int($state{'state'}) == STATE->{'inactive'});
421 print_html_end(\*STDOUT);
428 if($name !~ /^[A-Za-z]+$/) {
432 $abbr = uc(substr($name,0,1));
433 $name = substr($name,1);
434 while($name =~ m/([A-Z])/g) {
443 if ($line =~ /^([a-z]*@)?([A-Za-z]*): (.*)$/) {
449 $name = $coin{'name'};
455 $abbr = abbr_name($name);
457 my $_name = html_entity_encode_dec($name , 1);
458 my $_abbr = html_entity_encode_dec($abbr , 1);
459 my $_text = html_entity_encode_dec($text , 1);
460 my $_server = html_entity_encode_dec($coin{'server'}, 1);
463 if ($action eq 'join@') {
464 return "$_name ($_abbr) joined the public chat on server $_server.";
466 elsif ($action eq 'leave@') {
467 return "$_name ($_abbr) left the public chat on server $_server.";
469 elsif ($action eq 'file@') {
470 return "$_name ($_abbr) sent the file $_text.";
477 return "<span class=\"$color\">$_abbr: $_text</span>";