]> git.vanrenterghem.biz Git - git.ikiwiki.info.git/blob - IkiWiki/Plugin/meta.pm
mention prefix_directives
[git.ikiwiki.info.git] / IkiWiki / Plugin / meta.pm
1 #!/usr/bin/perl
2 # Ikiwiki metadata plugin.
3 package IkiWiki::Plugin::meta;
5 use warnings;
6 use strict;
7 use IkiWiki 2.00;
9 my %meta;
10 my %title;
11 my %permalink;
12 my %author;
13 my %authorurl;
14 my %license;
15 my %copyright;
17 sub import { #{{{
18         hook(type => "needsbuild", id => "meta", call => \&needsbuild);
19         hook(type => "preprocess", id => "meta", call => \&preprocess, scan => 1);
20         hook(type => "pagetemplate", id => "meta", call => \&pagetemplate);
21 } # }}}
23 sub needsbuild (@) { #{{{
24         my $needsbuild=shift;
25         foreach my $page (keys %pagestate) {
26                 if (exists $pagestate{$page}{meta}) {
27                         if (grep { $_ eq $pagesources{$page} } @$needsbuild) {
28                                 # remove state, it will be re-added
29                                 # if the preprocessor directive is still
30                                 # there during the rebuild
31                                 delete $pagestate{$page}{meta};
32                         }
33                 }
34         }
35 }
37 sub scrub ($) { #{{{
38         if (IkiWiki::Plugin::htmlscrubber->can("sanitize")) {
39                 return IkiWiki::Plugin::htmlscrubber::sanitize(content => shift);
40         }
41         else {
42                 return shift;
43         }
44 } #}}}
46 sub htmlize ($$$) { #{{{
47         my $page = shift;
48         my $destpage = shift;
50         return IkiWiki::htmlize($page, pagetype($pagesources{$page}),
51                 IkiWiki::linkify($page, $destpage,
52                 IkiWiki::preprocess($page, $destpage, shift)));
53 }
55 sub preprocess (@) { #{{{
56         return "" unless @_;
57         my %params=@_;
58         my $key=shift;
59         my $value=$params{$key};
60         delete $params{$key};
61         my $page=$params{page};
62         delete $params{page};
63         my $destpage=$params{destpage};
64         delete $params{destpage};
65         delete $params{preview};
67         eval q{use HTML::Entities};
68         # Always decode, even if encoding later, since it might not be
69         # fully encoded.
70         $value=decode_entities($value);
72         # Metadata collection that needs to happen during the scan pass.
73         if ($key eq 'title') {
74                 $title{$page}=HTML::Entities::encode_numeric($value);
75         }
76         elsif ($key eq 'license') {
77                 push @{$meta{$page}}, '<link rel="license" href="#page_license" />';
78                 $license{$page}=$value;
79                 return "";
80         }
81         elsif ($key eq 'copyright') {
82                 push @{$meta{$page}}, '<link rel="copyright" href="#page_copyright" />';
83                 $copyright{$page}=$value;
84                 return "";
85         }
86         elsif ($key eq 'link' && ! %params) {
87                 # hidden WikiLink
88                 push @{$links{$page}}, $value;
89                 return "";
90         }
91         elsif ($key eq 'author') {
92                 $author{$page}=$value;
93                 # fallthorough
94         }
95         elsif ($key eq 'authorurl') {
96                 $authorurl{$page}=$value;
97                 # fallthrough
98         }
100         if (! defined wantarray) {
101                 # avoid collecting duplicate data during scan pass
102                 return;
103         }
105         # Metadata collection that happens only during preprocessing pass.
106         if ($key eq 'date') {
107                 eval q{use Date::Parse};
108                 if (! $@) {
109                         my $time = str2time($value);
110                         $IkiWiki::pagectime{$page}=$time if defined $time;
111                 }
112         }
113         elsif ($key eq 'permalink') {
114                 $permalink{$page}=$value;
115                 push @{$meta{$page}}, scrub('<link rel="bookmark" href="'.encode_entities($value).'" />');
116         }
117         elsif ($key eq 'stylesheet') {
118                 my $rel=exists $params{rel} ? $params{rel} : "alternate stylesheet";
119                 my $title=exists $params{title} ? $params{title} : $value;
120                 # adding .css to the value prevents using any old web
121                 # editable page as a stylesheet
122                 my $stylesheet=bestlink($page, $value.".css");
123                 if (! length $stylesheet) {
124                         return "[[meta ".gettext("stylesheet not found")."]]";
125                 }
126                 push @{$meta{$page}}, '<link href="'.urlto($stylesheet, $page).
127                         '" rel="'.encode_entities($rel).
128                         '" title="'.encode_entities($title).
129                         "\" type=\"text/css\" />";
130         }
131         elsif ($key eq 'openid') {
132                 if (exists $params{server}) {
133                         push @{$meta{$page}}, '<link href="'.encode_entities($params{server}).
134                                 '" rel="openid.server" />';
135                 }
136                 push @{$meta{$page}}, '<link href="'.encode_entities($value).
137                         '" rel="openid.delegate" />';
138         }
139         elsif ($key eq 'redir') {
140                 return "" if $page ne $destpage;
141                 my $safe=0;
142                 if ($value !~ /^\w+:\/\//) {
143                         my ($redir_page, $redir_anchor) = split /\#/, $value;
145                         add_depends($page, $redir_page);
146                         my $link=bestlink($page, $redir_page);
147                         if (! length $link) {
148                                 return "[[meta ".gettext("redir page not found")."]]";
149                         }
151                         $value=urlto($link, $page);
152                         $value.='#'.$redir_anchor if defined $redir_anchor;
153                         $safe=1;
155                         # redir cycle detection
156                         $pagestate{$page}{meta}{redir}=$link;
157                         my $at=$page;
158                         my %seen;
159                         while (exists $pagestate{$at}{meta}{redir}) {
160                                 if ($seen{$at}) {
161                                         return "[[meta ".gettext("redir cycle is not allowed")."]]";
162                                 }
163                                 $seen{$at}=1;
164                                 $at=$pagestate{$at}{meta}{redir};
165                         }
166                 }
167                 else {
168                         $value=encode_entities($value);
169                 }
170                 my $delay=int(exists $params{delay} ? $params{delay} : 0);
171                 my $redir="<meta http-equiv=\"refresh\" content=\"$delay; URL=$value\">";
172                 if (! $safe) {
173                         $redir=scrub($redir);
174                 }
175                 push @{$meta{$page}}, $redir;
176         }
177         elsif ($key eq 'link') {
178                 if (%params) {
179                         push @{$meta{$page}}, scrub("<link href=\"".encode_entities($value)."\" ".
180                                 join(" ", map {
181                                         encode_entities($_)."=\"".encode_entities(decode_entities($params{$_}))."\""
182                                 } keys %params).
183                                 " />\n");
184                 }
185         }
186         else {
187                 push @{$meta{$page}}, scrub('<meta name="'.encode_entities($key).
188                         '" content="'.encode_entities($value).'" />');
189         }
191         return "";
192 } # }}}
194 sub pagetemplate (@) { #{{{
195         my %params=@_;
196         my $page=$params{page};
197         my $destpage=$params{destpage};
198         my $template=$params{template};
200         if (exists $meta{$page} && $template->query(name => "meta")) {
201                 # avoid duplicate meta lines
202                 my %seen;
203                 $template->param(meta => join("\n", grep { (! $seen{$_}) && ($seen{$_}=1) } @{$meta{$page}}));
204         }
205         if (exists $title{$page} && $template->query(name => "title")) {
206                 $template->param(title => $title{$page});
207                 $template->param(title_overridden => 1);
208         }
209         $template->param(permalink => $permalink{$page})
210                 if exists $permalink{$page} && $template->query(name => "permalink");
211         $template->param(author => $author{$page})
212                 if exists $author{$page} && $template->query(name => "author");
213         $template->param(authorurl => $authorurl{$page})
214                 if exists $authorurl{$page} && $template->query(name => "authorurl");
216         if (exists $license{$page} && $template->query(name => "license") &&
217             ($page eq $destpage || ! exists $license{$destpage} ||
218              $license{$page} ne $license{$destpage})) {
219                 $template->param(license => htmlize($page, $destpage, $license{$page}));
220         }
221         if (exists $copyright{$page} && $template->query(name => "copyright") &&
222             ($page eq $destpage || ! exists $copyright{$destpage} ||
223              $copyright{$page} ne $copyright{$destpage})) {
224                 $template->param(copyright => htmlize($page, $destpage, $copyright{$page}));
225         }
226 } # }}}