summaryrefslogtreecommitdiff
path: root/IkiWiki/Plugin/editpage.pm
blob: 658d664af38d76b330ae9c27b74e77e5f28172a6 (plain)
  1. #!/usr/bin/perl
  2. package IkiWiki::Plugin::editpage;
  3. use warnings;
  4. use strict;
  5. use IkiWiki;
  6. use open qw{:utf8 :std};
  7. sub import {
  8. hook(type => "getsetup", id => "editpage", call => \&getsetup);
  9. hook(type => "refresh", id => "editpage", call => \&refresh);
  10. hook(type => "sessioncgi", id => "editpage", call => \&IkiWiki::cgi_editpage);
  11. }
  12. sub getsetup () {
  13. return
  14. plugin => {
  15. safe => 1,
  16. rebuild => 1,
  17. },
  18. }
  19. sub refresh () {
  20. if (exists $wikistate{editpage} && exists $wikistate{editpage}{previews}) {
  21. # Expire old preview files after one hour.
  22. my $expire=time - (60 * 60);
  23. my @previews;
  24. foreach my $file (@{$wikistate{editpage}{previews}}) {
  25. my $mtime=(stat("$config{destdir}/$file"))[9];
  26. if (defined $mtime && $mtime <= $expire) {
  27. # Avoid deleting a preview that was later saved.
  28. my $delete=1;
  29. foreach my $page (keys %renderedfiles) {
  30. if (grep { $_ eq $file } @{$renderedfiles{$page}}) {
  31. $delete=0;
  32. }
  33. }
  34. if ($delete) {
  35. debug(sprintf(gettext("removing old preview %s"), $file));
  36. IkiWiki::prune("$config{destdir}/$file");
  37. }
  38. }
  39. elsif (defined $mtime) {
  40. push @previews, $file;
  41. }
  42. }
  43. $wikistate{editpage}{previews}=\@previews;
  44. }
  45. }
  46. # Back to ikiwiki namespace for the rest, this code is very much
  47. # internal to ikiwiki even though it's separated into a plugin,
  48. # and other plugins use the functions below.
  49. package IkiWiki;
  50. sub check_canedit ($$$;$) {
  51. my $page=shift;
  52. my $q=shift;
  53. my $session=shift;
  54. my $nonfatal=shift;
  55. my $canedit;
  56. run_hooks(canedit => sub {
  57. return if defined $canedit;
  58. my $ret=shift->($page, $q, $session);
  59. if (defined $ret) {
  60. if ($ret eq "") {
  61. $canedit=1;
  62. }
  63. elsif (ref $ret eq 'CODE') {
  64. $ret->() unless $nonfatal;
  65. $canedit=0;
  66. }
  67. elsif (defined $ret) {
  68. error($ret) unless $nonfatal;
  69. $canedit=0;
  70. }
  71. }
  72. });
  73. return defined $canedit ? $canedit : 1;
  74. }
  75. sub check_content (@) {
  76. my %params=@_;
  77. return 1 if ! exists $hooks{checkcontent}; # optimisation
  78. if (exists $pagesources{$params{page}}) {
  79. my @diff;
  80. my %old=map { $_ => 1 }
  81. split("\n", readfile(srcfile($pagesources{$params{page}})));
  82. foreach my $line (split("\n", $params{content})) {
  83. push @diff, $line if ! exists $old{$_};
  84. }
  85. $params{content}=join("\n", @diff);
  86. }
  87. my $ok;
  88. run_hooks(checkcontent => sub {
  89. return if defined $ok;
  90. my $ret=shift->(%params);
  91. if (defined $ret) {
  92. if ($ret eq "") {
  93. $ok=1;
  94. }
  95. elsif (ref $ret eq 'CODE') {
  96. $ret->();
  97. $ok=0;
  98. }
  99. elsif (defined $ret) {
  100. error($ret);
  101. }
  102. }
  103. });
  104. return defined $ok ? $ok : 1;
  105. }
  106. sub check_cansave ($$$$) {
  107. my $page=shift;
  108. my $content=shift;
  109. my $q=shift;
  110. my $session=shift;
  111. my $cansave;
  112. run_hooks(cansave => sub {
  113. return if defined $cansave;
  114. my $ret=shift->($page, $content, $q, $session);
  115. if (defined $ret) {
  116. if ($ret eq "") {
  117. $cansave=1;
  118. }
  119. elsif (ref $ret eq 'CODE') {
  120. $ret->();
  121. $cansave=0;
  122. }
  123. else {
  124. error($ret);
  125. $cansave=0;
  126. }
  127. }
  128. });
  129. return $cansave;
  130. }
  131. sub cgi_editpage ($$) {
  132. my $q=shift;
  133. my $session=shift;
  134. my $do=$q->param('do');
  135. return unless $do eq 'create' || $do eq 'edit';
  136. decode_cgi_utf8($q);
  137. my @fields=qw(do rcsinfo subpage from page type editcontent comments);
  138. my @buttons=("Save Page", "Preview", "Cancel");
  139. eval q{use CGI::FormBuilder};
  140. error($@) if $@;
  141. my $form = CGI::FormBuilder->new(
  142. fields => \@fields,
  143. charset => "utf-8",
  144. method => 'POST',
  145. required => [qw{editcontent}],
  146. javascript => 0,
  147. params => $q,
  148. action => $config{cgiurl},
  149. header => 0,
  150. table => 0,
  151. template => scalar template_params("editpage.tmpl"),
  152. );
  153. decode_form_utf8($form);
  154. run_hooks(formbuilder_setup => sub {
  155. shift->(form => $form, cgi => $q, session => $session,
  156. buttons => \@buttons);
  157. });
  158. decode_form_utf8($form);
  159. # This untaint is safe because we check file_pruned and
  160. # wiki_file_regexp.
  161. my ($page)=$form->field('page')=~/$config{wiki_file_regexp}/;
  162. $page=possibly_foolish_untaint($page);
  163. my $absolute=($page =~ s#^/+##);
  164. if (! defined $page || ! length $page ||
  165. file_pruned($page, $config{srcdir})) {
  166. error(gettext("bad page name"));
  167. }
  168. my $baseurl = urlto($page, undef, 1);
  169. my $from;
  170. if (defined $form->field('from')) {
  171. ($from)=$form->field('from')=~/$config{wiki_file_regexp}/;
  172. }
  173. my $file;
  174. my $type;
  175. if (exists $pagesources{$page} && $form->field("do") ne "create") {
  176. $file=$pagesources{$page};
  177. $type=pagetype($file);
  178. if (! defined $type || $type=~/^_/) {
  179. error(sprintf(gettext("%s is not an editable page"), $page));
  180. }
  181. if (! $form->submitted) {
  182. $form->field(name => "rcsinfo",
  183. value => rcs_prepedit($file), force => 1);
  184. }
  185. $form->field(name => "editcontent", validate => '/.*/');
  186. }
  187. else {
  188. $type=$form->param('type');
  189. if (defined $type && length $type && $hooks{htmlize}{$type}) {
  190. $type=possibly_foolish_untaint($type);
  191. }
  192. elsif (defined $from && exists $pagesources{$from}) {
  193. # favor the type of linking page
  194. $type=pagetype($pagesources{$from});
  195. }
  196. $type=$config{default_pageext} unless defined $type;
  197. $file=newpagefile($page, $type);
  198. if (! $form->submitted) {
  199. $form->field(name => "rcsinfo", value => "", force => 1);
  200. }
  201. $form->field(name => "editcontent", validate => '/.+/');
  202. }
  203. $form->field(name => "do", type => 'hidden');
  204. $form->field(name => "sid", type => "hidden", value => $session->id,
  205. force => 1);
  206. $form->field(name => "from", type => 'hidden');
  207. $form->field(name => "rcsinfo", type => 'hidden');
  208. $form->field(name => "subpage", type => 'hidden');
  209. $form->field(name => "page", value => $page, force => 1);
  210. $form->field(name => "type", value => $type, force => 1);
  211. $form->field(name => "comments", type => "text", size => 80);
  212. $form->field(name => "editcontent", type => "textarea", rows => 20,
  213. cols => 80);
  214. $form->tmpl_param("can_commit", $config{rcs});
  215. $form->tmpl_param("indexlink", indexlink());
  216. $form->tmpl_param("helponformattinglink",
  217. htmllink($page, $page, "ikiwiki/formatting",
  218. noimageinline => 1,
  219. linktext => "FormattingHelp"));
  220. if ($form->submitted eq "Cancel") {
  221. if ($form->field("do") eq "create" && defined $from) {
  222. redirect($q, urlto($from, undef, 1));
  223. }
  224. elsif ($form->field("do") eq "create") {
  225. redirect($q, $config{url});
  226. }
  227. else {
  228. redirect($q, urlto($page, undef, 1));
  229. }
  230. exit;
  231. }
  232. elsif ($form->submitted eq "Preview") {
  233. my $new=not exists $pagesources{$page};
  234. if ($new) {
  235. # temporarily record its type
  236. $pagesources{$page}=$page.".".$type;
  237. }
  238. my %wasrendered=map { $_ => 1 } @{$renderedfiles{$page}};
  239. my $content=$form->field('editcontent');
  240. run_hooks(editcontent => sub {
  241. $content=shift->(
  242. content => $content,
  243. page => $page,
  244. cgi => $q,
  245. session => $session,
  246. );
  247. });
  248. my $preview=htmlize($page, $page, $type,
  249. linkify($page, $page,
  250. preprocess($page, $page,
  251. filter($page, $page, $content), 0, 1)));
  252. run_hooks(format => sub {
  253. $preview=shift->(
  254. page => $page,
  255. content => $preview,
  256. );
  257. });
  258. $form->tmpl_param("page_preview", $preview);
  259. if ($new) {
  260. delete $pagesources{$page};
  261. }
  262. # Previewing may have created files on disk.
  263. # Keep a list of these to be deleted later.
  264. my %previews = map { $_ => 1 } @{$wikistate{editpage}{previews}};
  265. foreach my $f (@{$renderedfiles{$page}}) {
  266. $previews{$f}=1 unless $wasrendered{$f};
  267. }
  268. @{$wikistate{editpage}{previews}} = keys %previews;
  269. $renderedfiles{$page}=[keys %wasrendered];
  270. saveindex();
  271. }
  272. elsif ($form->submitted eq "Save Page") {
  273. $form->tmpl_param("page_preview", "");
  274. }
  275. if ($form->submitted ne "Save Page" || ! $form->validate) {
  276. if ($form->field("do") eq "create") {
  277. my @page_locs;
  278. my $best_loc;
  279. if (! defined $from || ! length $from ||
  280. $from ne $form->field('from') ||
  281. file_pruned($from, $config{srcdir}) ||
  282. $from=~/^\// ||
  283. $absolute ||
  284. $form->submitted) {
  285. @page_locs=$best_loc=$page;
  286. }
  287. else {
  288. my $dir=$from."/";
  289. $dir=~s![^/]+/+$!!;
  290. if ((defined $form->field('subpage') && length $form->field('subpage')) ||
  291. $page eq gettext('discussion')) {
  292. $best_loc="$from/$page";
  293. }
  294. else {
  295. $best_loc=$dir.$page;
  296. }
  297. push @page_locs, $dir.$page;
  298. push @page_locs, "$from/$page";
  299. while (length $dir) {
  300. $dir=~s![^/]+/+$!!;
  301. push @page_locs, $dir.$page;
  302. }
  303. push @page_locs, "$config{userdir}/$page"
  304. if length $config{userdir};
  305. }
  306. @page_locs = grep {
  307. ! exists $pagecase{lc $_}
  308. } @page_locs;
  309. if (! @page_locs) {
  310. # hmm, someone else made the page in the
  311. # meantime?
  312. if ($form->submitted eq "Preview") {
  313. # let them go ahead with the edit
  314. # and resolve the conflict at save
  315. # time
  316. @page_locs=$page;
  317. }
  318. else {
  319. redirect($q, urlto($page, undef, 1));
  320. exit;
  321. }
  322. }
  323. my @editable_locs = grep {
  324. check_canedit($_, $q, $session, 1)
  325. } @page_locs;
  326. if (! @editable_locs) {
  327. # let it throw an error this time
  328. map { check_canedit($_, $q, $session) } @page_locs;
  329. }
  330. my @page_types;
  331. if (exists $hooks{htmlize}) {
  332. @page_types=grep { !/^_/ }
  333. keys %{$hooks{htmlize}};
  334. }
  335. $form->tmpl_param("page_select", 1);
  336. $form->field(name => "page", type => 'select',
  337. options => [ map { [ $_, pagetitle($_, 1) ] } @editable_locs ],
  338. value => $best_loc);
  339. $form->field(name => "type", type => 'select',
  340. options => \@page_types);
  341. $form->title(sprintf(gettext("creating %s"), pagetitle($page)));
  342. }
  343. elsif ($form->field("do") eq "edit") {
  344. check_canedit($page, $q, $session);
  345. if (! defined $form->field('editcontent') ||
  346. ! length $form->field('editcontent')) {
  347. my $content="";
  348. if (exists $pagesources{$page}) {
  349. $content=readfile(srcfile($pagesources{$page}));
  350. $content=~s/\n/\r\n/g;
  351. }
  352. $form->field(name => "editcontent", value => $content,
  353. force => 1);
  354. }
  355. $form->tmpl_param("page_select", 0);
  356. $form->field(name => "page", type => 'hidden');
  357. $form->field(name => "type", type => 'hidden');
  358. $form->title(sprintf(gettext("editing %s"), pagetitle($page)));
  359. }
  360. showform($form, \@buttons, $session, $q, forcebaseurl => $baseurl);
  361. }
  362. else {
  363. # save page
  364. check_canedit($page, $q, $session);
  365. checksessionexpiry($q, $session, $q->param('sid'));
  366. my $exists=-e "$config{srcdir}/$file";
  367. if ($form->field("do") ne "create" && ! $exists &&
  368. ! defined srcfile($file, 1)) {
  369. $form->tmpl_param("message", template("editpagegone.tmpl")->output);
  370. $form->field(name => "do", value => "create", force => 1);
  371. $form->tmpl_param("page_select", 0);
  372. $form->field(name => "page", type => 'hidden');
  373. $form->field(name => "type", type => 'hidden');
  374. $form->title(sprintf(gettext("editing %s"), $page));
  375. showform($form, \@buttons, $session, $q, forcebaseurl => $baseurl);
  376. exit;
  377. }
  378. elsif ($form->field("do") eq "create" && $exists) {
  379. $form->tmpl_param("message", template("editcreationconflict.tmpl")->output);
  380. $form->field(name => "do", value => "edit", force => 1);
  381. $form->tmpl_param("page_select", 0);
  382. $form->field(name => "page", type => 'hidden');
  383. $form->field(name => "type", type => 'hidden');
  384. $form->title(sprintf(gettext("editing %s"), $page));
  385. $form->field("editcontent",
  386. value => readfile("$config{srcdir}/$file").
  387. "\n\n\n".$form->field("editcontent"),
  388. force => 1);
  389. showform($form, \@buttons, $session, $q, forcebaseurl => $baseurl);
  390. exit;
  391. }
  392. my $message="";
  393. if (defined $form->field('comments') &&
  394. length $form->field('comments')) {
  395. $message=$form->field('comments');
  396. }
  397. my $content=$form->field('editcontent');
  398. check_cansave($page, $content, $q, $session);
  399. check_content(content => $content, page => $page,
  400. cgi => $q, session => $session,
  401. subject => $message);
  402. run_hooks(editcontent => sub {
  403. $content=shift->(
  404. content => $content,
  405. page => $page,
  406. cgi => $q,
  407. session => $session,
  408. );
  409. });
  410. $content=~s/\r\n/\n/g;
  411. $content=~s/\r/\n/g;
  412. $content.="\n" if $content !~ /\n$/;
  413. $config{cgi}=0; # avoid cgi error message
  414. eval { writefile($file, $config{srcdir}, $content) };
  415. $config{cgi}=1;
  416. if ($@) {
  417. $form->field(name => "rcsinfo", value => rcs_prepedit($file),
  418. force => 1);
  419. my $mtemplate=template("editfailedsave.tmpl");
  420. $mtemplate->param(error_message => $@);
  421. $form->tmpl_param("message", $mtemplate->output);
  422. $form->field("editcontent", value => $content, force => 1);
  423. $form->tmpl_param("page_select", 0);
  424. $form->field(name => "page", type => 'hidden');
  425. $form->field(name => "type", type => 'hidden');
  426. $form->title(sprintf(gettext("editing %s"), $page));
  427. showform($form, \@buttons, $session, $q,
  428. forcebaseurl => $baseurl);
  429. exit;
  430. }
  431. my $conflict;
  432. if ($config{rcs}) {
  433. if (! $exists) {
  434. rcs_add($file);
  435. }
  436. # Prevent deadlock with post-commit hook by
  437. # signaling to it that it should not try to
  438. # do anything.
  439. disable_commit_hook();
  440. $conflict=rcs_commit($file, $message,
  441. $form->field("rcsinfo"),
  442. $session->param("name"), $ENV{REMOTE_ADDR});
  443. enable_commit_hook();
  444. rcs_update();
  445. }
  446. # Refresh even if there was a conflict, since other changes
  447. # may have been committed while the post-commit hook was
  448. # disabled.
  449. require IkiWiki::Render;
  450. refresh();
  451. saveindex();
  452. if (defined $conflict) {
  453. $form->field(name => "rcsinfo", value => rcs_prepedit($file),
  454. force => 1);
  455. $form->tmpl_param("message", template("editconflict.tmpl")->output);
  456. $form->field("editcontent", value => $conflict, force => 1);
  457. $form->field("do", "edit", force => 1);
  458. $form->tmpl_param("page_select", 0);
  459. $form->field(name => "page", type => 'hidden');
  460. $form->field(name => "type", type => 'hidden');
  461. $form->title(sprintf(gettext("editing %s"), $page));
  462. showform($form, \@buttons, $session, $q,
  463. forcebaseurl => $baseurl);
  464. }
  465. else {
  466. # The trailing question mark tries to avoid broken
  467. # caches and get the most recent version of the page.
  468. redirect($q, urlto($page, undef, 1)."?updated");
  469. }
  470. }
  471. exit;
  472. }
  473. 1