3 # mkindex was written by Omar Polo <op@openbsd.org> and is placed in
4 # the public domain. The author hereby disclaims copyright to this
7 use open ":std", ":encoding(UTF-8)";
12 use File::Temp qw(tempfile);
18 use GotMArc qw(parse san urlencode initpage endpage index_header
19 search thread_header threntry);
21 my $outdir = $ENV{'OUTDIR'};
22 die 'Set $OUTDIR' unless defined $outdir;
24 my $tfh; # thread file handle
25 my $pfh; # page file handle
40 my $threads_per_page = 100;
44 return $a unless defined $b;
45 return $a gt $b ? $a : $b;
50 return $a unless defined $b;
51 return $a lt $b ? $a : $b;
56 return $i == 1 && "index.html" || "$i.html";
60 say $pfh '</ul></div>';
62 push @pages, "$from_day - $to_day";
66 endfile if defined $pfh;
70 ($pfh, $path) = tempfile "/tmp/gotmarc.index.XXXXXXXXXX";
71 binmode($pfh, ':utf8');
73 say $pfh "<div class='thread'><ul>";
78 my ($first, $last) = (pagename(1), pagename($page));
79 my ($next, $prev) = (pagename($n+1), pagename($n-1));
82 say $pfh "<a href='$first'>First</a>" if $n > 2;
83 say $pfh "<a href='$prev'>Prev</a>" if $n > 1;
84 say $pfh "<a href='$next'>Next</a>" if $n < $page;
85 say $pfh "<a href='$last'>Last</a>" if $n < $page - 1;
92 # there are probably faster ways to do this like File::Copy,
93 # but it bypasses the bufio cache...
94 open(my $pfh, '<', $path) or die "can't open $path: $!";
95 print $fh $_ while <$pfh>;
101 for (my $i = 1; $i <= $page; $i++) {
102 my $name = pagename($i);
103 my $path = shift @files;
104 my $dest = "$outdir/$name";
106 open(my $pfh, '>', $dest)
107 or die "can't open $dest for writing: $!";
109 my $title = "Game of Trees Mail Archive | page $i";
110 my $subtitle = $pages[$i-1];
112 initpage($pfh, $title);
113 index_header $pfh, $i, $subtitle;
116 nav $pfh, $i if $page > 1;
118 copyfrom($path, $pfh);
119 nav $pfh, $i if $page > 1;
130 say $tfh "</ul></li>" x $last_level;
131 say $tfh "</ul>\n</div>\n";
139 endthread if defined $tfh;
140 my ($mid, $subj) = @_;
141 my $dest = "$outdir/thread/$mid.html";
142 open($tfh, '>', $dest) or die "can't open $dest: $!";
143 initpage($tfh, $subj);
144 thread_header $tfh, ["Thread: $subj"];
145 print $tfh "<div class='thread'><ul class='mails'>\n";
149 my ($fh, $mid, $date, $from, $subj) = @_;
151 # synthetic mail hash
160 threntry $fh, "thread", 0, 0, $mail;
163 unveil($outdir, "rwc") or die "unveil $outdir: $!";
165 # can't use tmppath because File::Temp checks wether /tmp exists.
166 unveil("/tmp", "rwc") or die "unveil /tmp: $!";
168 # fattr for File::Temp
169 pledge("stdio rpath wpath cpath fattr") or die "pledge: $!";
176 if ($mail->{level} == 0) {
177 nextthread $mail->{mid}, $mail->{subj};
180 if ($threads > $threads_per_page) {
187 my $day = $mail->{date} =~ s/ .*//r;
188 $to_day = mins $day, $to_day;
189 $from_day = maxs $day, $from_day;
192 $last_level = threntry $tfh, "mail", 0, $last_level, $mail;
195 index_entry $pfh, $last_tid, $last_date, $last_from, $last_subj
196 if defined $last_tid && $mail->{level} == 0;
198 # `gt' on dates works because the format used allow for
199 # lexicographic comparisons.
200 if ($mail->{level} == 0 || $mail->{date} gt $last_date) {
201 $last_date = $mail->{date};
202 $last_from = $mail->{from};
205 if ($mail->{level} == 0) {
206 $last_tid = $mail->{mid};
207 $last_subj = $mail->{subj};
211 index_entry $pfh, $last_tid, $last_date, $last_from, $last_subj
212 if defined $last_tid;
215 endthread if $threads_seen;