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
31 'read_data_file', 'write_data_file',
37 'STATE', 'TEXT_MODE', 'INTF_STATE',
38 'fail_method', 'fail_content_type',
39 'get_remote_addr', 'get_frame', 'get_password',
41 'print_viewer_page', 'write_index',
45 ###PERL_PATH_SEPARATOR: PATH_SEPARATOR = /
47 ###PERL_DATA_PATH: DATA_PATH = /botm/data/bsta/
48 ###PERL_DATA_DEFAULT_PATH: DATA_DEFAULT_PATH = /botm/data/bsta/default
49 ###PERL_DATA_NOACCESS_PATH: DATA_NOACCESS_PATH = /botm/data/bsta/noaccess
50 ###PERL_DATA_SETTINGS_PATH: DATA_SETTINGS_PATH = /botm/data/bsta/settings
51 ###PERL_DATA_STATE_PATH: DATA_STATE_PATH = /botm/data/bsta/state
52 ###PERL_DATA_STORY_PATH: DATA_STORY_PATH = /botm/data/bsta/story
54 binmode STDIN, ':encoding(UTF-8)';
55 binmode STDOUT, ':encoding(UTF-8)';
56 binmode STDERR, ':encoding(UTF-8)';
75 my $next_frame_data_path;
87 delete @ENV{qw(IFS CDPATH ENV BASH_ENV)};
88 ###PERL_SET_PATH: $ENV{'PATH'} = /usr/local/bin:/usr/bin:/bin;
90 if ($ENV{'REQUEST_METHOD'} =~ /^(HEAD|GET|POST)$/) {
94 exit fail_method($ENV{'REQUEST_METHOD'}, 'GET, POST, HEAD');
97 %http = read_header_env(\%ENV);
98 %cgi = url_query_decode($ENV{'QUERY_STRING'});
100 if ($method eq 'POST') {
101 if ($http{'content-type'} eq 'application/x-www-form-urlencoded') {
102 my %cgi_post = url_query_decode( <STDIN> );
103 %cgi = merge_settings(\%cgi, \%cgi_post);
105 # multipart not supported
107 exit fail_content_type($method, $http{'content-type'});
111 $IP = get_remote_addr();
112 $frame = get_frame(\%cgi);
113 $password = get_password(\%cgi);
115 %settings = read_data_file(DATA_SETTINGS_PATH());
116 %default = read_data_file(DATA_DEFAULT_PATH());
119 $frame_data_path = join_path(PATH_SEPARATOR(), DATA_PATH(), $frame);
120 %frame_data= read_data_file($frame_data_path);
123 $password_ok = ($password eq $settings{'password'});
125 # state & activation logic
126 if (open_encoded($state_file, "+<:encoding(UTF-8)", DATA_STATE_PATH())) {
127 if (flock($state_file, 2)) {
129 %state = read_data_file($state_file);
132 $frame = int($state{'last'}) + $frame +1;
133 $frame_data_path = join_path(PATH_SEPARATOR(), DATA_PATH(), $frame);
134 %frame_data = read_data_file($frame_data_path);
138 (int($state{'state'}) == STATE->{'waiting'}) &&
139 ($frame == int($state{'last'})) &&
140 ($method ne 'HEAD') &&
143 # register IP for progress
144 my %new_state = %state;
146 ($state{'ip1'} eq $IP) ||
147 ($state{'ip2'} eq $IP) ||
148 ($state{'ip3'} eq $IP)
151 if ($state{'ip1'} eq '') {
152 $new_state{'ip1'} = $IP;
154 elsif ($state{'ip2'} eq '') {
155 $new_state{'ip2'} = $IP;
157 elsif ($state{'ip3'} eq '') {
158 $new_state{'ip3'} = $IP;
159 $new_state{'state'} = STATE->{'ready'};
162 $new_state{'state'} = STATE->{'ready'};
164 write_data_file($state_file, '', '', \%new_state);
168 (int($state{'state'}) == STATE->{'inactive'}) &&
172 # NOTE: at this point frame 0 is already ONGed.
174 my $ong_time = int($settings{'firstongtime'});
177 %story = read_data_file(DATA_STORY_PATH());
180 (int($story{'state'}) == INTF_STATE->{'>|'} ) &&
181 (int($story{'pass'}) == 1)
183 # conditions met; ACTIVATE!
186 $state{'state'} = STATE->{'waiting'};
188 $state{'ip1'} = '0.0.0.0';
189 $state{'ip2'} = '0.0.0.0';
191 $state{'nextong'} = (int($time / 3600) + int($settings{'firstongtime'})) * 3600 ;
192 $state{'ongtime'} = $ong_time;
194 # prepare to ONG frame 1
208 $r = write_index(\%state, \%settings);
211 $r = write_data_file($state_file, '','', \%state);
214 # FAILED ONG! Story as if it was inactive!
215 $state{'state'} = STATE->{'inactive'};
221 # FAILED GET STATE! Story as if it was inactive!
222 $state{'state'} = STATE->{'inactive'};
227 $state{'state'} = STATE->{'inactive'};
230 $next_frame_data_path = join_path(PATH_SEPARATOR(), DATA_PATH(), $frame+1);
231 %next_frame_data = read_data_file($next_frame_data_path);
234 %frame_data = merge_settings(\%default, \%frame_data);
235 %next_frame_data = merge_settings(\%default, \%next_frame_data);
237 $timer = int($state{'nextong'}) - $time;
238 $ongtime = int($state{'ongtime'});
240 $ongtime = int($settings{'ongtime'})
243 $show_command = ($timer < ($ongtime*3600/3));
244 if ($state{'state'} >= STATE->{'ready'}) {
247 elsif ($state{'ip3'} ne '') {
250 elsif ($state{'ip2'} ne '') {
253 elsif ($state{'ip1'} ne '') {
262 (int($state{'state'}) >= STATE->{'waiting'}) &&
263 ($frame <= int($state{'last'})) &&
271 # replace frame data with fail state replacement
272 %frame_data = read_data_file(DATA_NOACCESS_PATH());
273 %frame_data = merge_settings(\%default, \%frame_data);
276 $text_mode = int($cgi{'b'});
278 $text_mode = TEXT_MODE->{'normal'};
281 print "Content-type: text/html\n";
283 print "Status: 403 Forbidden\n";
286 if($method eq 'HEAD') {
296 'password_ok' => $password_ok,
297 'timer_unlocked'=> $timer_unlocked,
300 'show_command' => $show_command,
301 'text_mode' => $text_mode,