2007-12-12 07:52:26 +01:00
|
|
|
#!/usr/bin/perl
|
|
|
|
package IkiWiki::Plugin::edittemplate;
|
|
|
|
|
|
|
|
use warnings;
|
|
|
|
use strict;
|
2008-12-23 22:34:19 +01:00
|
|
|
use IkiWiki 3.00;
|
2007-12-12 08:45:44 +01:00
|
|
|
use HTML::Template;
|
|
|
|
use Encode;
|
2007-12-12 07:52:26 +01:00
|
|
|
|
2008-12-17 21:22:16 +01:00
|
|
|
sub import {
|
2008-08-03 22:40:12 +02:00
|
|
|
hook(type => "getsetup", id => "edittemplate",
|
|
|
|
call => \&getsetup);
|
2007-12-12 07:52:26 +01:00
|
|
|
hook(type => "needsbuild", id => "edittemplate",
|
|
|
|
call => \&needsbuild);
|
|
|
|
hook(type => "preprocess", id => "edittemplate",
|
|
|
|
call => \&preprocess);
|
2007-12-12 09:01:15 +01:00
|
|
|
hook(type => "formbuilder", id => "edittemplate",
|
|
|
|
call => \&formbuilder);
|
2008-12-17 21:22:16 +01:00
|
|
|
}
|
2007-12-12 07:52:26 +01:00
|
|
|
|
2008-12-17 21:22:16 +01:00
|
|
|
sub getsetup () {
|
2008-08-03 22:40:12 +02:00
|
|
|
return
|
|
|
|
plugin => {
|
|
|
|
safe => 1,
|
|
|
|
rebuild => undef,
|
2010-02-12 10:22:15 +01:00
|
|
|
section => "web",
|
2008-08-03 22:40:12 +02:00
|
|
|
},
|
2008-12-17 21:22:16 +01:00
|
|
|
}
|
2008-08-03 22:40:12 +02:00
|
|
|
|
2008-12-17 21:22:16 +01:00
|
|
|
sub needsbuild (@) {
|
2007-12-12 07:52:26 +01:00
|
|
|
my $needsbuild=shift;
|
|
|
|
|
|
|
|
foreach my $page (keys %pagestate) {
|
|
|
|
if (exists $pagestate{$page}{edittemplate}) {
|
2008-01-29 23:36:25 +01:00
|
|
|
if (exists $pagesources{$page} &&
|
|
|
|
grep { $_ eq $pagesources{$page} } @$needsbuild) {
|
2007-12-12 07:52:26 +01:00
|
|
|
# remove state, it will be re-added
|
|
|
|
# if the preprocessor directive is still
|
|
|
|
# there during the rebuild
|
|
|
|
delete $pagestate{$page}{edittemplate};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-09-07 18:08:59 +02:00
|
|
|
|
|
|
|
return $needsbuild;
|
2008-12-17 21:22:16 +01:00
|
|
|
}
|
2007-12-12 07:52:26 +01:00
|
|
|
|
2008-12-17 21:22:16 +01:00
|
|
|
sub preprocess (@) {
|
2007-12-12 07:52:26 +01:00
|
|
|
my %params=@_;
|
|
|
|
|
|
|
|
return "" if $params{page} ne $params{destpage};
|
|
|
|
|
|
|
|
if (! exists $params{template} || ! length($params{template})) {
|
2008-07-13 21:05:34 +02:00
|
|
|
error gettext("template not specified")
|
2007-12-12 07:52:26 +01:00
|
|
|
}
|
|
|
|
if (! exists $params{match} || ! length($params{match})) {
|
2008-07-13 21:05:34 +02:00
|
|
|
error gettext("match not specified")
|
2007-12-12 07:52:26 +01:00
|
|
|
}
|
|
|
|
|
2008-09-27 20:14:36 +02:00
|
|
|
my $link=linkpage($params{template});
|
2010-06-13 04:43:34 +02:00
|
|
|
add_depends($params{page}, $link, deptype("presence"));
|
2009-10-18 19:47:30 +02:00
|
|
|
my $bestlink=bestlink($params{page}, $link);
|
2010-06-13 04:43:34 +02:00
|
|
|
if (! length $bestlink) {
|
|
|
|
add_depends($params{page}, "templates/$link", deptype("presence"));
|
|
|
|
$link="/templates/".$link;
|
|
|
|
$bestlink=bestlink($params{page}, $link);
|
|
|
|
}
|
2009-10-18 19:47:30 +02:00
|
|
|
$pagestate{$params{page}}{edittemplate}{$params{match}}=$bestlink;
|
2007-12-12 07:52:26 +01:00
|
|
|
|
2010-06-13 04:20:22 +02:00
|
|
|
return "" if ($params{silent} && IkiWiki::yesno($params{silent})) &&
|
|
|
|
length $bestlink;
|
2008-09-20 22:21:04 +02:00
|
|
|
return sprintf(gettext("edittemplate %s registered for %s"),
|
2008-09-21 05:00:19 +02:00
|
|
|
htmllink($params{page}, $params{destpage}, $link),
|
|
|
|
$params{match});
|
2008-12-17 21:22:16 +01:00
|
|
|
}
|
2007-12-12 07:52:26 +01:00
|
|
|
|
2008-12-17 21:22:16 +01:00
|
|
|
sub formbuilder (@) {
|
2007-12-12 07:52:26 +01:00
|
|
|
my %params=@_;
|
|
|
|
my $form=$params{form};
|
2007-12-12 09:35:23 +01:00
|
|
|
|
2008-08-26 02:51:10 +02:00
|
|
|
return if $form->field("do") ne "create" ||
|
2008-09-21 04:51:42 +02:00
|
|
|
(defined $form->field("editcontent") && length $form->field("editcontent"));
|
2008-08-26 02:51:10 +02:00
|
|
|
|
2007-12-12 08:45:44 +01:00
|
|
|
my $page=$form->field("page");
|
|
|
|
|
|
|
|
# The tricky bit here is that $page is probably just the base
|
|
|
|
# page name, without any subdir, but the pagespec for a template
|
|
|
|
# probably does include the subdir (ie, "bugs/*"). We don't know
|
2007-12-12 09:35:23 +01:00
|
|
|
# what subdir the user will pick to put the page in. So, try them
|
|
|
|
# all, starting with the one that was made default.
|
2007-12-12 08:45:44 +01:00
|
|
|
my @page_locs=$page;
|
2007-12-12 09:35:23 +01:00
|
|
|
foreach my $field ($form->field) {
|
|
|
|
if ($field eq 'page') {
|
|
|
|
@page_locs=$field->def_value;
|
2009-10-18 19:56:35 +02:00
|
|
|
|
|
|
|
# FormBuilder is on the bad crack. See #551499
|
|
|
|
my @options=map { ref $_ ? @$_ : $_ } $field->options;
|
|
|
|
|
|
|
|
push @page_locs, @options;
|
2007-12-12 08:45:44 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
foreach my $p (@page_locs) {
|
|
|
|
foreach my $registering_page (keys %pagestate) {
|
|
|
|
if (exists $pagestate{$registering_page}{edittemplate}) {
|
|
|
|
foreach my $pagespec (sort keys %{$pagestate{$registering_page}{edittemplate}}) {
|
|
|
|
if (pagespec_match($p, $pagespec, location => $registering_page)) {
|
2008-09-20 22:23:15 +02:00
|
|
|
my $template=$pagestate{$registering_page}{edittemplate}{$pagespec};
|
2007-12-12 08:45:44 +01:00
|
|
|
$form->field(name => "editcontent",
|
2008-09-20 22:23:15 +02:00
|
|
|
value => filltemplate($template, $page));
|
2010-11-20 19:54:43 +01:00
|
|
|
my $type=pagetype($pagesources{$template})
|
2008-09-20 22:23:15 +02:00
|
|
|
if $pagesources{$template};
|
2010-11-20 19:54:43 +01:00
|
|
|
$form->field(name => "type",
|
|
|
|
value => $type)
|
|
|
|
if defined $type;
|
2007-12-12 08:45:44 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-12-17 21:22:16 +01:00
|
|
|
}
|
2007-12-12 08:45:44 +01:00
|
|
|
|
2008-12-17 21:22:16 +01:00
|
|
|
sub filltemplate ($$) {
|
2007-12-12 08:45:44 +01:00
|
|
|
my $template_page=shift;
|
|
|
|
my $page=shift;
|
|
|
|
|
|
|
|
my $template;
|
|
|
|
eval {
|
2010-04-23 21:11:25 +02:00
|
|
|
# force page name absolute so it doesn't look in templates/
|
|
|
|
$template=template("/".$template_page);
|
2007-12-12 08:45:44 +01:00
|
|
|
};
|
|
|
|
if ($@) {
|
protect $@ whenever a block using $@ is non-trivial
As noted in the Try::Tiny man page, eval/$@ can be quite awkward in
corner cases, because $@ has the same properties and problems as C's
errno. While writing a regression test for definetemplate
in which it couldn't find an appropriate template, I received
<span class="error">Error: failed to process template
<span class="createlink">deftmpl</span> </span>
instead of the intended
<span class="error">Error: failed to process template
<span class="createlink">deftmpl</span> template deftmpl not
found</span>
which turned out to be because the "catch"-analogous block called
gettext before it used $@, and gettext can call define_gettext,
which uses eval.
This commit alters all current "catch"-like blocks that use $@, except
those that just do trivial things with $@ (string interpolation, string
concatenation) and call a function (die, error, print, etc.)
2014-02-21 18:06:36 +01:00
|
|
|
# gettext can clobber $@
|
|
|
|
my $error = $@;
|
2008-07-13 21:05:34 +02:00
|
|
|
# Indicate that the earlier preprocessor directive set
|
|
|
|
# up a template that doesn't work.
|
protect $@ whenever a block using $@ is non-trivial
As noted in the Try::Tiny man page, eval/$@ can be quite awkward in
corner cases, because $@ has the same properties and problems as C's
errno. While writing a regression test for definetemplate
in which it couldn't find an appropriate template, I received
<span class="error">Error: failed to process template
<span class="createlink">deftmpl</span> </span>
instead of the intended
<span class="error">Error: failed to process template
<span class="createlink">deftmpl</span> template deftmpl not
found</span>
which turned out to be because the "catch"-analogous block called
gettext before it used $@, and gettext can call define_gettext,
which uses eval.
This commit alters all current "catch"-like blocks that use $@, except
those that just do trivial things with $@ (string interpolation, string
concatenation) and call a function (die, error, print, etc.)
2014-02-21 18:06:36 +01:00
|
|
|
return "[[!edittemplate ".gettext("failed to process template:")." $error]]";
|
2010-04-23 21:11:25 +02:00
|
|
|
}
|
2007-12-12 08:45:44 +01:00
|
|
|
|
|
|
|
$template->param(name => $page);
|
2007-12-12 07:52:26 +01:00
|
|
|
|
2014-07-01 09:23:13 +02:00
|
|
|
if ($template->query(name => 'uuid')) {
|
2014-07-01 09:29:20 +02:00
|
|
|
my $uuid;
|
|
|
|
if (open(my $fh, "<", "/proc/sys/kernel/random/uuid")) {
|
|
|
|
$uuid = <$fh>;
|
|
|
|
chomp $uuid;
|
|
|
|
close $fh;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
eval {
|
|
|
|
require UUID::Tiny;
|
|
|
|
$uuid = UUID::Tiny::create_uuid_as_string(UUID::Tiny::UUID_V4());
|
|
|
|
};
|
|
|
|
}
|
|
|
|
$template->param(uuid => $uuid);
|
2014-07-01 09:23:13 +02:00
|
|
|
}
|
2013-01-25 06:42:23 +01:00
|
|
|
|
2013-01-27 08:38:46 +01:00
|
|
|
my $time = time();
|
2014-07-01 09:52:48 +02:00
|
|
|
$template->param(time => IkiWiki::date_3339($time));
|
2013-01-27 08:38:46 +01:00
|
|
|
|
2007-12-12 08:45:44 +01:00
|
|
|
return $template->output;
|
2008-12-17 21:22:16 +01:00
|
|
|
}
|
2007-12-12 07:52:26 +01:00
|
|
|
|
|
|
|
1
|