]> git.vanrenterghem.biz Git - git.ikiwiki.info.git/blob - IkiWiki/Plugin/mercurial.pm
avoid dying if cannot chdir to an underlaydir
[git.ikiwiki.info.git] / IkiWiki / Plugin / mercurial.pm
1 #!/usr/bin/perl
2 package IkiWiki::Plugin::mercurial;
4 use warnings;
5 use strict;
6 use IkiWiki;
7 use Encode;
8 use open qw{:utf8 :std};
10 sub import {
11         hook(type => "checkconfig", id => "mercurial", call => \&checkconfig);
12         hook(type => "getsetup", id => "mercurial", call => \&getsetup);
13         hook(type => "rcs", id => "rcs_update", call => \&rcs_update);
14         hook(type => "rcs", id => "rcs_prepedit", call => \&rcs_prepedit);
15         hook(type => "rcs", id => "rcs_commit", call => \&rcs_commit);
16         hook(type => "rcs", id => "rcs_commit_staged", call => \&rcs_commit_staged);
17         hook(type => "rcs", id => "rcs_add", call => \&rcs_add);
18         hook(type => "rcs", id => "rcs_remove", call => \&rcs_remove);
19         hook(type => "rcs", id => "rcs_rename", call => \&rcs_rename);
20         hook(type => "rcs", id => "rcs_recentchanges", call => \&rcs_recentchanges);
21         hook(type => "rcs", id => "rcs_diff", call => \&rcs_diff);
22         hook(type => "rcs", id => "rcs_getctime", call => \&rcs_getctime);
23         hook(type => "rcs", id => "rcs_getmtime", call => \&rcs_getmtime);
24 }
26 sub checkconfig () {
27         if (exists $config{mercurial_wrapper} && length $config{mercurial_wrapper}) {
28                 push @{$config{wrappers}}, {
29                         wrapper => $config{mercurial_wrapper},
30                         wrappermode => (defined $config{mercurial_wrappermode} ? $config{mercurial_wrappermode} : "06755"),
31                 };
32         }
33 }
35 sub getsetup () {
36         return
37                 plugin => {
38                         safe => 0, # rcs plugin
39                         rebuild => undef,
40                         section => "rcs",
41                 },
42                 mercurial_wrapper => {
43                         type => "string",
44                         #example => # FIXME add example
45                         description => "mercurial post-commit hook to generate",
46                         safe => 0, # file
47                         rebuild => 0,
48                 },
49                 mercurial_wrappermode => {
50                         type => "string",
51                         example => '06755',
52                         description => "mode for mercurial_wrapper (can safely be made suid)",
53                         safe => 0,
54                         rebuild => 0,
55                 },
56                 historyurl => {
57                         type => "string",
58                         example => "http://example.com:8000/log/tip/[[file]]",
59                         description => "url to hg serve'd repository, to show file history ([[file]] substituted)",
60                         safe => 1,
61                         rebuild => 1,
62                 },
63                 diffurl => {
64                         type => "string",
65                         example => "http://localhost:8000/?fd=[[r2]];file=[[file]]",
66                         description => "url to hg serve'd repository, to show diff ([[file]] and [[r2]] substituted)",
67                         safe => 1,
68                         rebuild => 1,
69                 },
70 }
72 sub mercurial_log ($) {
73         my $out = shift;
74         my @infos;
76         while (<$out>) {
77                 my $line = $_;
78                 my ($key, $value);
80                 if (/^description:/) {
81                         $key = "description";
82                         $value = "";
84                         # slurp everything as the description text 
85                         # until the next changeset
86                         while (<$out>) {
87                                 if (/^changeset: /) {
88                                         $line = $_;
89                                         last;
90                                 }
92                                 $value .= $_;
93                         }
95                         local $/ = "";
96                         chomp $value;
97                         $infos[$#infos]{$key} = $value;
98                 }
100                 chomp $line;
101                 ($key, $value) = split /: +/, $line, 2;
103                 if ($key eq "changeset") {
104                         push @infos, {};
106                         # remove the revision index, which is strictly 
107                         # local to the repository
108                         $value =~ s/^\d+://;
109                 }
111                 $infos[$#infos]{$key} = $value;
112         }
113         close $out;
115         return @infos;
118 sub rcs_update () {
119         my @cmdline = ("hg", "-q", "-R", "$config{srcdir}", "update");
120         if (system(@cmdline) != 0) {
121                 warn "'@cmdline' failed: $!";
122         }
125 sub rcs_prepedit ($) {
126         return "";
129 sub rcs_commit ($$$;$$) {
130         my ($file, $message, $rcstoken, $user, $ipaddr) = @_;
132         if (defined $user) {
133                 $user = IkiWiki::possibly_foolish_untaint($user);
134         }
135         elsif (defined $ipaddr) {
136                 $user = "Anonymous from ".IkiWiki::possibly_foolish_untaint($ipaddr);
137         }
138         else {
139                 $user = "Anonymous";
140         }
142         $message = IkiWiki::possibly_foolish_untaint($message);
143         if (! length $message) {
144                 $message = "no message given";
145         }
147         my @cmdline = ("hg", "-q", "-R", $config{srcdir}, "commit", 
148                        "-m", $message, "-u", $user);
149         if (system(@cmdline) != 0) {
150                 warn "'@cmdline' failed: $!";
151         }
153         return undef; # success
156 sub rcs_commit_staged ($$$) {
157         # Commits all staged changes. Changes can be staged using rcs_add,
158         # rcs_remove, and rcs_rename.
159         my ($message, $user, $ipaddr)=@_;
160         
161         error("rcs_commit_staged not implemented for mercurial"); # TODO
164 sub rcs_add ($) {
165         my ($file) = @_;
167         my @cmdline = ("hg", "-q", "-R", "$config{srcdir}", "add", "$config{srcdir}/$file");
168         if (system(@cmdline) != 0) {
169                 warn "'@cmdline' failed: $!";
170         }
173 sub rcs_remove ($) {
174         my ($file) = @_;
176         error("rcs_remove not implemented for mercurial"); # TODO
179 sub rcs_rename ($$) {
180         my ($src, $dest) = @_;
182         error("rcs_rename not implemented for mercurial"); # TODO
185 sub rcs_recentchanges ($) {
186         my ($num) = @_;
188         my @cmdline = ("hg", "-R", $config{srcdir}, "log", "-v", "-l", $num,
189                 "--style", "default");
190         open (my $out, "@cmdline |");
192         eval q{use Date::Parse};
193         error($@) if $@;
195         my @ret;
196         foreach my $info (mercurial_log($out)) {
197                 my @pages = ();
198                 my @message = ();
199         
200                 foreach my $msgline (split(/\n/, $info->{description})) {
201                         push @message, { line => $msgline };
202                 }
204                 foreach my $file (split / /,$info->{files}) {
205                         my $diffurl = defined $config{diffurl} ? $config{'diffurl'} : "";
206                         $diffurl =~ s/\[\[file\]\]/$file/go;
207                         $diffurl =~ s/\[\[r2\]\]/$info->{changeset}/go;
209                         push @pages, {
210                                 page => pagename($file),
211                                 diffurl => $diffurl,
212                         };
213                 }
215                 my $user = $info->{"user"};
216                 $user =~ s/\s*<.*>\s*$//;
217                 $user =~ s/^\s*//;
219                 push @ret, {
220                         rev        => $info->{"changeset"},
221                         user       => $user,
222                         committype => "hg",
223                         when       => str2time($info->{"date"}),
224                         message    => [@message],
225                         pages      => [@pages],
226                 };
227         }
229         return @ret;
232 sub rcs_diff ($) {
233         # TODO
236 sub rcs_getctime ($) {
237         my ($file) = @_;
239         # XXX filename passes through the shell here, should try to avoid
240         # that just in case
241         my @cmdline = ("hg", "-R", $config{srcdir}, "log", "-v",
242                 "--style", "default", "$config{srcdir}/$file");
243         open (my $out, "@cmdline |");
245         my @log = mercurial_log($out);
247         if (length @log < 1) {
248                 return 0;
249         }
251         eval q{use Date::Parse};
252         error($@) if $@;
253         
254         my $ctime = str2time($log[$#log]->{"date"});
255         return $ctime;
258 sub rcs_getmtime ($) {
259         error "rcs_getmtime is not implemented for mercurial\n"; # TODO