up
[git.ikiwiki.info.git] / ikiwiki
1 #!/usr/bin/perl -T
3 use warnings;
4 use strict;
5 use File::Find;
6 use Memoize;
7 use File::Spec;
9 $ENV{PATH}="/usr/local/bin:/usr/bin:/bin";
11 BEGIN {
12         $blosxom::version="is a proper perl module too much to ask?";
13         do "/usr/bin/markdown";
14 }
16 my ($srcdir, $destdir, %links, %oldlinks, %oldpagemtime, %renderedfiles,
17     %pagesources);
18 my $wiki_link_regexp=qr/\[\[([^\s]+)\]\]/;
19 my $wiki_file_regexp=qr/(^[-A-Za-z0-9_.:\/+]+$)/;
20 my $wiki_file_prune_regexp=qr!((^|/).svn/|\.\.)!;
21 my $verbose=0;
22 my $wikiname="wiki";
23 my $default_pagetype=".mdwn";
24 my $cgi=0;
25 my $url="";
26 my $svn=1;
28 sub usage {
29         die "usage: ikiwiki [options] source dest\n";
30 }
32 sub error ($) {
33         if ($cgi) {
34                 print "Content-type: text/html\n\n";
35                 print "Error: @_\n";
36                 exit 1;
37         }
38         else {
39                 die @_;
40         }
41 }
43 sub debug ($) {
44         print "@_\n" if $verbose;
45 }
47 sub mtime ($) {
48         my $page=shift;
49         
50         return (stat($page))[9];
51 }
53 sub possibly_foolish_untaint ($) {
54         my $tainted=shift;
55         my ($untainted)=$tainted=~/(.*)/;
56         return $untainted;
57 }
59 sub basename {
60         my $file=shift;
62         $file=~s!.*/!!;
63         return $file;
64 }
66 sub dirname {
67         my $file=shift;
69         $file=~s!/?[^/]+$!!;
70         return $file;
71 }
73 sub pagetype ($) {
74         my $page=shift;
75         
76         if ($page =~ /\.mdwn$/) {
77                 return ".mdwn";
78         }
79         else {
80                 return "unknown";
81         }
82 }
84 sub pagename ($) {
85         my $file=shift;
87         my $type=pagetype($file);
88         my $page=$file;
89         $page=~s/\Q$type\E*$// unless $type eq 'unknown';
90         return $page;
91 }
93 sub htmlpage ($) {
94         my $page=shift;
96         return $page.".html";
97 }
99 sub readfile ($) {
100         my $file=shift;
102         local $/=undef;
103         open (IN, "$file") || error("failed to read $file: $!");
104         my $ret=<IN>;
105         close IN;
106         return $ret;
109 sub writefile ($$) {
110         my $file=shift;
111         my $content=shift;
113         my $dir=dirname($file);
114         if (! -d $dir) {
115                 my $d="";
116                 foreach my $s (split(m!/+!, $dir)) {
117                         $d.="$s/";
118                         if (! -d $d) {
119                                 mkdir($d) || error("failed to create directory $d: $!");
120                         }
121                 }
122         }
123         
124         open (OUT, ">$file") || error("failed to write $file: $!");
125         print OUT $content;
126         close OUT;
129 sub findlinks {
130         my $content=shift;
132         my @links;
133         while ($content =~ /$wiki_link_regexp/g) {
134                 push @links, lc($1);
135         }
136         return @links;
139 # Given a page and the text of a link on the page, determine which existing
140 # page that link best points to. Prefers pages under a subdirectory with
141 # the same name as the source page, failing that goes down the directory tree
142 # to the base looking for matching pages.
143 sub bestlink ($$) {
144         my $page=shift;
145         my $link=lc(shift);
146         
147         my $cwd=$page;
148         do {
149                 my $l=$cwd;
150                 $l.="/" if length $l;
151                 $l.=$link;
153                 if (exists $links{$l}) {
154                         #debug("for $page, \"$link\", use $l");
155                         return $l;
156                 }
157         } while $cwd=~s!/?[^/]+$!!;
159         #print STDERR "warning: page $page, broken link: $link\n";
160         return "";
163 sub isinlinableimage ($) {
164         my $file=shift;
165         
166         $file=~/\.(png|gif|jpg|jpeg)$/;
169 sub htmllink ($$) {
170         my $page=shift;
171         my $link=shift;
173         my $bestlink=bestlink($page, $link);
175         return $link if $page eq $bestlink;
176         
177         # TODO BUG: %renderedfiles may not have it, if the linked to page
178         # was also added and isn't yet rendered! Note that this bug is
179         # masked by the bug mentioned below that makes all new files
180         # be rendered twice.
181         if (! grep { $_ eq $bestlink } values %renderedfiles) {
182                 $bestlink=htmlpage($bestlink);
183         }
184         if (! grep { $_ eq $bestlink } values %renderedfiles) {
185                 return "<a href=\"?\">?</a>$link"
186         }
187         
188         $bestlink=File::Spec->abs2rel($bestlink, dirname($page));
189         
190         if (isinlinableimage($bestlink)) {
191                 return "<img src=\"$bestlink\">";
192         }
193         return "<a href=\"$bestlink\">$link</a>";
196 sub linkify ($$) {
197         my $content=shift;
198         my $file=shift;
200         $content =~ s/$wiki_link_regexp/htmllink(pagename($file), $1)/eg;
201         
202         return $content;
205 sub htmlize ($$) {
206         my $type=shift;
207         my $content=shift;
208         
209         if ($type eq '.mdwn') {
210                 return Markdown::Markdown($content);
211         }
212         else {
213                 error("htmlization of $type not supported");
214         }
217 sub linkbacks ($$) {
218         my $content=shift;
219         my $page=shift;
221         my @links;
222         foreach my $p (keys %links) {
223                 next if bestlink($page, $p) eq $page;
224                 if (grep { length $_ && bestlink($p, $_) eq $page } @{$links{$p}}) {
225                         my $href=File::Spec->abs2rel(htmlpage($p), dirname($page));
226                         
227                         # Trim common dir prefixes from both pages.
228                         my $p_trimmed=$p;
229                         my $page_trimmed=$page;
230                         my $dir;
231                         1 while (($dir)=$page_trimmed=~m!^([^/]+/)!) &&
232                                 defined $dir &&
233                                 $p_trimmed=~s/^\Q$dir\E// &&
234                                 $page_trimmed=~s/^\Q$dir\E//;
235                                        
236                         push @links, "<a href=\"$href\">$p_trimmed</a>";
237                 }
238         }
240         $content.="<hr><p>Links: ".join(" ", sort @links)."</p>\n" if @links;
241         return $content;
244 sub finalize ($$) {
245         my $content=shift;
246         my $page=shift;
248         my $title=basename($page);
249         $title=~s/_/ /g;
250         
251         my $pagelink="";
252         my $path="";
253         foreach my $dir (reverse split("/", $page)) {
254                 if (length($pagelink)) {
255                         $pagelink="<a href=\"$path$dir.html\">$dir</a>/ $pagelink";
256                 }
257                 else {
258                         $pagelink=$dir;
259                 }
260                 $path.="../";
261         }
262         $path=~s/\.\.\/$/index.html/;
263         $pagelink="<a href=\"$path\">$wikiname</a>/ $pagelink";
264         
265         $content="<html>\n<head><title>$title</title></head>\n<body>\n".
266                   "<h1>$pagelink</h1>\n".
267                   $content.
268                   "</body>\n</html>\n";
269         
270         return $content;
273 sub render ($) {
274         my $file=shift;
275         
276         my $type=pagetype($file);
277         my $content=readfile("$srcdir/$file");
278         if ($type ne 'unknown') {
279                 my $page=pagename($file);
280                 
281                 $links{$page}=[findlinks($content)];
282                 
283                 $content=linkify($content, $file);
284                 $content=htmlize($type, $content);
285                 $content=linkbacks($content, $page);
286                 $content=finalize($content, $page);
287                 
288                 writefile("$destdir/".htmlpage($page), $content);
289                 $oldpagemtime{$page}=time;
290                 $renderedfiles{$page}=htmlpage($page);
291         }
292         else {
293                 $links{$file}=[];
294                 writefile("$destdir/$file", $content);
295                 $oldpagemtime{$file}=time;
296                 $renderedfiles{$file}=$file;
297         }
300 sub loadindex () {
301         open (IN, "$srcdir/.index") || return;
302         while (<IN>) {
303                 $_=possibly_foolish_untaint($_);
304                 chomp;
305                 my ($mtime, $file, $rendered, @links)=split(' ', $_);
306                 my $page=pagename($file);
307                 $pagesources{$page}=$file;
308                 $oldpagemtime{$page}=$mtime;
309                 $oldlinks{$page}=[@links];
310                 $links{$page}=[@links];
311                 $renderedfiles{$page}=$rendered;
312         }
313         close IN;
314 }       
316 sub saveindex () {
317         open (OUT, ">$srcdir/.index") || error("cannot write to .index: $!");
318         foreach my $page (keys %oldpagemtime) {
319         print OUT "$oldpagemtime{$page} $pagesources{$page} $renderedfiles{$page} ".
320                   join(" ", @{$links{$page}})."\n"
321                         if $oldpagemtime{$page};
322         }
323         close OUT;
326 sub rcs_update () {
327         if (-d "$srcdir/.svn") {
328                 if (system("svn", "update", "--quiet", $srcdir) != 0) {
329                         warn("svn update failed\n");
330                 }
331         }
334 sub rcs_commit ($) {
335         my $message=shift;
337         if (-d "$srcdir/.svn") {
338                 if (system("svn", "commit", "--quiet", "-m",
339                            possibly_foolish_untaint($message), $srcdir) != 0) {
340                         warn("svn commit failed\n");
341                 }
342         }
345 sub rcs_ad ($) {
346         my $file=shift;
348         if (-d "$srcdir/.svn") {
349                 if (system("svn", "add", "--quiet", $file) != 0) {
350                         warn("svn add failed\n");
351                 }
352         }
355 sub prune ($) {
356         my $file=shift;
358         unlink($file);
359         my $dir=dirname($file);
360         while (rmdir($dir)) {
361                 $dir=dirname($dir);
362         }
365 sub refresh () {
366         # Find existing pages.
367         my %exists;
368         my @files;
369         find({
370                 no_chdir => 1,
371                 wanted => sub {
372                         if (/$wiki_file_prune_regexp/) {
373                                 $File::Find::prune=1;
374                         }
375                         elsif (! -d $_ && ! /\.html$/ && ! /\/\./) {
376                                 my ($f)=/$wiki_file_regexp/; # untaint
377                                 if (! defined $f) {
378                                         warn("skipping bad filename $_\n");
379                                 }
380                                 else {
381                                         $f=~s/^\Q$srcdir\E\/?//;
382                                         push @files, $f;
383                                         $exists{pagename($f)}=1;
384                                 }
385                         }
386                 },
387         }, $srcdir);
389         my %rendered;
391         # check for added or removed pages
392         my @add;
393         foreach my $file (@files) {
394                 my $page=pagename($file);
395                 if (! $oldpagemtime{$page}) {
396                         debug("new page $page");
397                         push @add, $file;
398                         $links{$page}=[];
399                         $pagesources{$page}=$file;
400                 }
401         }
402         my @del;
403         foreach my $page (keys %oldpagemtime) {
404                 if (! $exists{$page}) {
405                         debug("removing old page $page");
406                         push @del, $renderedfiles{$page};
407                         prune($destdir."/".$renderedfiles{$page});
408                         delete $renderedfiles{$page};
409                         $oldpagemtime{$page}=0;
410                         delete $pagesources{$page};
411                 }
412         }
413         
414         # render any updated files
415         foreach my $file (@files) {
416                 my $page=pagename($file);
417                 
418                 if (! exists $oldpagemtime{$page} ||
419                     mtime("$srcdir/$file") > $oldpagemtime{$page}) {
420                         debug("rendering changed file $file");
421                         render($file);
422                         $rendered{$file}=1;
423                 }
424         }
425         
426         # if any files were added or removed, check to see if each page
427         # needs an update due to linking to them
428         # TODO: inefficient; pages may get rendered above and again here;
429         # problem is the bestlink may have changed and we won't know until
430         # now
431         if (@add || @del) {
432 FILE:           foreach my $file (@files) {
433                         my $page=pagename($file);
434                         foreach my $f (@add, @del) {
435                                 my $p=pagename($f);
436                                 foreach my $link (@{$links{$page}}) {
437                                         if (bestlink($page, $link) eq $p) {
438                                                 debug("rendering $file, which links to $p");
439                                                 render($file);
440                                                 $rendered{$file}=1;
441                                                 next FILE;
442                                         }
443                                 }
444                         }
445                 }
446         }
448         # handle linkbacks; if a page has added/removed links, update the
449         # pages it links to
450         # TODO: inefficient; pages may get rendered above and again here;
451         # problem is the linkbacks could be wrong in the first pass render
452         # above
453         if (%rendered) {
454                 my %linkchanged;
455                 foreach my $file (keys %rendered, @del) {
456                         my $page=pagename($file);
457                         if (exists $links{$page}) {
458                                 foreach my $link (@{$links{$page}}) {
459                                         $link=bestlink($page, $link);
460                                         if (length $link &&
461                                             ! exists $oldlinks{$page} ||
462                                             ! grep { $_ eq $link } @{$oldlinks{$page}}) {
463                                                 $linkchanged{$link}=1;
464                                         }
465                                 }
466                         }
467                         if (exists $oldlinks{$page}) {
468                                 foreach my $link (@{$oldlinks{$page}}) {
469                                         $link=bestlink($page, $link);
470                                         if (length $link &&
471                                             ! exists $links{$page} ||
472                                             ! grep { $_ eq $link } @{$links{$page}}) {
473                                                 $linkchanged{$link}=1;
474                                         }
475                                 }
476                         }
477                 }
478                 foreach my $link (keys %linkchanged) {
479                         my $linkfile=$pagesources{$link};
480                         if (defined $linkfile) {
481                                 debug("rendering $linkfile, to update its linkbacks");
482                                 render($linkfile);
483                         }
484                 }
485         }
488 # Generates a C wrapper program for running ikiwiki in a specific way.
489 # The wrapper may be safely made suid.
490 sub gen_wrapper ($$) {
491         my ($svn, $rebuild)=@_;
493         eval {use Cwd 'abs_path'};
494         $srcdir=abs_path($srcdir);
495         $destdir=abs_path($destdir);
496         my $this=abs_path($0);
497         if (! -x $this) {
498                 error("$this doesn't seem to be executable");
499         }
501         my @params=($srcdir, $destdir, "--wikiname=$wikiname");
502         push @params, "--verbose" if $verbose;
503         push @params, "--rebuild" if $rebuild;
504         push @params, "--nosvn" if !$svn;
505         push @params, "--cgi" if $cgi;
506         push @params, "--url=$url" if $url;
507         my $params=join(" ", @params);
508         my $call='';
509         foreach my $p ($this, $this, @params) {
510                 $call.=qq{"$p", };
511         }
512         $call.="NULL";
513         
514         my @envsave;
515         push @envsave, qw{REMOTE_ADDR QUERY_STRING REQUEST_METHOD REQUEST_URI
516                        CONTENT_TYPE CONTENT_LENGTH GATEWAY_INTERFACE} if $cgi;
517         my $envsave="";
518         foreach my $var (@envsave) {
519                 $envsave.=<<"EOF"
520         if ((s=getenv("$var")))
521                 asprintf(&newenviron[i++], "%s=%s", "$var", s);
522 EOF
523         }
524         
525         open(OUT, ">ikiwiki-wrap.c") || error("failed to write ikiwiki-wrap.c: $!");;
526         print OUT <<"EOF";
527 /* A wrapper for ikiwiki, can be safely made suid. */
528 #define _GNU_SOURCE
529 #include <stdio.h>
530 #include <unistd.h>
531 #include <stdlib.h>
532 #include <string.h>
534 extern char **environ;
536 int main (int argc, char **argv) {
537         /* Sanitize environment. */
538         char *s;
539         char *newenviron[$#envsave+3];
540         int i=0;
541 $envsave
542         newenviron[i++]="HOME=$ENV{HOME}";
543         newenviron[i]=NULL;
544         environ=newenviron;
546         if (argc == 2 && strcmp(argv[1], "--params") == 0) {
547                 printf("$params\\n");
548                 exit(0);
549         }
550         
551         execl($call);
552         perror("failed to run $this");
553         exit(1);
555 EOF
556         close OUT;
557         if (system("gcc", "ikiwiki-wrap.c", "-o", "ikiwiki-wrap") != 0) {
558                 error("failed to compile ikiwiki-wrap.c");
559         }
560         unlink("ikiwiki-wrap.c");
561         print "successfully generated ikiwiki-wrap\n";
562         exit 0;
565 sub cgi () {
566         eval q{use CGI};
567         my $q=CGI->new;
569         my $do=$q->param('do');
570         if (! defined $do || ! length $do) {
571                 error("\"do\" parameter missing");
572         }
573         
574         my ($page)=$q->param('page')=~/$wiki_file_regexp/; # untaint
575         if (! defined $page || ! length $page || $page ne $q->param('page') ||
576             $page=~/$wiki_file_prune_regexp/ || $page=~/^\//) {
577                 error("bad page name");
578         }
579         
580         my $action=$q->request_uri;
581         $action=~s/\?.*//;
582         
583         if ($do eq 'edit') {
584                 my $content="";
585                 if (exists $pagesources{lc($page)}) {
586                         $content=readfile("$srcdir/$pagesources{lc($page)}");
587                         $content=~s/\n/\r\n/g;
588                 }
589                 $q->param("do", "save");
590                 print $q->header,
591                       $q->start_html("$wikiname: Editing $page"),
592                       $q->h1("$wikiname: Editing $page"),
593                       $q->start_form(-action => $action),
594                       $q->hidden('do'),
595                       $q->hidden('page'),
596                       $q->textarea(-name => 'content',
597                                -default => $content,
598                                -rows => 20,
599                                -columns => 80),
600                       $q->br,
601                       "Optional comment about this change",
602                       $q->br,
603                       $q->textfield(-name => "comments", -size => 80),
604                       $q->br,
605                       $q->submit("Save Changes"),
606                       $q->end_form,
607                       $q->end_html;
608         }
609         elsif ($do eq 'save') {
610                 my $file=$page.$default_pagetype;
611                 my $newfile=1;
612                 if (exists $pagesources{lc($page)}) {
613                         $file=$pagesources{lc($page)};
614                         $newfile=0;
615                 }
616                 
617                 my $content=$q->param('content');
618                 $content=~s/\r\n/\n/g;
619                 $content=~s/\r/\n/g;
620                 writefile("$srcdir/$file", $content);
621                 
622                 my $message="web commit from $ENV{REMOTE_ADDR}";
623                 if (defined $q->param('comments')) {
624                         $message.=": ".$q->param('comments');
625                 }
626                 
627                 if ($svn) {
628                         if ($newfile) {
629                                 rcs_add($file);
630                         }
631                         # presumably the commit will trigger an update
632                         # of the wiki
633                         rcs_commit($message);
634                 }
635                 else {
636                         refresh();
637                 }
638                 
639                 print $q->redirect("$url/".htmlpage($page));
640         }
641         else {
642                 error("unknown do parameter");
643         }
646 my $rebuild=0;
647 my $wrapper=0;
648 if (grep /^-/, @ARGV) {
649         eval {use Getopt::Long};
650         GetOptions(
651                 "wikiname=s" => \$wikiname,
652                 "verbose|v" => \$verbose,
653                 "rebuild" => \$rebuild,
654                 "wrapper" => \$wrapper,
655                 "svn!" => \$svn,
656                 "cgi" => \$cgi,
657                 "url=s" => \$url,
658         ) || usage();
660 usage() unless @ARGV == 2;
661 ($srcdir) = possibly_foolish_untaint(shift);
662 ($destdir) = possibly_foolish_untaint(shift);
664 if ($cgi && ! length $url) {
665         error("Must specify url to wiki with --url when using --cgi");
668 gen_wrapper($svn, $rebuild) if $wrapper;
669 memoize('pagename');
670 memoize('bestlink');
671 loadindex() unless $rebuild;
672 if ($cgi) {
673         cgi();
675 else {
676         rcs_update() if $svn;
677         refresh();
678         saveindex();