1 ###RUN_PERL: #!/usr/bin/perl
4 # viewer is generated from viewer.1.pl.
8 # Copyright (C) 2016, 2017, 2019, 2020, 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 # use Encode::Locale ('decode_argv');
26 use Encode ('encode', 'decode');
28 ###PERL_LIB: use lib /botm/lib/bsta
32 'read_data_file', 'write_data_file',
39 'STATE', 'TEXT_MODE', 'INTF_STATE',
40 'fail_method', 'fail_content_type',
41 'get_remote_addr', 'get_frame', 'get_password',
43 'print_viewer_page', 'write_index',
47 ###PERL_PATH_SEPARATOR: PATH_SEPARATOR = /
49 ###PERL_DATA_PATH: DATA_PATH = /botm/data/bsta/
50 ###PERL_DATA_DEFAULT_PATH: DATA_DEFAULT_PATH = /botm/data/bsta/default
51 ###PERL_DATA_NOACCESS_PATH: DATA_NOACCESS_PATH = /botm/data/bsta/noaccess
52 ###PERL_DATA_SETTINGS_PATH: DATA_SETTINGS_PATH = /botm/data/bsta/settings
53 ###PERL_DATA_STATE_PATH: DATA_STATE_PATH = /botm/data/bsta/state
54 ###PERL_DATA_STORY_PATH: DATA_STORY_PATH = /botm/data/bsta/story
55 ###PERL_DATA_WORDS_PATH: DATA_WORDS_PATH = /botm/data/bsta/words/
57 binmode STDIN, ':encoding(UTF-8)';
58 binmode STDOUT, ':encoding(UTF-8)';
59 binmode STDERR, ':encoding(UTF-8)';
80 my $prev_frame_data_path;
81 my $next_frame_data_path;
95 delete @ENV{qw(IFS CDPATH ENV BASH_ENV)};
96 ###PERL_SET_PATH: $ENV{'PATH'} = /usr/local/bin:/usr/bin:/bin;
98 if ($ENV{'REQUEST_METHOD'} =~ /^(HEAD|GET|POST)$/) {
102 exit fail_method($ENV{'REQUEST_METHOD'}, ['GET', 'POST', 'HEAD']);
105 %http = read_header_env(\%ENV);
106 %cgi = url_query_decode($ENV{'QUERY_STRING'});
108 if ($method eq 'POST') {
109 if ($http{'content-type'} eq 'application/x-www-form-urlencoded') {
110 my %cgi_post = url_query_decode( <STDIN> );
111 %cgi = merge_settings(\%cgi, \%cgi_post);
113 # multipart not supported
115 exit fail_content_type($method, $http{'content-type'});
119 $IP = get_remote_addr();
120 $frame = get_frame(\%cgi);
121 $password = get_password(\%cgi);
123 %settings = read_data_file(DATA_SETTINGS_PATH());
124 %default = read_data_file(DATA_DEFAULT_PATH());
127 $frame_data_path = join_path(PATH_SEPARATOR(), DATA_PATH(), $frame);
128 %frame_data= read_data_file($frame_data_path);
131 $password_ok = ($password eq $settings{'password'});
133 # state & activation logic
134 if (open_encoded($fh, "+<:encoding(UTF-8)", DATA_STATE_PATH())) {
137 %state = read_data_file($fh);
140 $frame = int($state{'last'}) + $frame +1;
141 $frame_data_path = join_path(PATH_SEPARATOR(), DATA_PATH(), $frame);
142 %frame_data = read_data_file($frame_data_path);
146 (int($state{'state'}) == STATE->{'waiting'}) &&
147 ($frame == int($state{'last'})) &&
148 ($method ne 'HEAD') &&
151 # register IP for progress
152 my %new_state = %state;
154 ($state{'ip1'} eq $IP) ||
155 ($state{'ip2'} eq $IP) ||
156 ($state{'ip3'} eq $IP)
159 if ($state{'ip1'} eq '') {
160 $new_state{'ip1'} = $IP;
162 elsif ($state{'ip2'} eq '') {
163 $new_state{'ip2'} = $IP;
165 elsif ($state{'ip3'} eq '') {
166 $new_state{'ip3'} = $IP;
167 $new_state{'state'} = STATE->{'ready'};
170 $new_state{'state'} = STATE->{'ready'};
172 write_data_file($fh, \%new_state);
176 (int($state{'state'}) == STATE->{'inactive'}) &&
180 # NOTE: at this point frame 0 is already ONGed.
182 my $ong_time = int($settings{'firstongtime'});
185 %story = read_data_file(DATA_STORY_PATH());
188 (int($story{'state'}) == INTF_STATE->{'>|'} ) &&
189 (int($story{'pass'}) == 1)
191 # conditions met; ACTIVATE!
194 $state{'state'} = STATE->{'waiting'};
196 $state{'ip1'} = '0.0.0.0';
197 $state{'ip2'} = '0.0.0.0';
199 $state{'nextong'} = (int($time / 3600) + int($settings{'firstongtime'})) * 3600 ;
200 $state{'ongtime'} = $ong_time;
202 # prepare to ONG frame 1
216 $r = write_index(\%state, \%settings);
219 $r = write_data_file($fh, \%state);
222 # FAILED ONG! Story as if it was inactive!
223 $state{'state'} = STATE->{'inactive'};
229 # FAILED GET STATE! Story as if it was inactive!
230 $state{'state'} = STATE->{'inactive'};
235 $state{'state'} = STATE->{'inactive'};
238 $timer = int($state{'nextong'}) - $time;
239 $ongtime = int($state{'ongtime'});
241 $ongtime = int($settings{'ongtime'})
244 $show_command = ($timer < ($ongtime*3600/3));
245 if ($state{'state'} >= STATE->{'ready'}) {
248 elsif ($state{'ip3'} ne '') {
251 elsif ($state{'ip2'} ne '') {
254 elsif ($state{'ip1'} ne '') {
263 (int($state{'state'}) >= STATE->{'waiting'}) &&
264 ($frame <= int($state{'last'})) &&
270 $prev_frame_data_path = join_path(PATH_SEPARATOR(), DATA_PATH(), $frame-1);
271 %prev_frame_data = read_data_file($prev_frame_data_path);
273 $next_frame_data_path = join_path(PATH_SEPARATOR(), DATA_PATH(), $frame+1);
274 %next_frame_data = read_data_file($next_frame_data_path);
276 %frame_data = merge_settings(\%default, \%frame_data);
277 %prev_frame_data = merge_settings(\%default, \%prev_frame_data);
278 %next_frame_data = merge_settings(\%default, \%next_frame_data);
282 # replace frame data with fail state replacement
283 %frame_data = read_data_file(DATA_NOACCESS_PATH());
284 %frame_data = merge_settings(\%default, \%frame_data);
287 $text_mode = int($cgi{'b'});
288 if($text_mode > TEXT_MODE->{'words'}) {
289 $text_mode = TEXT_MODE->{'normal'};
291 $words_page = int($cgi{'i'});
293 $words_data_path = join_path(PATH_SEPARATOR(), DATA_WORDS_PATH(), $frame);
294 %words_data = read_data_file(
295 $words_data_path, # file
298 ($text_mode != TEXT_MODE->{'words'}), # header only
303 print http_header_status(HTTP_STATUS->{'forbidden'});
305 print "Content-type: text/html; charset=UTF-8\n\n";
306 if($method eq 'HEAD') {
316 'password_ok' => $password_ok,
317 'timer_unlocked'=> $timer_unlocked,
320 'show_command' => $show_command,
321 'text_mode' => $text_mode,
322 'words_page' => $words_page
327 $access ? \%prev_frame_data : \%frame_data,
328 $access ? \%next_frame_data : \%frame_data,