]> git.vanrenterghem.biz Git - git.ikiwiki.info.git/blob - IkiWiki/CGI.pm
75f096e137b99e4fccd39fc9c6d290bf9106a1e2
[git.ikiwiki.info.git] / IkiWiki / CGI.pm
1 #!/usr/bin/perl
3 use warnings;
4 use strict;
5 use IkiWiki;
6 use IkiWiki::UserInfo;
7 use open qw{:utf8 :std};
8 use Encode;
10 package IkiWiki;
12 sub printheader ($) { #{{{
13         my $session=shift;
14         
15         if ($config{sslcookie}) {
16                 print $session->header(-charset => 'utf-8',
17                         -cookie => $session->cookie(-secure => 1));
18         } else {
19                 print $session->header(-charset => 'utf-8');
20         }
22 } #}}}
23         
24 sub showform ($$$$) { #{{{
25         my $form=shift;
26         my $buttons=shift;
27         my $session=shift;
28         my $cgi=shift;
30         if (exists $hooks{formbuilder}) {
31                 run_hooks(formbuilder => sub {
32                         shift->(form => $form, cgi => $cgi, session => $session,
33                                 buttons => $buttons);
34                 });
35         }
36         else {
37                 printheader($session);
38                 print misctemplate($form->title, $form->render(submit => $buttons));
39         }
40 }
42 sub redirect ($$) { #{{{
43         my $q=shift;
44         my $url=shift;
45         if (! $config{w3mmode}) {
46                 print $q->redirect($url);
47         }
48         else {
49                 print "Content-type: text/plain\n";
50                 print "W3m-control: GOTO $url\n\n";
51         }
52 } #}}}
54 sub check_canedit ($$$;$) { #{{{
55         my $page=shift;
56         my $q=shift;
57         my $session=shift;
58         my $nonfatal=shift;
59         
60         my $canedit;
61         run_hooks(canedit => sub {
62                 return if defined $canedit;
63                 my $ret=shift->($page, $q, $session);
64                 if (defined $ret && $ret eq "") {
65                         $canedit=1;
66                 }
67                 elsif (defined $ret) {
68                         $canedit=0;
69                         error($ret) unless $nonfatal;
70                 }
71         });
72         return $canedit;
73 } #}}}
75 sub decode_form_utf8 ($) { #{{{
76         my $form = shift;
77         foreach my $f ($form->field) {
78                 next if Encode::is_utf8(scalar $form->field($f));
79                 $form->field(name  => $f,
80                              value => decode_utf8($form->field($f)),
81                              force => 1,
82                             );
83         }
84 } #}}}
86 sub cgi_recentchanges ($) { #{{{
87         my $q=shift;
88         
89         # Optimisation: building recentchanges means calculating lots of
90         # links. Memoizing htmllink speeds it up a lot (can't be memoized
91         # during page builds as the return values may change, but they
92         # won't here.)
93         eval q{use Memoize};
94         error($@) if $@;
95         memoize("htmllink");
97         eval q{use Time::Duration};
98         error($@) if $@;
100         my $changelog=[rcs_recentchanges(100)];
101         foreach my $change (@$changelog) {
102                 $change->{when} = concise(ago($change->{when}));
104                 $change->{user} = userlink($change->{user});
106                 my $is_excess = exists $change->{pages}[10]; # limit pages to first 10
107                 delete @{$change->{pages}}[10 .. @{$change->{pages}}] if $is_excess;
108                 $change->{pages} = [
109                         map {
110                                 $_->{link} = htmllink("", "", $_->{page},
111                                         noimageinline => 1,
112                                         linktext => pagetitle($_->{page}));
113                                 $_;
114                         } @{$change->{pages}}
115                 ];
116                 push @{$change->{pages}}, { link => '...' } if $is_excess;
117         }
119         my $template=template("recentchanges.tmpl"); 
120         $template->param(
121                 title => "RecentChanges",
122                 indexlink => indexlink(),
123                 wikiname => $config{wikiname},
124                 changelog => $changelog,
125                 baseurl => baseurl(),
126         );
127         run_hooks(pagetemplate => sub {
128                 shift->(page => "", destpage => "", template => $template);
129         });
130         print $q->header(-charset => 'utf-8'), $template->output;
131 } #}}}
133 # Check if the user is signed in. If not, redirect to the signin form and
134 # save their place to return to later.
135 sub needsignin ($$) { #{{{
136         my $q=shift;
137         my $session=shift;
139         if (! defined $session->param("name") ||
140             ! userinfo_get($session->param("name"), "regdate")) {
141                 if (! defined $session->param("postsignin")) {
142                         $session->param(postsignin => $ENV{QUERY_STRING});
143                 }
144                 cgi_signin($q, $session);
145                 cgi_savesession($session);
146                 exit;
147         }
148 } #}}}  
150 sub cgi_signin ($$) { #{{{
151         my $q=shift;
152         my $session=shift;
154         eval q{use CGI::FormBuilder};
155         error($@) if $@;
156         my $form = CGI::FormBuilder->new(
157                 title => "signin",
158                 name => "signin",
159                 charset => "utf-8",
160                 method => 'POST',
161                 required => 'NONE',
162                 javascript => 0,
163                 params => $q,
164                 action => $config{cgiurl},
165                 header => 0,
166                 template => {type => 'div'},
167                 stylesheet => baseurl()."style.css",
168         );
169         my $buttons=["Login"];
170         
171         if ($q->param("do") ne "signin" && !$form->submitted) {
172                 $form->text(gettext("You need to log in first."));
173         }
174         $form->field(name => "do", type => "hidden", value => "signin",
175                 force => 1);
176         
177         decode_form_utf8($form);
178         
179         run_hooks(formbuilder_setup => sub {
180                 shift->(form => $form, cgi => $q, session => $session,
181                         buttons => $buttons);
182         });
184         if ($form->submitted) {
185                 $form->validate;
186         }
188         showform($form, $buttons, $session, $q);
189 } #}}}
191 sub cgi_postsignin ($$) { #{{{
192         my $q=shift;
193         my $session=shift;
194         
195         # Continue with whatever was being done before the signin process.
196         if (defined $session->param("postsignin")) {
197                 my $postsignin=CGI->new($session->param("postsignin"));
198                 $session->clear("postsignin");
199                 cgi($postsignin, $session);
200                 cgi_savesession($session);
201                 exit;
202         }
203         else {
204                 # This can occur, for example, if a user went to the signin
205                 # url via a bookmark.
206                 redirect($q, $config{url});
207         }
208 } #}}}
210 sub cgi_prefs ($$) { #{{{
211         my $q=shift;
212         my $session=shift;
214         needsignin($q, $session);
216         eval q{use CGI::FormBuilder};
217         error($@) if $@;
218         my $form = CGI::FormBuilder->new(
219                 title => "preferences",
220                 name => "preferences",
221                 header => 0,
222                 charset => "utf-8",
223                 method => 'POST',
224                 validate => {
225                         email => 'EMAIL',
226                 },
227                 required => 'NONE',
228                 javascript => 0,
229                 params => $q,
230                 action => $config{cgiurl},
231                 template => {type => 'div'},
232                 stylesheet => baseurl()."style.css",
233                 fieldsets => [
234                         [login => gettext("Login")],
235                         [preferences => gettext("Preferences")],
236                         [admin => gettext("Admin")]
237                 ],
238         );
239         my $buttons=["Save Preferences", "Logout", "Cancel"];
241         decode_form_utf8($form);
243         run_hooks(formbuilder_setup => sub {
244                 shift->(form => $form, cgi => $q, session => $session,
245                         buttons => $buttons);
246         });
247         
248         $form->field(name => "do", type => "hidden");
249         $form->field(name => "email", size => 50, fieldset => "preferences");
250         $form->field(name => "subscriptions", size => 50,
251                 fieldset => "preferences",
252                 comment => "(".htmllink("", "", "ikiwiki/PageSpec", noimageinline => 1).")");
253         $form->field(name => "banned_users", size => 50,
254                 fieldset => "admin");
255         
256         my $user_name=$session->param("name");
257         if (! is_admin($user_name)) {
258                 $form->field(name => "banned_users", type => "hidden");
259         }
261         if (! $form->submitted) {
262                 $form->field(name => "email", force => 1,
263                         value => userinfo_get($user_name, "email"));
264                 $form->field(name => "subscriptions", force => 1,
265                         value => userinfo_get($user_name, "subscriptions"));
266                 if (is_admin($user_name)) {
267                         $form->field(name => "banned_users", force => 1,
268                                 value => join(" ", get_banned_users()));
269                 }
270         }
271         
272         if ($form->submitted eq 'Logout') {
273                 $session->delete();
274                 redirect($q, $config{url});
275                 return;
276         }
277         elsif ($form->submitted eq 'Cancel') {
278                 redirect($q, $config{url});
279                 return;
280         }
281         elsif ($form->submitted eq 'Save Preferences' && $form->validate) {
282                 foreach my $field (qw(email subscriptions)) {
283                         if (defined $form->field($field) && length $form->field($field)) {
284                                 userinfo_set($user_name, $field, $form->field($field)) ||
285                                         error("failed to set $field");
286                         }
287                 }
288                 if (is_admin($user_name)) {
289                         set_banned_users(grep { ! is_admin($_) }
290                                         split(' ',
291                                                 $form->field("banned_users"))) ||
292                                 error("failed saving changes");
293                 }
294                 $form->text(gettext("Preferences saved."));
295         }
296         
297         showform($form, $buttons, $session, $q);
298 } #}}}
300 sub cgi_editpage ($$) { #{{{
301         my $q=shift;
302         my $session=shift;
304         my @fields=qw(do rcsinfo subpage from page type editcontent comments);
305         my @buttons=("Save Page", "Preview", "Cancel");
306         
307         eval q{use CGI::FormBuilder};
308         error($@) if $@;
309         my $form = CGI::FormBuilder->new(
310                 title => "editpage",
311                 fields => \@fields,
312                 charset => "utf-8",
313                 method => 'POST',
314                 validate => {
315                         editcontent => '/.+/',
316                 },
317                 required => [qw{editcontent}],
318                 javascript => 0,
319                 params => $q,
320                 action => $config{cgiurl},
321                 header => 0,
322                 table => 0,
323                 template => scalar template_params("editpage.tmpl"),
324                 wikiname => $config{wikiname},
325         );
326         
327         decode_form_utf8($form);
328         
329         run_hooks(formbuilder_setup => sub {
330                 shift->(form => $form, cgi => $q, session => $session,
331                         buttons => \@buttons);
332         });
333         
334         # This untaint is safe because titlepage removes any problematic
335         # characters.
336         my ($page)=$form->field('page');
337         $page=titlepage(possibly_foolish_untaint($page));
338         if (! defined $page || ! length $page ||
339             file_pruned($page, $config{srcdir}) || $page=~/^\//) {
340                 error("bad page name");
341         }
342         
343         my $from;
344         if (defined $form->field('from')) {
345                 ($from)=$form->field('from')=~/$config{wiki_file_regexp}/;
346         }
347         
348         my $file;
349         my $type;
350         if (exists $pagesources{$page} && $form->field("do") ne "create") {
351                 $file=$pagesources{$page};
352                 $type=pagetype($file);
353                 if (! defined $type) {
354                         error(sprintf(gettext("%s is not an editable page"), $page));
355                 }
356                 if (! $form->submitted) {
357                         $form->field(name => "rcsinfo",
358                                 value => rcs_prepedit($file), force => 1);
359                 }
360         }
361         else {
362                 $type=$form->param('type');
363                 if (defined $type && length $type && $hooks{htmlize}{$type}) {
364                         $type=possibly_foolish_untaint($type);
365                 }
366                 elsif (defined $from) {
367                         # favor the type of linking page
368                         $type=pagetype($pagesources{$from});
369                 }
370                 $type=$config{default_pageext} unless defined $type;
371                 $file=$page.".".$type;
372                 if (! $form->submitted) {
373                         $form->field(name => "rcsinfo", value => "", force => 1);
374                 }
375         }
377         $form->field(name => "do", type => 'hidden');
378         $form->field(name => "from", type => 'hidden');
379         $form->field(name => "rcsinfo", type => 'hidden');
380         $form->field(name => "subpage", type => 'hidden');
381         $form->field(name => "page", value => pagetitle($page, 1), force => 1);
382         $form->field(name => "type", value => $type, force => 1);
383         $form->field(name => "comments", type => "text", size => 80);
384         $form->field(name => "editcontent", type => "textarea", rows => 20,
385                 cols => 80);
386         $form->tmpl_param("can_commit", $config{rcs});
387         $form->tmpl_param("indexlink", indexlink());
388         $form->tmpl_param("helponformattinglink",
389                 htmllink("", "", "ikiwiki/formatting",
390                         noimageinline => 1,
391                         linktext => "FormattingHelp"));
392         $form->tmpl_param("baseurl", baseurl());
393         
394         if ($form->submitted eq "Cancel") {
395                 if ($form->field("do") eq "create" && defined $from) {
396                         redirect($q, "$config{url}/".htmlpage($from));
397                 }
398                 elsif ($form->field("do") eq "create") {
399                         redirect($q, $config{url});
400                 }
401                 else {
402                         redirect($q, "$config{url}/".htmlpage($page));
403                 }
404                 return;
405         }
406         elsif ($form->submitted eq "Preview") {
407                 my $content=$form->field('editcontent');
408                 run_hooks(editcontent => sub {
409                         $content=shift->(
410                                 content => $content,
411                                 page => $page,
412                                 cgi => $q,
413                                 session => $session,
414                         );
415                 });
416                 $form->tmpl_param("page_preview",
417                         htmlize($page, $type,
418                         linkify($page, "",
419                         preprocess($page, $page,
420                         filter($page, $page, $content), 0, 1))));
421         }
422         elsif ($form->submitted eq "Save Page") {
423                 $form->tmpl_param("page_preview", "");
424         }
425         $form->tmpl_param("page_conflict", "");
426         
427         if ($form->submitted ne "Save Page" || ! $form->validate) {
428                 if ($form->field("do") eq "create") {
429                         my @page_locs;
430                         my $best_loc;
431                         if (! defined $from || ! length $from ||
432                             $from ne $form->field('from') ||
433                             file_pruned($from, $config{srcdir}) ||
434                             $from=~/^\// ||
435                             $form->submitted eq "Preview") {
436                                 @page_locs=$best_loc=$page;
437                         }
438                         else {
439                                 my $dir=$from."/";
440                                 $dir=~s![^/]+/+$!!;
441                                 
442                                 if ((defined $form->field('subpage') && length $form->field('subpage')) ||
443                                     $page eq gettext('discussion')) {
444                                         $best_loc="$from/$page";
445                                 }
446                                 else {
447                                         $best_loc=$dir.$page;
448                                 }
449                                 
450                                 push @page_locs, $dir.$page;
451                                 push @page_locs, "$from/$page";
452                                 while (length $dir) {
453                                         $dir=~s![^/]+/+$!!;
454                                         push @page_locs, $dir.$page;
455                                 }
456                         }
457                         push @page_locs, "$config{userdir}/$page"
458                                 if length $config{userdir};
460                         @page_locs = grep {
461                                 ! exists $pagecase{lc $_}
462                         } @page_locs;
463                         if (! @page_locs) {
464                                 # hmm, someone else made the page in the
465                                 # meantime?
466                                 redirect($q, "$config{url}/".htmlpage($page));
467                                 return;
468                         }
470                         my @editable_locs = grep {
471                                 check_canedit($_, $q, $session, 1)
472                         } @page_locs;
473                         if (! @editable_locs) {
474                                 # let it throw an error this time
475                                 map { check_canedit($_, $q, $session) } @page_locs;
476                         }
477                         
478                         my @page_types;
479                         if (exists $hooks{htmlize}) {
480                                 @page_types=keys %{$hooks{htmlize}};
481                         }
482                         
483                         $form->tmpl_param("page_select", 1);
484                         $form->field(name => "page", type => 'select',
485                                 options => [ map { pagetitle($_, 1) } @editable_locs ],
486                                 value => pagetitle($best_loc, 1));
487                         $form->field(name => "type", type => 'select',
488                                 options => \@page_types);
489                         $form->title(sprintf(gettext("creating %s"), pagetitle($page)));
490                         
491                 }
492                 elsif ($form->field("do") eq "edit") {
493                         check_canedit($page, $q, $session);
494                         if (! defined $form->field('editcontent') || 
495                             ! length $form->field('editcontent')) {
496                                 my $content="";
497                                 if (exists $pagesources{$page}) {
498                                         $content=readfile(srcfile($pagesources{$page}));
499                                         $content=~s/\n/\r\n/g;
500                                 }
501                                 $form->field(name => "editcontent", value => $content,
502                                         force => 1);
503                         }
504                         $form->tmpl_param("page_select", 0);
505                         $form->field(name => "page", type => 'hidden');
506                         $form->field(name => "type", type => 'hidden');
507                         $form->title(sprintf(gettext("editing %s"), pagetitle($page)));
508                 }
509                 
510                 showform($form, \@buttons, $session, $q);
511                 saveindex();
512         }
513         else {
514                 # save page
515                 check_canedit($page, $q, $session);
517                 my $exists=-e "$config{srcdir}/$file";
519                 if ($form->field("do") ne "create" && ! $exists &&
520                     ! eval { srcfile($file) }) {
521                         $form->tmpl_param("page_gone", 1);
522                         $form->field(name => "do", value => "create", force => 1);
523                         $form->tmpl_param("page_select", 0);
524                         $form->field(name => "page", type => 'hidden');
525                         $form->field(name => "type", type => 'hidden');
526                         $form->title(sprintf(gettext("editing %s"), $page));
527                         showform($form, \@buttons, $session, $q);
528                         return;
529                 }
530                 elsif ($form->field("do") eq "create" && $exists) {
531                         $form->tmpl_param("creation_conflict", 1);
532                         $form->field(name => "do", value => "edit", force => 1);
533                         $form->tmpl_param("page_select", 0);
534                         $form->field(name => "page", type => 'hidden');
535                         $form->field(name => "type", type => 'hidden');
536                         $form->title(sprintf(gettext("editing %s"), $page));
537                         $form->field("editcontent", 
538                                 value => readfile("$config{srcdir}/$file").
539                                          "\n\n\n".$form->field("editcontent"),
540                                 force => 1);
541                         showform($form, \@buttons, $session, $q);
542                         return;
543                 }
544                 
545                 my $content=$form->field('editcontent');
546                 run_hooks(editcontent => sub {
547                         $content=shift->(
548                                 content => $content,
549                                 page => $page,
550                                 cgi => $q,
551                                 session => $session,
552                         );
553                 });
554                 $content=~s/\r\n/\n/g;
555                 $content=~s/\r/\n/g;
557                 $config{cgi}=0; # avoid cgi error message
558                 eval { writefile($file, $config{srcdir}, $content) };
559                 $config{cgi}=1;
560                 if ($@) {
561                         $form->field(name => "rcsinfo", value => rcs_prepedit($file),
562                                 force => 1);
563                         $form->tmpl_param("failed_save", 1);
564                         $form->tmpl_param("error_message", $@);
565                         $form->field("editcontent", value => $content, force => 1);
566                         $form->tmpl_param("page_select", 0);
567                         $form->field(name => "page", type => 'hidden');
568                         $form->field(name => "type", type => 'hidden');
569                         $form->title(sprintf(gettext("editing %s"), $page));
570                         showform($form, \@buttons, $session, $q);
571                         return;
572                 }
573                 
574                 my $conflict;
575                 if ($config{rcs}) {
576                         my $message="";
577                         if (defined $form->field('comments') &&
578                             length $form->field('comments')) {
579                                 $message=$form->field('comments');
580                         }
581                         
582                         if (! $exists) {
583                                 rcs_add($file);
584                         }
586                         # Prevent deadlock with post-commit hook by
587                         # signaling to it that it should not try to
588                         # do anything (except send commit mails).
589                         disable_commit_hook();
590                         $conflict=rcs_commit($file, $message,
591                                 $form->field("rcsinfo"),
592                                 $session->param("name"), $ENV{REMOTE_ADDR});
593                         enable_commit_hook();
594                         rcs_update();
595                 }
596                 
597                 # Refresh even if there was a conflict, since other changes
598                 # may have been committed while the post-commit hook was
599                 # disabled.
600                 require IkiWiki::Render;
601                 # Reload index, since the first time it's loaded is before
602                 # the wiki is locked, and things may have changed in the
603                 # meantime.
604                 loadindex();
605                 refresh();
606                 saveindex();
608                 if (defined $conflict) {
609                         $form->field(name => "rcsinfo", value => rcs_prepedit($file),
610                                 force => 1);
611                         $form->tmpl_param("page_conflict", 1);
612                         $form->field("editcontent", value => $conflict, force => 1);
613                         $form->field("do", "edit", force => 1);
614                         $form->tmpl_param("page_select", 0);
615                         $form->field(name => "page", type => 'hidden');
616                         $form->field(name => "type", type => 'hidden');
617                         $form->title(sprintf(gettext("editing %s"), $page));
618                         showform($form, \@buttons, $session, $q);
619                         return;
620                 }
621                 else {
622                         # The trailing question mark tries to avoid broken
623                         # caches and get the most recent version of the page.
624                         redirect($q, "$config{url}/".htmlpage($page)."?updated");
625                 }
626         }
627 } #}}}
629 sub cgi_getsession ($) { #{{{
630         my $q=shift;
632         eval q{use CGI::Session};
633         CGI::Session->name("ikiwiki_session_".encode_utf8($config{wikiname}));
634         
635         my $oldmask=umask(077);
636         my $session = CGI::Session->new("driver:DB_File", $q,
637                 { FileName => "$config{wikistatedir}/sessions.db" });
638         umask($oldmask);
640         return $session;
641 } #}}}
643 sub cgi_savesession ($) { #{{{
644         my $session=shift;
646         # Force session flush with safe umask.
647         my $oldmask=umask(077);
648         $session->flush;
649         umask($oldmask);
650 } #}}}
652 sub cgi (;$$) { #{{{
653         my $q=shift;
654         my $session=shift;
656         if (! $q) {
657                 eval q{use CGI};
658                 error($@) if $@;
659         
660                 $q=CGI->new;
661         
662                 run_hooks(cgi => sub { shift->($q) });
663         }
665         my $do=$q->param('do');
666         if (! defined $do || ! length $do) {
667                 my $error = $q->cgi_error;
668                 if ($error) {
669                         error("Request not processed: $error");
670                 }
671                 else {
672                         error("\"do\" parameter missing");
673                 }
674         }
675         
676         # Things that do not need a session.
677         if ($do eq 'recentchanges') {
678                 cgi_recentchanges($q);
679                 return;
680         }
682         # Need to lock the wiki before getting a session.
683         lockwiki();
684         
685         if (! $session) {
686                 $session=cgi_getsession($q);
687         }
688         
689         # Auth hooks can sign a user in.
690         if ($do ne 'signin' && ! defined $session->param("name")) {
691                 run_hooks(auth => sub {
692                         shift->($q, $session)
693                 });
694                 if (defined $session->param("name")) {
695                         # Make sure whatever user was authed is in the
696                         # userinfo db.
697                         if (! userinfo_get($session->param("name"), "regdate")) {
698                                 userinfo_setall($session->param("name"), {
699                                         email => "",
700                                         password => "",
701                                         regdate => time,
702                                 }) || error("failed adding user");
703                         }
704                 }
705         }
706         
707         if (defined $session->param("name") &&
708             userinfo_get($session->param("name"), "banned")) {
709                 print $q->header(-status => "403 Forbidden");
710                 $session->delete();
711                 print gettext("You are banned.");
712                 cgi_savesession($session);
713         }
715         run_hooks(sessioncgi => sub { shift->($q, $session) });
717         if ($do eq 'signin') {
718                 cgi_signin($q, $session);
719                 cgi_savesession($session);
720         }
721         elsif (defined $session->param("postsignin")) {
722                 cgi_postsignin($q, $session);
723         }
724         elsif ($do eq 'prefs') {
725                 cgi_prefs($q, $session);
726         }
727         elsif ($do eq 'create' || $do eq 'edit') {
728                 cgi_editpage($q, $session);
729         }
730         elsif ($do eq 'postsignin') {
731                 error(gettext("login failed, perhaps you need to turn on cookies?"));
732         }
733         else {
734                 error("unknown do parameter");
735         }
736 } #}}}
738 sub userlink ($) { #{{{
739         my $user=shift;
741         eval q{use CGI 'escapeHTML'};
742         error($@) if $@;
743         if ($user =~ m!^https?://! &&
744             eval q{use Net::OpenID::VerifiedIdentity; 1} && !$@) {
745                 # Munge user-urls, as used by eg, OpenID.
746                 my $oid=Net::OpenID::VerifiedIdentity->new(identity => $user);
747                 my $display=$oid->display;
748                 # Convert "user.somehost.com" to "user [somehost.com]".
749                 if ($display !~ /\[/) {
750                         $display=~s/^(.*?)\.([^.]+\.[a-z]+)$/$1 [$2]/;
751                 }
752                 # Convert "http://somehost.com/user" to "user [somehost.com]".
753                 if ($display !~ /\[/) {
754                         $display=~s/^https?:\/\/(.+)\/([^\/]+)$/$2 [$1]/;
755                 }
756                 $display=~s!^https?://!!; # make sure this is removed
757                 return "<a href=\"$user\">".escapeHTML($display)."</a>";
758         }
759         else {
760                 return htmllink("", "", escapeHTML(
761                         length $config{userdir} ? $config{userdir}."/".$user : $user
762                 ), noimageinline => 1);
763         }
764 } #}}}