]> git.ipfire.org Git - thirdparty/git.git/blame_incremental - contrib/git-svn/git-svn.perl
git-svn: add 'log' command, a facsimile of basic `svn log'
[thirdparty/git.git] / contrib / git-svn / git-svn.perl
... / ...
CommitLineData
1#!/usr/bin/env perl
2# Copyright (C) 2006, Eric Wong <normalperson@yhbt.net>
3# License: GPL v2 or later
4use warnings;
5use strict;
6use vars qw/ $AUTHOR $VERSION
7 $SVN_URL $SVN_INFO $SVN_WC $SVN_UUID
8 $GIT_SVN_INDEX $GIT_SVN
9 $GIT_DIR $REV_DIR $GIT_SVN_DIR/;
10$AUTHOR = 'Eric Wong <normalperson@yhbt.net>';
11$VERSION = '1.1.0-pre';
12
13use Cwd qw/abs_path/;
14$GIT_DIR = abs_path($ENV{GIT_DIR} || '.git');
15$ENV{GIT_DIR} = $GIT_DIR;
16
17my $LC_ALL = $ENV{LC_ALL};
18my $TZ = $ENV{TZ};
19# make sure the svn binary gives consistent output between locales and TZs:
20$ENV{TZ} = 'UTC';
21$ENV{LC_ALL} = 'C';
22
23# If SVN:: library support is added, please make the dependencies
24# optional and preserve the capability to use the command-line client.
25# use eval { require SVN::... } to make it lazy load
26# We don't use any modules not in the standard Perl distribution:
27use Carp qw/croak/;
28use IO::File qw//;
29use File::Basename qw/dirname basename/;
30use File::Path qw/mkpath/;
31use Getopt::Long qw/:config gnu_getopt no_ignore_case auto_abbrev pass_through/;
32use File::Spec qw//;
33use POSIX qw/strftime/;
34my $sha1 = qr/[a-f\d]{40}/;
35my $sha1_short = qr/[a-f\d]{4,40}/;
36my ($_revision,$_stdin,$_no_ignore_ext,$_no_stop_copy,$_help,$_rmdir,$_edit,
37 $_find_copies_harder, $_l, $_cp_similarity,
38 $_repack, $_repack_nr, $_repack_flags,
39 $_template, $_shared, $_no_default_regex, $_no_graft_copy,
40 $_limit, $_verbose, $_incremental, $_oneline, $_l_fmt, $_show_commit,
41 $_version, $_upgrade, $_authors, $_branch_all_refs, @_opt_m);
42my (@_branch_from, %tree_map, %users, %rusers);
43my ($_svn_co_url_revs, $_svn_pg_peg_revs);
44my @repo_path_split_cache;
45
46my %fc_opts = ( 'no-ignore-externals' => \$_no_ignore_ext,
47 'branch|b=s' => \@_branch_from,
48 'branch-all-refs|B' => \$_branch_all_refs,
49 'authors-file|A=s' => \$_authors,
50 'repack:i' => \$_repack,
51 'repack-flags|repack-args|repack-opts=s' => \$_repack_flags);
52
53my ($_trunk, $_tags, $_branches);
54my %multi_opts = ( 'trunk|T=s' => \$_trunk,
55 'tags|t=s' => \$_tags,
56 'branches|b=s' => \$_branches );
57my %init_opts = ( 'template=s' => \$_template, 'shared' => \$_shared );
58
59# yes, 'native' sets "\n". Patches to fix this for non-*nix systems welcome:
60my %EOL = ( CR => "\015", LF => "\012", CRLF => "\015\012", native => "\012" );
61
62my %cmd = (
63 fetch => [ \&fetch, "Download new revisions from SVN",
64 { 'revision|r=s' => \$_revision, %fc_opts } ],
65 init => [ \&init, "Initialize a repo for tracking" .
66 " (requires URL argument)",
67 \%init_opts ],
68 commit => [ \&commit, "Commit git revisions to SVN",
69 { 'stdin|' => \$_stdin,
70 'edit|e' => \$_edit,
71 'rmdir' => \$_rmdir,
72 'find-copies-harder' => \$_find_copies_harder,
73 'l=i' => \$_l,
74 'copy-similarity|C=i'=> \$_cp_similarity,
75 %fc_opts,
76 } ],
77 'show-ignore' => [ \&show_ignore, "Show svn:ignore listings", { } ],
78 rebuild => [ \&rebuild, "Rebuild git-svn metadata (after git clone)",
79 { 'no-ignore-externals' => \$_no_ignore_ext,
80 'upgrade' => \$_upgrade } ],
81 'graft-branches' => [ \&graft_branches,
82 'Detect merges/branches from already imported history',
83 { 'merge-rx|m' => \@_opt_m,
84 'no-default-regex' => \$_no_default_regex,
85 'no-graft-copy' => \$_no_graft_copy } ],
86 'multi-init' => [ \&multi_init,
87 'Initialize multiple trees (like git-svnimport)',
88 { %multi_opts, %fc_opts } ],
89 'multi-fetch' => [ \&multi_fetch,
90 'Fetch multiple trees (like git-svnimport)',
91 \%fc_opts ],
92 'log' => [ \&show_log, 'Show commit logs',
93 { 'limit=i' => \$_limit,
94 'revision|r=s' => \$_revision,
95 'verbose|v' => \$_verbose,
96 'incremental' => \$_incremental,
97 'oneline' => \$_oneline,
98 'show-commit' => \$_show_commit,
99 'authors-file|A=s' => \$_authors,
100 } ],
101);
102
103my $cmd;
104for (my $i = 0; $i < @ARGV; $i++) {
105 if (defined $cmd{$ARGV[$i]}) {
106 $cmd = $ARGV[$i];
107 splice @ARGV, $i, 1;
108 last;
109 }
110};
111
112my %opts = %{$cmd{$cmd}->[2]} if (defined $cmd);
113
114read_repo_config(\%opts);
115my $rv = GetOptions(%opts, 'help|H|h' => \$_help,
116 'version|V' => \$_version,
117 'id|i=s' => \$GIT_SVN);
118exit 1 if (!$rv && $cmd ne 'log');
119
120set_default_vals();
121usage(0) if $_help;
122version() if $_version;
123usage(1) unless defined $cmd;
124init_vars();
125load_authors() if $_authors;
126load_all_refs() if $_branch_all_refs;
127svn_compat_check();
128migration_check() unless $cmd =~ /^(?:init|multi-init)$/;
129$cmd{$cmd}->[0]->(@ARGV);
130exit 0;
131
132####################### primary functions ######################
133sub usage {
134 my $exit = shift || 0;
135 my $fd = $exit ? \*STDERR : \*STDOUT;
136 print $fd <<"";
137git-svn - bidirectional operations between a single Subversion tree and git
138Usage: $0 <command> [options] [arguments]\n
139
140 print $fd "Available commands:\n" unless $cmd;
141
142 foreach (sort keys %cmd) {
143 next if $cmd && $cmd ne $_;
144 print $fd ' ',pack('A13',$_),$cmd{$_}->[1],"\n";
145 foreach (keys %{$cmd{$_}->[2]}) {
146 # prints out arguments as they should be passed:
147 my $x = s#[:=]s$## ? '<arg>' : s#[:=]i$## ? '<num>' : '';
148 print $fd ' ' x 17, join(', ', map { length $_ > 1 ?
149 "--$_" : "-$_" }
150 split /\|/,$_)," $x\n";
151 }
152 }
153 print $fd <<"";
154\nGIT_SVN_ID may be set in the environment or via the --id/-i switch to an
155arbitrary identifier if you're tracking multiple SVN branches/repositories in
156one git repository and want to keep them separate. See git-svn(1) for more
157information.
158
159 exit $exit;
160}
161
162sub version {
163 print "git-svn version $VERSION\n";
164 exit 0;
165}
166
167sub rebuild {
168 $SVN_URL = shift or undef;
169 my $newest_rev = 0;
170 if ($_upgrade) {
171 sys('git-update-ref',"refs/remotes/$GIT_SVN","$GIT_SVN-HEAD");
172 } else {
173 check_upgrade_needed();
174 }
175
176 my $pid = open(my $rev_list,'-|');
177 defined $pid or croak $!;
178 if ($pid == 0) {
179 exec("git-rev-list","refs/remotes/$GIT_SVN") or croak $!;
180 }
181 my $latest;
182 while (<$rev_list>) {
183 chomp;
184 my $c = $_;
185 croak "Non-SHA1: $c\n" unless $c =~ /^$sha1$/o;
186 my @commit = grep(/^git-svn-id: /,`git-cat-file commit $c`);
187 next if (!@commit); # skip merges
188 my ($url, $rev, $uuid) = extract_metadata($commit[$#commit]);
189 if (!$rev || !$uuid) {
190 croak "Unable to extract revision or UUID from ",
191 "$c, $commit[$#commit]\n";
192 }
193
194 # if we merged or otherwise started elsewhere, this is
195 # how we break out of it
196 next if (defined $SVN_UUID && ($uuid ne $SVN_UUID));
197 next if (defined $SVN_URL && defined $url && ($url ne $SVN_URL));
198
199 print "r$rev = $c\n";
200 unless (defined $latest) {
201 if (!$SVN_URL && !$url) {
202 croak "SVN repository location required: $url\n";
203 }
204 $SVN_URL ||= $url;
205 $SVN_UUID ||= $uuid;
206 setup_git_svn();
207 $latest = $rev;
208 }
209 assert_revision_eq_or_unknown($rev, $c);
210 sys('git-update-ref',"svn/$GIT_SVN/revs/$rev",$c);
211 $newest_rev = $rev if ($rev > $newest_rev);
212 }
213 close $rev_list or croak $?;
214 if (!chdir $SVN_WC) {
215 svn_cmd_checkout($SVN_URL, $latest, $SVN_WC);
216 chdir $SVN_WC or croak $!;
217 }
218
219 $pid = fork;
220 defined $pid or croak $!;
221 if ($pid == 0) {
222 my @svn_up = qw(svn up);
223 push @svn_up, '--ignore-externals' unless $_no_ignore_ext;
224 sys(@svn_up,"-r$newest_rev");
225 $ENV{GIT_INDEX_FILE} = $GIT_SVN_INDEX;
226 index_changes();
227 exec('git-write-tree') or croak $!;
228 }
229 waitpid $pid, 0;
230 croak $? if $?;
231
232 if ($_upgrade) {
233 print STDERR <<"";
234Keeping deprecated refs/head/$GIT_SVN-HEAD for now. Please remove it
235when you have upgraded your tools and habits to use refs/remotes/$GIT_SVN
236
237 }
238}
239
240sub init {
241 $SVN_URL = shift or die "SVN repository location required " .
242 "as a command-line argument\n";
243 $SVN_URL =~ s!/+$!!; # strip trailing slash
244 unless (-d $GIT_DIR) {
245 my @init_db = ('git-init-db');
246 push @init_db, "--template=$_template" if defined $_template;
247 push @init_db, "--shared" if defined $_shared;
248 sys(@init_db);
249 }
250 setup_git_svn();
251}
252
253sub fetch {
254 my (@parents) = @_;
255 check_upgrade_needed();
256 $SVN_URL ||= file_to_s("$GIT_SVN_DIR/info/url");
257 my @log_args = -d $SVN_WC ? ($SVN_WC) : ($SVN_URL);
258 unless ($_revision) {
259 $_revision = -d $SVN_WC ? 'BASE:HEAD' : '0:HEAD';
260 }
261 push @log_args, "-r$_revision";
262 push @log_args, '--stop-on-copy' unless $_no_stop_copy;
263
264 my $svn_log = svn_log_raw(@log_args);
265
266 my $base = next_log_entry($svn_log) or croak "No base revision!\n";
267 my $last_commit = undef;
268 unless (-d $SVN_WC) {
269 svn_cmd_checkout($SVN_URL,$base->{revision},$SVN_WC);
270 chdir $SVN_WC or croak $!;
271 read_uuid();
272 $last_commit = git_commit($base, @parents);
273 assert_tree($last_commit);
274 } else {
275 chdir $SVN_WC or croak $!;
276 read_uuid();
277 eval { $last_commit = file_to_s("$REV_DIR/$base->{revision}") };
278 # looks like a user manually cp'd and svn switch'ed
279 unless ($last_commit) {
280 sys(qw/svn revert -R ./);
281 assert_svn_wc_clean($base->{revision});
282 $last_commit = git_commit($base, @parents);
283 assert_tree($last_commit);
284 }
285 }
286 my @svn_up = qw(svn up);
287 push @svn_up, '--ignore-externals' unless $_no_ignore_ext;
288 my $last = $base;
289 while (my $log_msg = next_log_entry($svn_log)) {
290 assert_tree($last_commit);
291 if ($last->{revision} >= $log_msg->{revision}) {
292 croak "Out of order: last >= current: ",
293 "$last->{revision} >= $log_msg->{revision}\n";
294 }
295 # Revert is needed for cases like:
296 # https://svn.musicpd.org/Jamming/trunk (r166:167), but
297 # I can't seem to reproduce something like that on a test...
298 sys(qw/svn revert -R ./);
299 assert_svn_wc_clean($last->{revision});
300 sys(@svn_up,"-r$log_msg->{revision}");
301 $last_commit = git_commit($log_msg, $last_commit, @parents);
302 $last = $log_msg;
303 }
304 unless (-e "$GIT_DIR/refs/heads/master") {
305 sys(qw(git-update-ref refs/heads/master),$last_commit);
306 }
307 close $svn_log->{fh};
308 return $last;
309}
310
311sub commit {
312 my (@commits) = @_;
313 check_upgrade_needed();
314 if ($_stdin || !@commits) {
315 print "Reading from stdin...\n";
316 @commits = ();
317 while (<STDIN>) {
318 if (/\b($sha1_short)\b/o) {
319 unshift @commits, $1;
320 }
321 }
322 }
323 my @revs;
324 foreach my $c (@commits) {
325 chomp(my @tmp = safe_qx('git-rev-parse',$c));
326 if (scalar @tmp == 1) {
327 push @revs, $tmp[0];
328 } elsif (scalar @tmp > 1) {
329 push @revs, reverse (safe_qx('git-rev-list',@tmp));
330 } else {
331 die "Failed to rev-parse $c\n";
332 }
333 }
334 chomp @revs;
335
336 chdir $SVN_WC or croak "Unable to chdir $SVN_WC: $!\n";
337 my $info = svn_info('.');
338 my $fetched = fetch();
339 if ($info->{Revision} != $fetched->{revision}) {
340 print STDERR "There are new revisions that were fetched ",
341 "and need to be merged (or acknowledged) ",
342 "before committing.\n";
343 exit 1;
344 }
345 $info = svn_info('.');
346 read_uuid($info);
347 my $svn_current_rev = $info->{'Last Changed Rev'};
348 foreach my $c (@revs) {
349 my $mods = svn_checkout_tree($svn_current_rev, $c);
350 if (scalar @$mods == 0) {
351 print "Skipping, no changes detected\n";
352 next;
353 }
354 $svn_current_rev = svn_commit_tree($svn_current_rev, $c);
355 }
356 print "Done committing ",scalar @revs," revisions to SVN\n";
357}
358
359sub show_ignore {
360 require File::Find or die $!;
361 my $exclude_file = "$GIT_DIR/info/exclude";
362 open my $fh, '<', $exclude_file or croak $!;
363 chomp(my @excludes = (<$fh>));
364 close $fh or croak $!;
365
366 $SVN_URL ||= file_to_s("$GIT_SVN_DIR/info/url");
367 chdir $SVN_WC or croak $!;
368 my %ign;
369 File::Find::find({wanted=>sub{if(lstat $_ && -d _ && -d "$_/.svn"){
370 s#^\./##;
371 @{$ign{$_}} = svn_propget_base('svn:ignore', $_);
372 }}, no_chdir=>1},'.');
373
374 print "\n# /\n";
375 foreach (@{$ign{'.'}}) { print '/',$_ if /\S/ }
376 delete $ign{'.'};
377 foreach my $i (sort keys %ign) {
378 print "\n# ",$i,"\n";
379 foreach (@{$ign{$i}}) { print '/',$i,'/',$_ if /\S/ }
380 }
381}
382
383sub graft_branches {
384 my $gr_file = "$GIT_DIR/info/grafts";
385 my ($grafts, $comments) = read_grafts($gr_file);
386 my $gr_sha1;
387
388 if (%$grafts) {
389 # temporarily disable our grafts file to make this idempotent
390 chomp($gr_sha1 = safe_qx(qw/git-hash-object -w/,$gr_file));
391 rename $gr_file, "$gr_file~$gr_sha1" or croak $!;
392 }
393
394 my $l_map = read_url_paths();
395 my @re = map { qr/$_/is } @_opt_m if @_opt_m;
396 unless ($_no_default_regex) {
397 push @re, ( qr/\b(?:merge|merging|merged)\s+(\S.+)/is,
398 qr/\b(?:from|of)\s+(\S.+)/is );
399 }
400 foreach my $u (keys %$l_map) {
401 if (@re) {
402 foreach my $p (keys %{$l_map->{$u}}) {
403 graft_merge_msg($grafts,$l_map,$u,$p);
404 }
405 }
406 graft_file_copy($grafts,$l_map,$u) unless $_no_graft_copy;
407 }
408
409 write_grafts($grafts, $comments, $gr_file);
410 unlink "$gr_file~$gr_sha1" if $gr_sha1;
411}
412
413sub multi_init {
414 my $url = shift;
415 $_trunk ||= 'trunk';
416 $_trunk =~ s#/+$##;
417 $url =~ s#/+$## if $url;
418 if ($_trunk !~ m#^[a-z\+]+://#) {
419 $_trunk = '/' . $_trunk if ($_trunk !~ m#^/#);
420 unless ($url) {
421 print STDERR "E: '$_trunk' is not a complete URL ",
422 "and a separate URL is not specified\n";
423 exit 1;
424 }
425 $_trunk = $url . $_trunk;
426 }
427 if ($GIT_SVN eq 'git-svn') {
428 print "GIT_SVN_ID set to 'trunk' for $_trunk\n";
429 $GIT_SVN = $ENV{GIT_SVN_ID} = 'trunk';
430 }
431 init_vars();
432 init($_trunk);
433 complete_url_ls_init($url, $_branches, '--branches/-b', '');
434 complete_url_ls_init($url, $_tags, '--tags/-t', 'tags/');
435}
436
437sub multi_fetch {
438 # try to do trunk first, since branches/tags
439 # may be descended from it.
440 if (-d "$GIT_DIR/svn/trunk") {
441 print "Fetching trunk\n";
442 defined(my $pid = fork) or croak $!;
443 if (!$pid) {
444 $GIT_SVN = $ENV{GIT_SVN_ID} = 'trunk';
445 init_vars();
446 fetch(@_);
447 exit 0;
448 }
449 waitpid $pid, 0;
450 croak $? if $?;
451 }
452 rec_fetch('', "$GIT_DIR/svn", @_);
453}
454
455sub show_log {
456 my (@args) = @_;
457 my ($r_min, $r_max);
458 my $r_last = -1; # prevent dupes
459 rload_authors() if $_authors;
460 if (defined $TZ) {
461 $ENV{TZ} = $TZ;
462 } else {
463 delete $ENV{TZ};
464 }
465 if (defined $_revision) {
466 if ($_revision =~ /^(\d+):(\d+)$/) {
467 ($r_min, $r_max) = ($1, $2);
468 } elsif ($_revision =~ /^\d+$/) {
469 $r_min = $r_max = $_revision;
470 } else {
471 print STDERR "-r$_revision is not supported, use ",
472 "standard \'git log\' arguments instead\n";
473 exit 1;
474 }
475 }
476
477 my $pid = open(my $log,'-|');
478 defined $pid or croak $!;
479 if (!$pid) {
480 my @rl = (qw/git-log --abbrev-commit --pretty=raw
481 --default/, "remotes/$GIT_SVN");
482 push @rl, '--raw' if $_verbose;
483 exec(@rl, @args) or croak $!;
484 }
485 setup_pager();
486 my (@k, $c, $d);
487 while (<$log>) {
488 if (/^commit ($sha1_short)/o) {
489 my $cmt = $1;
490 if ($c && defined $c->{r} && $c->{r} != $r_last) {
491 $r_last = $c->{r};
492 process_commit($c, $r_min, $r_max, \@k) or
493 goto out;
494 }
495 $d = undef;
496 $c = { c => $cmt };
497 } elsif (/^author (.+) (\d+) ([\-\+]?\d+)$/) {
498 get_author_info($c, $1, $2, $3);
499 } elsif (/^(?:tree|parent|committer) /) {
500 # ignore
501 } elsif (/^:\d{6} \d{6} $sha1_short/o) {
502 push @{$c->{raw}}, $_;
503 } elsif (/^diff /) {
504 $d = 1;
505 push @{$c->{diff}}, $_;
506 } elsif ($d) {
507 push @{$c->{diff}}, $_;
508 } elsif (/^ (git-svn-id:.+)$/) {
509 my ($url, $rev, $uuid) = extract_metadata($1);
510 $c->{r} = $rev;
511 } elsif (s/^ //) {
512 push @{$c->{l}}, $_;
513 }
514 }
515 if ($c && defined $c->{r} && $c->{r} != $r_last) {
516 $r_last = $c->{r};
517 process_commit($c, $r_min, $r_max, \@k);
518 }
519 if (@k) {
520 my $swap = $r_max;
521 $r_max = $r_min;
522 $r_min = $swap;
523 process_commit($_, $r_min, $r_max) foreach reverse @k;
524 }
525out:
526 close $log;
527 print '-' x72,"\n" unless $_incremental || $_oneline;
528}
529
530########################### utility functions #########################
531
532sub rec_fetch {
533 my ($pfx, $p, @args) = @_;
534 my @dir;
535 foreach (sort <$p/*>) {
536 if (-r "$_/info/url") {
537 $pfx .= '/' if $pfx && $pfx !~ m!/$!;
538 my $id = $pfx . basename $_;
539 next if $id eq 'trunk';
540 print "Fetching $id\n";
541 defined(my $pid = fork) or croak $!;
542 if (!$pid) {
543 $GIT_SVN = $ENV{GIT_SVN_ID} = $id;
544 init_vars();
545 fetch(@args);
546 exit 0;
547 }
548 waitpid $pid, 0;
549 croak $? if $?;
550 } elsif (-d $_) {
551 push @dir, $_;
552 }
553 }
554 foreach (@dir) {
555 my $x = $_;
556 $x =~ s!^\Q$GIT_DIR\E/svn/!!;
557 rec_fetch($x, $_);
558 }
559}
560
561sub complete_url_ls_init {
562 my ($url, $var, $switch, $pfx) = @_;
563 unless ($var) {
564 print STDERR "W: $switch not specified\n";
565 return;
566 }
567 $var =~ s#/+$##;
568 if ($var !~ m#^[a-z\+]+://#) {
569 $var = '/' . $var if ($var !~ m#^/#);
570 unless ($url) {
571 print STDERR "E: '$var' is not a complete URL ",
572 "and a separate URL is not specified\n";
573 exit 1;
574 }
575 $var = $url . $var;
576 }
577 chomp(my @ls = safe_qx(qw/svn ls --non-interactive/, $var));
578 my $old = $GIT_SVN;
579 defined(my $pid = fork) or croak $!;
580 if (!$pid) {
581 foreach my $u (map { "$var/$_" } (grep m!/$!, @ls)) {
582 $u =~ s#/+$##;
583 if ($u !~ m!\Q$var\E/(.+)$!) {
584 print STDERR "W: Unrecognized URL: $u\n";
585 die "This should never happen\n";
586 }
587 my $id = $pfx.$1;
588 print "init $u => $id\n";
589 $GIT_SVN = $ENV{GIT_SVN_ID} = $id;
590 init_vars();
591 init($u);
592 }
593 exit 0;
594 }
595 waitpid $pid, 0;
596 croak $? if $?;
597}
598
599sub common_prefix {
600 my $paths = shift;
601 my %common;
602 foreach (@$paths) {
603 my @tmp = split m#/#, $_;
604 my $p = '';
605 while (my $x = shift @tmp) {
606 $p .= "/$x";
607 $common{$p} ||= 0;
608 $common{$p}++;
609 }
610 }
611 foreach (sort {length $b <=> length $a} keys %common) {
612 if ($common{$_} == @$paths) {
613 return $_;
614 }
615 }
616 return '';
617}
618
619# this isn't funky-filename safe, but good enough for now...
620sub graft_file_copy {
621 my ($grafts, $l_map, $u) = @_;
622 my $paths = $l_map->{$u};
623 my $pfx = common_prefix([keys %$paths]);
624
625 my $pid = open my $fh, '-|';
626 defined $pid or croak $!;
627 unless ($pid) {
628 exec(qw/svn log -v/, $u.$pfx) or croak $!;
629 }
630 my ($r, $mp) = (undef, undef);
631 while (<$fh>) {
632 chomp;
633 if (/^\-{72}$/) {
634 $mp = $r = undef;
635 } elsif (/^r(\d+) \| /) {
636 $r = $1 unless defined $r;
637 } elsif (/^Changed paths:/) {
638 $mp = 1;
639 } elsif ($mp && m#^ [AR] /(\S.*?) \(from /(\S+?):(\d+)\)$#) {
640 my $dbg = "r$r | $_";
641 my ($p1, $p0, $r0) = ($1, $2, $3);
642 my $c;
643 foreach my $x (keys %$paths) {
644 next unless ($p1 =~ /^\Q$x\E/);
645 my $i = $paths->{$x};
646 my $f = "$GIT_DIR/svn/$i/revs/$r";
647 unless (-r $f) {
648 print STDERR "r$r of $i not imported,",
649 " $dbg\n";
650 next;
651 }
652 $c = file_to_s($f);
653 }
654 next unless $c;
655 foreach my $x (keys %$paths) {
656 next unless ($p0 =~ /^\Q$x\E/);
657 my $i = $paths->{$x};
658 my $f = "$GIT_DIR/svn/$i/revs/$r0";
659 while ($r0 && !-r $f) {
660 # could be an older revision, too...
661 $r0--;
662 $f = "$GIT_DIR/svn/$i/revs/$r0";
663 }
664 unless (-r $f) {
665 print STDERR "r$r0 of $i not imported,",
666 " $dbg\n";
667 next;
668 }
669 my $r1 = file_to_s($f);
670 $grafts->{$c}->{$r1} = 1;
671 }
672 }
673 }
674}
675
676sub process_merge_msg_matches {
677 my ($grafts, $l_map, $u, $p, $c, @matches) = @_;
678 my (@strong, @weak);
679 foreach (@matches) {
680 # merging with ourselves is not interesting
681 next if $_ eq $p;
682 if ($l_map->{$u}->{$_}) {
683 push @strong, $_;
684 } else {
685 push @weak, $_;
686 }
687 }
688 foreach my $w (@weak) {
689 last if @strong;
690 # no exact match, use branch name as regexp.
691 my $re = qr/\Q$w\E/i;
692 foreach (keys %{$l_map->{$u}}) {
693 if (/$re/) {
694 push @strong, $_;
695 last;
696 }
697 }
698 last if @strong;
699 $w = basename($w);
700 $re = qr/\Q$w\E/i;
701 foreach (keys %{$l_map->{$u}}) {
702 if (/$re/) {
703 push @strong, $_;
704 last;
705 }
706 }
707 }
708 my ($rev) = ($c->{m} =~ /^git-svn-id:\s(?:\S+?)\@(\d+)
709 \s(?:[a-f\d\-]+)$/xsm);
710 unless (defined $rev) {
711 ($rev) = ($c->{m} =~/^git-svn-id:\s(\d+)
712 \@(?:[a-f\d\-]+)/xsm);
713 return unless defined $rev;
714 }
715 foreach my $m (@strong) {
716 my ($r0, $s0) = find_rev_before($rev, $m);
717 $grafts->{$c->{c}}->{$s0} = 1 if defined $s0;
718 }
719}
720
721sub graft_merge_msg {
722 my ($grafts, $l_map, $u, $p, @re) = @_;
723
724 my $x = $l_map->{$u}->{$p};
725 my $rl = rev_list_raw($x);
726 while (my $c = next_rev_list_entry($rl)) {
727 foreach my $re (@re) {
728 my (@br) = ($c->{m} =~ /$re/g);
729 next unless @br;
730 process_merge_msg_matches($grafts,$l_map,$u,$p,$c,@br);
731 }
732 }
733}
734
735sub read_uuid {
736 return if $SVN_UUID;
737 my $info = shift || svn_info('.');
738 $SVN_UUID = $info->{'Repository UUID'} or
739 croak "Repository UUID unreadable\n";
740 s_to_file($SVN_UUID,"$GIT_SVN_DIR/info/uuid");
741}
742
743sub quiet_run {
744 my $pid = fork;
745 defined $pid or croak $!;
746 if (!$pid) {
747 open my $null, '>', '/dev/null' or croak $!;
748 open STDERR, '>&', $null or croak $!;
749 open STDOUT, '>&', $null or croak $!;
750 exec @_ or croak $!;
751 }
752 waitpid $pid, 0;
753 return $?;
754}
755
756sub repo_path_split {
757 my $full_url = shift;
758 $full_url =~ s#/+$##;
759
760 foreach (@repo_path_split_cache) {
761 if ($full_url =~ s#$_##) {
762 my $u = $1;
763 $full_url =~ s#^/+##;
764 return ($u, $full_url);
765 }
766 }
767
768 my ($url, $path) = ($full_url =~ m!^([a-z\+]+://[^/]*)(.*)$!i);
769 $path =~ s#^/+##;
770 my @paths = split(m#/+#, $path);
771
772 while (quiet_run(qw/svn ls --non-interactive/, $url)) {
773 my $n = shift @paths || last;
774 $url .= "/$n";
775 }
776 push @repo_path_split_cache, qr/^(\Q$url\E)/;
777 $path = join('/',@paths);
778 return ($url, $path);
779}
780
781sub setup_git_svn {
782 defined $SVN_URL or croak "SVN repository location required\n";
783 unless (-d $GIT_DIR) {
784 croak "GIT_DIR=$GIT_DIR does not exist!\n";
785 }
786 mkpath([$GIT_SVN_DIR]);
787 mkpath(["$GIT_SVN_DIR/info"]);
788 mkpath([$REV_DIR]);
789 s_to_file($SVN_URL,"$GIT_SVN_DIR/info/url");
790
791 open my $fd, '>>', "$GIT_SVN_DIR/info/exclude" or croak $!;
792 print $fd '.svn',"\n";
793 close $fd or croak $!;
794 my ($url, $path) = repo_path_split($SVN_URL);
795 s_to_file($url, "$GIT_SVN_DIR/info/repo_url");
796 s_to_file($path, "$GIT_SVN_DIR/info/repo_path");
797}
798
799sub assert_svn_wc_clean {
800 my ($svn_rev) = @_;
801 croak "$svn_rev is not an integer!\n" unless ($svn_rev =~ /^\d+$/);
802 my $lcr = svn_info('.')->{'Last Changed Rev'};
803 if ($svn_rev != $lcr) {
804 print STDERR "Checking for copy-tree ... ";
805 my @diff = grep(/^Index: /,(safe_qx(qw(svn diff),
806 "-r$lcr:$svn_rev")));
807 if (@diff) {
808 croak "Nope! Expected r$svn_rev, got r$lcr\n";
809 } else {
810 print STDERR "OK!\n";
811 }
812 }
813 my @status = grep(!/^Performing status on external/,(`svn status`));
814 @status = grep(!/^\s*$/,@status);
815 if (scalar @status) {
816 print STDERR "Tree ($SVN_WC) is not clean:\n";
817 print STDERR $_ foreach @status;
818 croak;
819 }
820}
821
822sub assert_tree {
823 my ($treeish) = @_;
824 croak "Not a sha1: $treeish\n" unless $treeish =~ /^$sha1$/o;
825 chomp(my $type = `git-cat-file -t $treeish`);
826 my $expected;
827 while ($type eq 'tag') {
828 chomp(($treeish, $type) = `git-cat-file tag $treeish`);
829 }
830 if ($type eq 'commit') {
831 $expected = (grep /^tree /,`git-cat-file commit $treeish`)[0];
832 ($expected) = ($expected =~ /^tree ($sha1)$/);
833 die "Unable to get tree from $treeish\n" unless $expected;
834 } elsif ($type eq 'tree') {
835 $expected = $treeish;
836 } else {
837 die "$treeish is a $type, expected tree, tag or commit\n";
838 }
839
840 my $old_index = $ENV{GIT_INDEX_FILE};
841 my $tmpindex = $GIT_SVN_INDEX.'.assert-tmp';
842 if (-e $tmpindex) {
843 unlink $tmpindex or croak $!;
844 }
845 $ENV{GIT_INDEX_FILE} = $tmpindex;
846 index_changes(1);
847 chomp(my $tree = `git-write-tree`);
848 if ($old_index) {
849 $ENV{GIT_INDEX_FILE} = $old_index;
850 } else {
851 delete $ENV{GIT_INDEX_FILE};
852 }
853 if ($tree ne $expected) {
854 croak "Tree mismatch, Got: $tree, Expected: $expected\n";
855 }
856 unlink $tmpindex;
857}
858
859sub parse_diff_tree {
860 my $diff_fh = shift;
861 local $/ = "\0";
862 my $state = 'meta';
863 my @mods;
864 while (<$diff_fh>) {
865 chomp $_; # this gets rid of the trailing "\0"
866 if ($state eq 'meta' && /^:(\d{6})\s(\d{6})\s
867 $sha1\s($sha1)\s([MTCRAD])\d*$/xo) {
868 push @mods, { mode_a => $1, mode_b => $2,
869 sha1_b => $3, chg => $4 };
870 if ($4 =~ /^(?:C|R)$/) {
871 $state = 'file_a';
872 } else {
873 $state = 'file_b';
874 }
875 } elsif ($state eq 'file_a') {
876 my $x = $mods[$#mods] or croak "Empty array\n";
877 if ($x->{chg} !~ /^(?:C|R)$/) {
878 croak "Error parsing $_, $x->{chg}\n";
879 }
880 $x->{file_a} = $_;
881 $state = 'file_b';
882 } elsif ($state eq 'file_b') {
883 my $x = $mods[$#mods] or croak "Empty array\n";
884 if (exists $x->{file_a} && $x->{chg} !~ /^(?:C|R)$/) {
885 croak "Error parsing $_, $x->{chg}\n";
886 }
887 if (!exists $x->{file_a} && $x->{chg} =~ /^(?:C|R)$/) {
888 croak "Error parsing $_, $x->{chg}\n";
889 }
890 $x->{file_b} = $_;
891 $state = 'meta';
892 } else {
893 croak "Error parsing $_\n";
894 }
895 }
896 close $diff_fh or croak $!;
897
898 return \@mods;
899}
900
901sub svn_check_prop_executable {
902 my $m = shift;
903 return if -l $m->{file_b};
904 if ($m->{mode_b} =~ /755$/) {
905 chmod((0755 &~ umask),$m->{file_b}) or croak $!;
906 if ($m->{mode_a} !~ /755$/) {
907 sys(qw(svn propset svn:executable 1), $m->{file_b});
908 }
909 -x $m->{file_b} or croak "$m->{file_b} is not executable!\n";
910 } elsif ($m->{mode_b} !~ /755$/ && $m->{mode_a} =~ /755$/) {
911 sys(qw(svn propdel svn:executable), $m->{file_b});
912 chmod((0644 &~ umask),$m->{file_b}) or croak $!;
913 -x $m->{file_b} and croak "$m->{file_b} is executable!\n";
914 }
915}
916
917sub svn_ensure_parent_path {
918 my $dir_b = dirname(shift);
919 svn_ensure_parent_path($dir_b) if ($dir_b ne File::Spec->curdir);
920 mkpath([$dir_b]) unless (-d $dir_b);
921 sys(qw(svn add -N), $dir_b) unless (-d "$dir_b/.svn");
922}
923
924sub precommit_check {
925 my $mods = shift;
926 my (%rm_file, %rmdir_check, %added_check);
927
928 my %o = ( D => 0, R => 1, C => 2, A => 3, M => 3, T => 3 );
929 foreach my $m (sort { $o{$a->{chg}} <=> $o{$b->{chg}} } @$mods) {
930 if ($m->{chg} eq 'R') {
931 if (-d $m->{file_b}) {
932 err_dir_to_file("$m->{file_a} => $m->{file_b}");
933 }
934 # dir/$file => dir/file/$file
935 my $dirname = dirname($m->{file_b});
936 while ($dirname ne File::Spec->curdir) {
937 if ($dirname ne $m->{file_a}) {
938 $dirname = dirname($dirname);
939 next;
940 }
941 err_file_to_dir("$m->{file_a} => $m->{file_b}");
942 }
943 # baz/zzz => baz (baz is a file)
944 $dirname = dirname($m->{file_a});
945 while ($dirname ne File::Spec->curdir) {
946 if ($dirname ne $m->{file_b}) {
947 $dirname = dirname($dirname);
948 next;
949 }
950 err_dir_to_file("$m->{file_a} => $m->{file_b}");
951 }
952 }
953 if ($m->{chg} =~ /^(D|R)$/) {
954 my $t = $1 eq 'D' ? 'file_b' : 'file_a';
955 $rm_file{ $m->{$t} } = 1;
956 my $dirname = dirname( $m->{$t} );
957 my $basename = basename( $m->{$t} );
958 $rmdir_check{$dirname}->{$basename} = 1;
959 } elsif ($m->{chg} =~ /^(?:A|C)$/) {
960 if (-d $m->{file_b}) {
961 err_dir_to_file($m->{file_b});
962 }
963 my $dirname = dirname( $m->{file_b} );
964 my $basename = basename( $m->{file_b} );
965 $added_check{$dirname}->{$basename} = 1;
966 while ($dirname ne File::Spec->curdir) {
967 if ($rm_file{$dirname}) {
968 err_file_to_dir($m->{file_b});
969 }
970 $dirname = dirname $dirname;
971 }
972 }
973 }
974 return (\%rmdir_check, \%added_check);
975
976 sub err_dir_to_file {
977 my $file = shift;
978 print STDERR "Node change from directory to file ",
979 "is not supported by Subversion: ",$file,"\n";
980 exit 1;
981 }
982 sub err_file_to_dir {
983 my $file = shift;
984 print STDERR "Node change from file to directory ",
985 "is not supported by Subversion: ",$file,"\n";
986 exit 1;
987 }
988}
989
990sub svn_checkout_tree {
991 my ($svn_rev, $treeish) = @_;
992 my $from = file_to_s("$REV_DIR/$svn_rev");
993 assert_tree($from);
994 print "diff-tree $from $treeish\n";
995 my $pid = open my $diff_fh, '-|';
996 defined $pid or croak $!;
997 if ($pid == 0) {
998 my @diff_tree = qw(git-diff-tree -z -r);
999 if ($_cp_similarity) {
1000 push @diff_tree, "-C$_cp_similarity";
1001 } else {
1002 push @diff_tree, '-C';
1003 }
1004 push @diff_tree, '--find-copies-harder' if $_find_copies_harder;
1005 push @diff_tree, "-l$_l" if defined $_l;
1006 exec(@diff_tree, $from, $treeish) or croak $!;
1007 }
1008 my $mods = parse_diff_tree($diff_fh);
1009 unless (@$mods) {
1010 # git can do empty commits, but SVN doesn't allow it...
1011 return $mods;
1012 }
1013 my ($rm, $add) = precommit_check($mods);
1014
1015 my %o = ( D => 1, R => 0, C => -1, A => 3, M => 3, T => 3 );
1016 foreach my $m (sort { $o{$a->{chg}} <=> $o{$b->{chg}} } @$mods) {
1017 if ($m->{chg} eq 'C') {
1018 svn_ensure_parent_path( $m->{file_b} );
1019 sys(qw(svn cp), $m->{file_a}, $m->{file_b});
1020 apply_mod_line_blob($m);
1021 svn_check_prop_executable($m);
1022 } elsif ($m->{chg} eq 'D') {
1023 sys(qw(svn rm --force), $m->{file_b});
1024 } elsif ($m->{chg} eq 'R') {
1025 svn_ensure_parent_path( $m->{file_b} );
1026 sys(qw(svn mv --force), $m->{file_a}, $m->{file_b});
1027 apply_mod_line_blob($m);
1028 svn_check_prop_executable($m);
1029 } elsif ($m->{chg} eq 'M') {
1030 apply_mod_line_blob($m);
1031 svn_check_prop_executable($m);
1032 } elsif ($m->{chg} eq 'T') {
1033 sys(qw(svn rm --force),$m->{file_b});
1034 apply_mod_line_blob($m);
1035 sys(qw(svn add --force), $m->{file_b});
1036 svn_check_prop_executable($m);
1037 } elsif ($m->{chg} eq 'A') {
1038 svn_ensure_parent_path( $m->{file_b} );
1039 apply_mod_line_blob($m);
1040 sys(qw(svn add --force), $m->{file_b});
1041 svn_check_prop_executable($m);
1042 } else {
1043 croak "Invalid chg: $m->{chg}\n";
1044 }
1045 }
1046
1047 assert_tree($treeish);
1048 if ($_rmdir) { # remove empty directories
1049 handle_rmdir($rm, $add);
1050 }
1051 assert_tree($treeish);
1052 return $mods;
1053}
1054
1055# svn ls doesn't work with respect to the current working tree, but what's
1056# in the repository. There's not even an option for it... *sigh*
1057# (added files don't show up and removed files remain in the ls listing)
1058sub svn_ls_current {
1059 my ($dir, $rm, $add) = @_;
1060 chomp(my @ls = safe_qx('svn','ls',$dir));
1061 my @ret = ();
1062 foreach (@ls) {
1063 s#/$##; # trailing slashes are evil
1064 push @ret, $_ unless $rm->{$dir}->{$_};
1065 }
1066 if (exists $add->{$dir}) {
1067 push @ret, keys %{$add->{$dir}};
1068 }
1069 return \@ret;
1070}
1071
1072sub handle_rmdir {
1073 my ($rm, $add) = @_;
1074
1075 foreach my $dir (sort {length $b <=> length $a} keys %$rm) {
1076 my $ls = svn_ls_current($dir, $rm, $add);
1077 next if (scalar @$ls);
1078 sys(qw(svn rm --force),$dir);
1079
1080 my $dn = dirname $dir;
1081 $rm->{ $dn }->{ basename $dir } = 1;
1082 $ls = svn_ls_current($dn, $rm, $add);
1083 while (scalar @$ls == 0 && $dn ne File::Spec->curdir) {
1084 sys(qw(svn rm --force),$dn);
1085 $dir = basename $dn;
1086 $dn = dirname $dn;
1087 $rm->{ $dn }->{ $dir } = 1;
1088 $ls = svn_ls_current($dn, $rm, $add);
1089 }
1090 }
1091}
1092
1093sub svn_commit_tree {
1094 my ($svn_rev, $commit) = @_;
1095 my $commit_msg = "$GIT_SVN_DIR/.svn-commit.tmp.$$";
1096 my %log_msg = ( msg => '' );
1097 open my $msg, '>', $commit_msg or croak $!;
1098
1099 chomp(my $type = `git-cat-file -t $commit`);
1100 if ($type eq 'commit') {
1101 my $pid = open my $msg_fh, '-|';
1102 defined $pid or croak $!;
1103
1104 if ($pid == 0) {
1105 exec(qw(git-cat-file commit), $commit) or croak $!;
1106 }
1107 my $in_msg = 0;
1108 while (<$msg_fh>) {
1109 if (!$in_msg) {
1110 $in_msg = 1 if (/^\s*$/);
1111 } elsif (/^git-svn-id: /) {
1112 # skip this, we regenerate the correct one
1113 # on re-fetch anyways
1114 } else {
1115 print $msg $_ or croak $!;
1116 }
1117 }
1118 close $msg_fh or croak $!;
1119 }
1120 close $msg or croak $!;
1121
1122 if ($_edit || ($type eq 'tree')) {
1123 my $editor = $ENV{VISUAL} || $ENV{EDITOR} || 'vi';
1124 system($editor, $commit_msg);
1125 }
1126
1127 # file_to_s removes all trailing newlines, so just use chomp() here:
1128 open $msg, '<', $commit_msg or croak $!;
1129 { local $/; chomp($log_msg{msg} = <$msg>); }
1130 close $msg or croak $!;
1131
1132 my ($oneline) = ($log_msg{msg} =~ /([^\n\r]+)/);
1133 print "Committing $commit: $oneline\n";
1134
1135 if (defined $LC_ALL) {
1136 $ENV{LC_ALL} = $LC_ALL;
1137 } else {
1138 delete $ENV{LC_ALL};
1139 }
1140 my @ci_output = safe_qx(qw(svn commit -F),$commit_msg);
1141 $ENV{LC_ALL} = 'C';
1142 unlink $commit_msg;
1143 my ($committed) = ($ci_output[$#ci_output] =~ /(\d+)/);
1144 if (!defined $committed) {
1145 my $out = join("\n",@ci_output);
1146 print STDERR "W: Trouble parsing \`svn commit' output:\n\n",
1147 $out, "\n\nAssuming English locale...";
1148 ($committed) = ($out =~ /^Committed revision \d+\./sm);
1149 defined $committed or die " FAILED!\n",
1150 "Commit output failed to parse committed revision!\n",
1151 print STDERR " OK\n";
1152 }
1153
1154 my @svn_up = qw(svn up);
1155 push @svn_up, '--ignore-externals' unless $_no_ignore_ext;
1156 if ($committed == ($svn_rev + 1)) {
1157 push @svn_up, "-r$committed";
1158 sys(@svn_up);
1159 my $info = svn_info('.');
1160 my $date = $info->{'Last Changed Date'} or die "Missing date\n";
1161 if ($info->{'Last Changed Rev'} != $committed) {
1162 croak "$info->{'Last Changed Rev'} != $committed\n"
1163 }
1164 my ($Y,$m,$d,$H,$M,$S,$tz) = ($date =~
1165 /(\d{4})\-(\d\d)\-(\d\d)\s
1166 (\d\d)\:(\d\d)\:(\d\d)\s([\-\+]\d+)/x)
1167 or croak "Failed to parse date: $date\n";
1168 $log_msg{date} = "$tz $Y-$m-$d $H:$M:$S";
1169 $log_msg{author} = $info->{'Last Changed Author'};
1170 $log_msg{revision} = $committed;
1171 $log_msg{msg} .= "\n";
1172 my $parent = file_to_s("$REV_DIR/$svn_rev");
1173 git_commit(\%log_msg, $parent, $commit);
1174 return $committed;
1175 }
1176 # resync immediately
1177 push @svn_up, "-r$svn_rev";
1178 sys(@svn_up);
1179 return fetch("$committed=$commit")->{revision};
1180}
1181
1182sub rev_list_raw {
1183 my (@args) = @_;
1184 my $pid = open my $fh, '-|';
1185 defined $pid or croak $!;
1186 if (!$pid) {
1187 exec(qw/git-rev-list --pretty=raw/, @args) or croak $!;
1188 }
1189 return { fh => $fh, t => { } };
1190}
1191
1192sub next_rev_list_entry {
1193 my $rl = shift;
1194 my $fh = $rl->{fh};
1195 my $x = $rl->{t};
1196 while (<$fh>) {
1197 if (/^commit ($sha1)$/o) {
1198 if ($x->{c}) {
1199 $rl->{t} = { c => $1 };
1200 return $x;
1201 } else {
1202 $x->{c} = $1;
1203 }
1204 } elsif (/^parent ($sha1)$/o) {
1205 $x->{p}->{$1} = 1;
1206 } elsif (s/^ //) {
1207 $x->{m} ||= '';
1208 $x->{m} .= $_;
1209 }
1210 }
1211 return ($x != $rl->{t}) ? $x : undef;
1212}
1213
1214# read the entire log into a temporary file (which is removed ASAP)
1215# and store the file handle + parser state
1216sub svn_log_raw {
1217 my (@log_args) = @_;
1218 my $log_fh = IO::File->new_tmpfile or croak $!;
1219 my $pid = fork;
1220 defined $pid or croak $!;
1221 if (!$pid) {
1222 open STDOUT, '>&', $log_fh or croak $!;
1223 exec (qw(svn log), @log_args) or croak $!
1224 }
1225 waitpid $pid, 0;
1226 croak $? if $?;
1227 seek $log_fh, 0, 0 or croak $!;
1228 return { state => 'sep', fh => $log_fh };
1229}
1230
1231sub next_log_entry {
1232 my $log = shift; # retval of svn_log_raw()
1233 my $ret = undef;
1234 my $fh = $log->{fh};
1235
1236 while (<$fh>) {
1237 chomp;
1238 if (/^\-{72}$/) {
1239 if ($log->{state} eq 'msg') {
1240 if ($ret->{lines}) {
1241 $ret->{msg} .= $_."\n";
1242 unless(--$ret->{lines}) {
1243 $log->{state} = 'sep';
1244 }
1245 } else {
1246 croak "Log parse error at: $_\n",
1247 $ret->{revision},
1248 "\n";
1249 }
1250 next;
1251 }
1252 if ($log->{state} ne 'sep') {
1253 croak "Log parse error at: $_\n",
1254 "state: $log->{state}\n",
1255 $ret->{revision},
1256 "\n";
1257 }
1258 $log->{state} = 'rev';
1259
1260 # if we have an empty log message, put something there:
1261 if ($ret) {
1262 $ret->{msg} ||= "\n";
1263 delete $ret->{lines};
1264 return $ret;
1265 }
1266 next;
1267 }
1268 if ($log->{state} eq 'rev' && s/^r(\d+)\s*\|\s*//) {
1269 my $rev = $1;
1270 my ($author, $date, $lines) = split(/\s*\|\s*/, $_, 3);
1271 ($lines) = ($lines =~ /(\d+)/);
1272 my ($Y,$m,$d,$H,$M,$S,$tz) = ($date =~
1273 /(\d{4})\-(\d\d)\-(\d\d)\s
1274 (\d\d)\:(\d\d)\:(\d\d)\s([\-\+]\d+)/x)
1275 or croak "Failed to parse date: $date\n";
1276 $ret = { revision => $rev,
1277 date => "$tz $Y-$m-$d $H:$M:$S",
1278 author => $author,
1279 lines => $lines,
1280 msg => '' };
1281 if (defined $_authors && ! defined $users{$author}) {
1282 die "Author: $author not defined in ",
1283 "$_authors file\n";
1284 }
1285 $log->{state} = 'msg_start';
1286 next;
1287 }
1288 # skip the first blank line of the message:
1289 if ($log->{state} eq 'msg_start' && /^$/) {
1290 $log->{state} = 'msg';
1291 } elsif ($log->{state} eq 'msg') {
1292 if ($ret->{lines}) {
1293 $ret->{msg} .= $_."\n";
1294 unless (--$ret->{lines}) {
1295 $log->{state} = 'sep';
1296 }
1297 } else {
1298 croak "Log parse error at: $_\n",
1299 $ret->{revision},"\n";
1300 }
1301 }
1302 }
1303 return $ret;
1304}
1305
1306sub svn_info {
1307 my $url = shift || $SVN_URL;
1308
1309 my $pid = open my $info_fh, '-|';
1310 defined $pid or croak $!;
1311
1312 if ($pid == 0) {
1313 exec(qw(svn info),$url) or croak $!;
1314 }
1315
1316 my $ret = {};
1317 # only single-lines seem to exist in svn info output
1318 while (<$info_fh>) {
1319 chomp $_;
1320 if (m#^([^:]+)\s*:\s*(\S.*)$#) {
1321 $ret->{$1} = $2;
1322 push @{$ret->{-order}}, $1;
1323 }
1324 }
1325 close $info_fh or croak $!;
1326 return $ret;
1327}
1328
1329sub sys { system(@_) == 0 or croak $? }
1330
1331sub eol_cp {
1332 my ($from, $to) = @_;
1333 my $es = svn_propget_base('svn:eol-style', $to);
1334 open my $rfd, '<', $from or croak $!;
1335 binmode $rfd or croak $!;
1336 open my $wfd, '>', $to or croak $!;
1337 binmode $wfd or croak $!;
1338
1339 my $eol = $EOL{$es} or undef;
1340 my $buf;
1341 use bytes;
1342 while (1) {
1343 my ($r, $w, $t);
1344 defined($r = sysread($rfd, $buf, 4096)) or croak $!;
1345 return unless $r;
1346 if ($eol) {
1347 if ($buf =~ /\015$/) {
1348 my $c;
1349 defined($r = sysread($rfd,$c,1)) or croak $!;
1350 $buf .= $c if $r > 0;
1351 }
1352 $buf =~ s/(?:\015\012|\015|\012)/$eol/gs;
1353 $r = length($buf);
1354 }
1355 for ($w = 0; $w < $r; $w += $t) {
1356 $t = syswrite($wfd, $buf, $r - $w, $w) or croak $!;
1357 }
1358 }
1359 no bytes;
1360}
1361
1362sub do_update_index {
1363 my ($z_cmd, $cmd, $no_text_base) = @_;
1364
1365 my $z = open my $p, '-|';
1366 defined $z or croak $!;
1367 unless ($z) { exec @$z_cmd or croak $! }
1368
1369 my $pid = open my $ui, '|-';
1370 defined $pid or croak $!;
1371 unless ($pid) {
1372 exec('git-update-index',"--$cmd",'-z','--stdin') or croak $!;
1373 }
1374 local $/ = "\0";
1375 while (my $x = <$p>) {
1376 chomp $x;
1377 if (!$no_text_base && lstat $x && ! -l _ &&
1378 svn_propget_base('svn:keywords', $x)) {
1379 my $mode = -x _ ? 0755 : 0644;
1380 my ($v,$d,$f) = File::Spec->splitpath($x);
1381 my $tb = File::Spec->catfile($d, '.svn', 'tmp',
1382 'text-base',"$f.svn-base");
1383 $tb =~ s#^/##;
1384 unless (-f $tb) {
1385 $tb = File::Spec->catfile($d, '.svn',
1386 'text-base',"$f.svn-base");
1387 $tb =~ s#^/##;
1388 }
1389 unlink $x or croak $!;
1390 eol_cp($tb, $x);
1391 chmod(($mode &~ umask), $x) or croak $!;
1392 }
1393 print $ui $x,"\0";
1394 }
1395 close $ui or croak $!;
1396}
1397
1398sub index_changes {
1399 my $no_text_base = shift;
1400 do_update_index([qw/git-diff-files --name-only -z/],
1401 'remove',
1402 $no_text_base);
1403 do_update_index([qw/git-ls-files -z --others/,
1404 "--exclude-from=$GIT_SVN_DIR/info/exclude"],
1405 'add',
1406 $no_text_base);
1407}
1408
1409sub s_to_file {
1410 my ($str, $file, $mode) = @_;
1411 open my $fd,'>',$file or croak $!;
1412 print $fd $str,"\n" or croak $!;
1413 close $fd or croak $!;
1414 chmod ($mode &~ umask, $file) if (defined $mode);
1415}
1416
1417sub file_to_s {
1418 my $file = shift;
1419 open my $fd,'<',$file or croak "$!: file: $file\n";
1420 local $/;
1421 my $ret = <$fd>;
1422 close $fd or croak $!;
1423 $ret =~ s/\s*$//s;
1424 return $ret;
1425}
1426
1427sub assert_revision_unknown {
1428 my $revno = shift;
1429 if (-f "$REV_DIR/$revno") {
1430 croak "$REV_DIR/$revno already exists! ",
1431 "Why are we refetching it?";
1432 }
1433}
1434
1435sub trees_eq {
1436 my ($x, $y) = @_;
1437 my @x = safe_qx('git-cat-file','commit',$x);
1438 my @y = safe_qx('git-cat-file','commit',$y);
1439 if (($y[0] ne $x[0]) || $x[0] !~ /^tree $sha1\n$/
1440 || $y[0] !~ /^tree $sha1\n$/) {
1441 print STDERR "Trees not equal: $y[0] != $x[0]\n";
1442 return 0
1443 }
1444 return 1;
1445}
1446
1447sub assert_revision_eq_or_unknown {
1448 my ($revno, $commit) = @_;
1449 if (-f "$REV_DIR/$revno") {
1450 my $current = file_to_s("$REV_DIR/$revno");
1451 if (($commit ne $current) && !trees_eq($commit, $current)) {
1452 croak "$REV_DIR/$revno already exists!\n",
1453 "current: $current\nexpected: $commit\n";
1454 }
1455 return;
1456 }
1457}
1458
1459sub git_commit {
1460 my ($log_msg, @parents) = @_;
1461 assert_revision_unknown($log_msg->{revision});
1462 my $out_fh = IO::File->new_tmpfile or croak $!;
1463
1464 map_tree_joins() if (@_branch_from && !%tree_map);
1465
1466 # commit parents can be conditionally bound to a particular
1467 # svn revision via: "svn_revno=commit_sha1", filter them out here:
1468 my @exec_parents;
1469 foreach my $p (@parents) {
1470 next unless defined $p;
1471 if ($p =~ /^(\d+)=($sha1_short)$/o) {
1472 if ($1 == $log_msg->{revision}) {
1473 push @exec_parents, $2;
1474 }
1475 } else {
1476 push @exec_parents, $p if $p =~ /$sha1_short/o;
1477 }
1478 }
1479
1480 my $pid = fork;
1481 defined $pid or croak $!;
1482 if ($pid == 0) {
1483 $ENV{GIT_INDEX_FILE} = $GIT_SVN_INDEX;
1484 index_changes();
1485 chomp(my $tree = `git-write-tree`);
1486 croak $? if $?;
1487 if (exists $tree_map{$tree}) {
1488 my %seen_parent = map { $_ => 1 } @exec_parents;
1489 foreach (@{$tree_map{$tree}}) {
1490 # MAXPARENT is defined to 16 in commit-tree.c:
1491 if ($seen_parent{$_} || @exec_parents > 16) {
1492 next;
1493 }
1494 push @exec_parents, $_;
1495 $seen_parent{$_} = 1;
1496 }
1497 }
1498 my $msg_fh = IO::File->new_tmpfile or croak $!;
1499 print $msg_fh $log_msg->{msg}, "\ngit-svn-id: ",
1500 "$SVN_URL\@$log_msg->{revision}",
1501 " $SVN_UUID\n" or croak $!;
1502 $msg_fh->flush == 0 or croak $!;
1503 seek $msg_fh, 0, 0 or croak $!;
1504
1505 set_commit_env($log_msg);
1506
1507 my @exec = ('git-commit-tree',$tree);
1508 push @exec, '-p', $_ foreach @exec_parents;
1509 open STDIN, '<&', $msg_fh or croak $!;
1510 open STDOUT, '>&', $out_fh or croak $!;
1511 exec @exec or croak $!;
1512 }
1513 waitpid($pid,0);
1514 croak $? if $?;
1515
1516 $out_fh->flush == 0 or croak $!;
1517 seek $out_fh, 0, 0 or croak $!;
1518 chomp(my $commit = do { local $/; <$out_fh> });
1519 if ($commit !~ /^$sha1$/o) {
1520 croak "Failed to commit, invalid sha1: $commit\n";
1521 }
1522 my @update_ref = ('git-update-ref',"refs/remotes/$GIT_SVN",$commit);
1523 if (my $primary_parent = shift @exec_parents) {
1524 $pid = fork;
1525 defined $pid or croak $!;
1526 if (!$pid) {
1527 close STDERR;
1528 close STDOUT;
1529 exec 'git-rev-parse','--verify',
1530 "refs/remotes/$GIT_SVN^0" or croak $!;
1531 }
1532 waitpid $pid, 0;
1533 push @update_ref, $primary_parent unless $?;
1534 }
1535 sys(@update_ref);
1536 sys('git-update-ref',"svn/$GIT_SVN/revs/$log_msg->{revision}",$commit);
1537 print "r$log_msg->{revision} = $commit\n";
1538 if ($_repack && (--$_repack_nr == 0)) {
1539 $_repack_nr = $_repack;
1540 sys("git repack $_repack_flags");
1541 }
1542 return $commit;
1543}
1544
1545sub set_commit_env {
1546 my ($log_msg) = @_;
1547 my $author = $log_msg->{author};
1548 my ($name,$email) = defined $users{$author} ? @{$users{$author}}
1549 : ($author,"$author\@$SVN_UUID");
1550 $ENV{GIT_AUTHOR_NAME} = $ENV{GIT_COMMITTER_NAME} = $name;
1551 $ENV{GIT_AUTHOR_EMAIL} = $ENV{GIT_COMMITTER_EMAIL} = $email;
1552 $ENV{GIT_AUTHOR_DATE} = $ENV{GIT_COMMITTER_DATE} = $log_msg->{date};
1553}
1554
1555sub apply_mod_line_blob {
1556 my $m = shift;
1557 if ($m->{mode_b} =~ /^120/) {
1558 blob_to_symlink($m->{sha1_b}, $m->{file_b});
1559 } else {
1560 blob_to_file($m->{sha1_b}, $m->{file_b});
1561 }
1562}
1563
1564sub blob_to_symlink {
1565 my ($blob, $link) = @_;
1566 defined $link or croak "\$link not defined!\n";
1567 croak "Not a sha1: $blob\n" unless $blob =~ /^$sha1$/o;
1568 if (-l $link || -f _) {
1569 unlink $link or croak $!;
1570 }
1571
1572 my $dest = `git-cat-file blob $blob`; # no newline, so no chomp
1573 symlink $dest, $link or croak $!;
1574}
1575
1576sub blob_to_file {
1577 my ($blob, $file) = @_;
1578 defined $file or croak "\$file not defined!\n";
1579 croak "Not a sha1: $blob\n" unless $blob =~ /^$sha1$/o;
1580 if (-l $file || -f _) {
1581 unlink $file or croak $!;
1582 }
1583
1584 open my $blob_fh, '>', $file or croak "$!: $file\n";
1585 my $pid = fork;
1586 defined $pid or croak $!;
1587
1588 if ($pid == 0) {
1589 open STDOUT, '>&', $blob_fh or croak $!;
1590 exec('git-cat-file','blob',$blob) or croak $!;
1591 }
1592 waitpid $pid, 0;
1593 croak $? if $?;
1594
1595 close $blob_fh or croak $!;
1596}
1597
1598sub safe_qx {
1599 my $pid = open my $child, '-|';
1600 defined $pid or croak $!;
1601 if ($pid == 0) {
1602 exec(@_) or croak $!;
1603 }
1604 my @ret = (<$child>);
1605 close $child or croak $?;
1606 die $? if $?; # just in case close didn't error out
1607 return wantarray ? @ret : join('',@ret);
1608}
1609
1610sub svn_compat_check {
1611 my @co_help = safe_qx(qw(svn co -h));
1612 unless (grep /ignore-externals/,@co_help) {
1613 print STDERR "W: Installed svn version does not support ",
1614 "--ignore-externals\n";
1615 $_no_ignore_ext = 1;
1616 }
1617 if (grep /usage: checkout URL\[\@REV\]/,@co_help) {
1618 $_svn_co_url_revs = 1;
1619 }
1620 if (grep /\[TARGET\[\@REV\]\.\.\.\]/, `svn propget -h`) {
1621 $_svn_pg_peg_revs = 1;
1622 }
1623
1624 # I really, really hope nobody hits this...
1625 unless (grep /stop-on-copy/, (safe_qx(qw(svn log -h)))) {
1626 print STDERR <<'';
1627W: The installed svn version does not support the --stop-on-copy flag in
1628 the log command.
1629 Lets hope the directory you're tracking is not a branch or tag
1630 and was never moved within the repository...
1631
1632 $_no_stop_copy = 1;
1633 }
1634}
1635
1636# *sigh*, new versions of svn won't honor -r<rev> without URL@<rev>,
1637# (and they won't honor URL@<rev> without -r<rev>, too!)
1638sub svn_cmd_checkout {
1639 my ($url, $rev, $dir) = @_;
1640 my @cmd = ('svn','co', "-r$rev");
1641 push @cmd, '--ignore-externals' unless $_no_ignore_ext;
1642 $url .= "\@$rev" if $_svn_co_url_revs;
1643 sys(@cmd, $url, $dir);
1644}
1645
1646sub check_upgrade_needed {
1647 my $old = eval {
1648 my $pid = open my $child, '-|';
1649 defined $pid or croak $!;
1650 if ($pid == 0) {
1651 close STDERR;
1652 exec('git-rev-parse',"$GIT_SVN-HEAD") or croak $!;
1653 }
1654 my @ret = (<$child>);
1655 close $child or croak $?;
1656 die $? if $?; # just in case close didn't error out
1657 return wantarray ? @ret : join('',@ret);
1658 };
1659 return unless $old;
1660 my $head = eval { safe_qx('git-rev-parse',"refs/remotes/$GIT_SVN") };
1661 if ($@ || !$head) {
1662 print STDERR "Please run: $0 rebuild --upgrade\n";
1663 exit 1;
1664 }
1665}
1666
1667# fills %tree_map with a reverse mapping of trees to commits. Useful
1668# for finding parents to commit on.
1669sub map_tree_joins {
1670 my %seen;
1671 foreach my $br (@_branch_from) {
1672 my $pid = open my $pipe, '-|';
1673 defined $pid or croak $!;
1674 if ($pid == 0) {
1675 exec(qw(git-rev-list --topo-order --pretty=raw), $br)
1676 or croak $!;
1677 }
1678 while (<$pipe>) {
1679 if (/^commit ($sha1)$/o) {
1680 my $commit = $1;
1681
1682 # if we've seen a commit,
1683 # we've seen its parents
1684 last if $seen{$commit};
1685 my ($tree) = (<$pipe> =~ /^tree ($sha1)$/o);
1686 unless (defined $tree) {
1687 die "Failed to parse commit $commit\n";
1688 }
1689 push @{$tree_map{$tree}}, $commit;
1690 $seen{$commit} = 1;
1691 }
1692 }
1693 close $pipe; # we could be breaking the pipe early
1694 }
1695}
1696
1697sub load_all_refs {
1698 if (@_branch_from) {
1699 print STDERR '--branch|-b parameters are ignored when ',
1700 "--branch-all-refs|-B is passed\n";
1701 }
1702
1703 # don't worry about rev-list on non-commit objects/tags,
1704 # it shouldn't blow up if a ref is a blob or tree...
1705 chomp(@_branch_from = `git-rev-parse --symbolic --all`);
1706}
1707
1708# '<svn username> = real-name <email address>' mapping based on git-svnimport:
1709sub load_authors {
1710 open my $authors, '<', $_authors or die "Can't open $_authors $!\n";
1711 while (<$authors>) {
1712 chomp;
1713 next unless /^(\S+?)\s*=\s*(.+?)\s*<(.+)>\s*$/;
1714 my ($user, $name, $email) = ($1, $2, $3);
1715 $users{$user} = [$name, $email];
1716 }
1717 close $authors or croak $!;
1718}
1719
1720sub rload_authors {
1721 open my $authors, '<', $_authors or die "Can't open $_authors $!\n";
1722 while (<$authors>) {
1723 chomp;
1724 next unless /^(\S+?)\s*=\s*(.+?)\s*<(.+)>\s*$/;
1725 my ($user, $name, $email) = ($1, $2, $3);
1726 $rusers{"$name <$email>"} = $user;
1727 }
1728 close $authors or croak $!;
1729}
1730
1731sub svn_propget_base {
1732 my ($p, $f) = @_;
1733 $f .= '@BASE' if $_svn_pg_peg_revs;
1734 return safe_qx(qw/svn propget/, $p, $f);
1735}
1736
1737sub git_svn_each {
1738 my $sub = shift;
1739 foreach (`git-rev-parse --symbolic --all`) {
1740 next unless s#^refs/remotes/##;
1741 chomp $_;
1742 next unless -f "$GIT_DIR/svn/$_/info/url";
1743 &$sub($_);
1744 }
1745}
1746
1747sub migration_check {
1748 return if (-d "$GIT_DIR/svn" || !-d $GIT_DIR);
1749 print "Upgrading repository...\n";
1750 unless (-d "$GIT_DIR/svn") {
1751 mkdir "$GIT_DIR/svn" or croak $!;
1752 }
1753 print "Data from a previous version of git-svn exists, but\n\t",
1754 "$GIT_SVN_DIR\n\t(required for this version ",
1755 "($VERSION) of git-svn) does not.\n";
1756
1757 foreach my $x (`git-rev-parse --symbolic --all`) {
1758 next unless $x =~ s#^refs/remotes/##;
1759 chomp $x;
1760 next unless -f "$GIT_DIR/$x/info/url";
1761 my $u = eval { file_to_s("$GIT_DIR/$x/info/url") };
1762 next unless $u;
1763 my $dn = dirname("$GIT_DIR/svn/$x");
1764 mkpath([$dn]) unless -d $dn;
1765 rename "$GIT_DIR/$x", "$GIT_DIR/svn/$x" or croak "$!: $x";
1766 my ($url, $path) = repo_path_split($u);
1767 s_to_file($url, "$GIT_DIR/svn/$x/info/repo_url");
1768 s_to_file($path, "$GIT_DIR/svn/$x/info/repo_path");
1769 }
1770 print "Done upgrading.\n";
1771}
1772
1773sub find_rev_before {
1774 my ($r, $git_svn_id) = @_;
1775 my @revs = map { basename $_ } <$GIT_DIR/svn/$git_svn_id/revs/*>;
1776 foreach my $r0 (sort { $b <=> $a } @revs) {
1777 next if $r0 >= $r;
1778 return ($r0, file_to_s("$GIT_DIR/svn/$git_svn_id/revs/$r0"));
1779 }
1780 return (undef, undef);
1781}
1782
1783sub init_vars {
1784 $GIT_SVN ||= $ENV{GIT_SVN_ID} || 'git-svn';
1785 $GIT_SVN_DIR = "$GIT_DIR/svn/$GIT_SVN";
1786 $GIT_SVN_INDEX = "$GIT_SVN_DIR/index";
1787 $SVN_URL = undef;
1788 $REV_DIR = "$GIT_SVN_DIR/revs";
1789 $SVN_WC = "$GIT_SVN_DIR/tree";
1790}
1791
1792# convert GetOpt::Long specs for use by git-repo-config
1793sub read_repo_config {
1794 return unless -d $GIT_DIR;
1795 my $opts = shift;
1796 foreach my $o (keys %$opts) {
1797 my $v = $opts->{$o};
1798 my ($key) = ($o =~ /^([a-z\-]+)/);
1799 $key =~ s/-//g;
1800 my $arg = 'git-repo-config';
1801 $arg .= ' --int' if ($o =~ /[:=]i$/);
1802 $arg .= ' --bool' if ($o !~ /[:=][sfi]$/);
1803 if (ref $v eq 'ARRAY') {
1804 chomp(my @tmp = `$arg --get-all svn.$key`);
1805 @$v = @tmp if @tmp;
1806 } else {
1807 chomp(my $tmp = `$arg --get svn.$key`);
1808 if ($tmp && !($arg =~ / --bool / && $tmp eq 'false')) {
1809 $$v = $tmp;
1810 }
1811 }
1812 }
1813}
1814
1815sub set_default_vals {
1816 if (defined $_repack) {
1817 $_repack = 1000 if ($_repack <= 0);
1818 $_repack_nr = $_repack;
1819 $_repack_flags ||= '';
1820 }
1821}
1822
1823sub read_grafts {
1824 my $gr_file = shift;
1825 my ($grafts, $comments) = ({}, {});
1826 if (open my $fh, '<', $gr_file) {
1827 my @tmp;
1828 while (<$fh>) {
1829 if (/^($sha1)\s+/) {
1830 my $c = $1;
1831 if (@tmp) {
1832 @{$comments->{$c}} = @tmp;
1833 @tmp = ();
1834 }
1835 foreach my $p (split /\s+/, $_) {
1836 $grafts->{$c}->{$p} = 1;
1837 }
1838 } else {
1839 push @tmp, $_;
1840 }
1841 }
1842 close $fh or croak $!;
1843 @{$comments->{'END'}} = @tmp if @tmp;
1844 }
1845 return ($grafts, $comments);
1846}
1847
1848sub write_grafts {
1849 my ($grafts, $comments, $gr_file) = @_;
1850
1851 open my $fh, '>', $gr_file or croak $!;
1852 foreach my $c (sort keys %$grafts) {
1853 if ($comments->{$c}) {
1854 print $fh $_ foreach @{$comments->{$c}};
1855 }
1856 my $p = $grafts->{$c};
1857 delete $p->{$c}; # commits are not self-reproducing...
1858 my $pid = open my $ch, '-|';
1859 defined $pid or croak $!;
1860 if (!$pid) {
1861 exec(qw/git-cat-file commit/, $c) or croak $!;
1862 }
1863 while (<$ch>) {
1864 if (/^parent ([a-f\d]{40})/) {
1865 $p->{$1} = 1;
1866 } else {
1867 last unless /^\S/i;
1868 }
1869 }
1870 close $ch; # breaking the pipe
1871 print $fh $c, ' ', join(' ', sort keys %$p),"\n";
1872 }
1873 if ($comments->{'END'}) {
1874 print $fh $_ foreach @{$comments->{'END'}};
1875 }
1876 close $fh or croak $!;
1877}
1878
1879sub read_url_paths {
1880 my $l_map = {};
1881 git_svn_each(sub { my $x = shift;
1882 my $u = file_to_s("$GIT_DIR/svn/$x/info/repo_url");
1883 my $p = file_to_s("$GIT_DIR/svn/$x/info/repo_path");
1884 # we hate trailing slashes
1885 if ($u =~ s#(?:^\/+|\/+$)##g) {
1886 s_to_file($u,"$GIT_DIR/svn/$x/info/repo_url");
1887 }
1888 if ($p =~ s#(?:^\/+|\/+$)##g) {
1889 s_to_file($p,"$GIT_DIR/svn/$x/info/repo_path");
1890 }
1891 $l_map->{$u}->{$p} = $x;
1892 });
1893 return $l_map;
1894}
1895
1896sub extract_metadata {
1897 my $id = shift;
1898 my ($url, $rev, $uuid) = ($id =~ /^git-svn-id:\s(\S+?)\@(\d+)
1899 \s([a-f\d\-]+)$/x);
1900 if (!$rev || !$uuid || !$url) {
1901 # some of the original repositories I made had
1902 # indentifiers like this:
1903 ($rev, $uuid) = ($id =~/^git-svn-id:\s(\d+)\@([a-f\d\-]+)/);
1904 }
1905 return ($url, $rev, $uuid);
1906}
1907
1908sub tz_to_s_offset {
1909 my ($tz) = @_;
1910 $tz =~ s/(\d\d)$//;
1911 return ($1 * 60) + ($tz * 3600);
1912}
1913
1914sub setup_pager { # translated to Perl from pager.c
1915 return unless (-t *STDOUT);
1916 my $pager = $ENV{PAGER};
1917 if (!defined $pager) {
1918 $pager = 'less';
1919 } elsif (length $pager == 0 || $pager eq 'cat') {
1920 return;
1921 }
1922 pipe my $rfd, my $wfd or return;
1923 defined(my $pid = fork) or croak $!;
1924 if (!$pid) {
1925 open STDOUT, '>&', $wfd or croak $!;
1926 return;
1927 }
1928 open STDIN, '<&', $rfd or croak $!;
1929 $ENV{LESS} ||= '-S';
1930 exec $pager or croak "Can't run pager: $!\n";;
1931}
1932
1933sub get_author_info {
1934 my ($dest, $author, $t, $tz) = @_;
1935 $author =~ s/(?:^\s*|\s*$)//g;
1936 my $_a;
1937 if ($_authors) {
1938 $_a = $rusers{$author} || undef;
1939 }
1940 if (!$_a) {
1941 ($_a) = ($author =~ /<([^>]+)\@[^>]+>$/);
1942 }
1943 $dest->{t} = $t;
1944 $dest->{tz} = $tz;
1945 $dest->{a} = $_a;
1946 # Date::Parse isn't in the standard Perl distro :(
1947 if ($tz =~ s/^\+//) {
1948 $t += tz_to_s_offset($tz);
1949 } elsif ($tz =~ s/^\-//) {
1950 $t -= tz_to_s_offset($tz);
1951 }
1952 $dest->{t_utc} = $t;
1953}
1954
1955sub process_commit {
1956 my ($c, $r_min, $r_max, $defer) = @_;
1957 if (defined $r_min && defined $r_max) {
1958 if ($r_min == $c->{r} && $r_min == $r_max) {
1959 show_commit($c);
1960 return 0;
1961 }
1962 return 1 if $r_min == $r_max;
1963 if ($r_min < $r_max) {
1964 # we need to reverse the print order
1965 return 0 if (defined $_limit && --$_limit < 0);
1966 push @$defer, $c;
1967 return 1;
1968 }
1969 if ($r_min != $r_max) {
1970 return 1 if ($r_min < $c->{r});
1971 return 1 if ($r_max > $c->{r});
1972 }
1973 }
1974 return 0 if (defined $_limit && --$_limit < 0);
1975 show_commit($c);
1976 return 1;
1977}
1978
1979sub show_commit {
1980 my $c = shift;
1981 if ($_oneline) {
1982 my $x = "\n";
1983 if (my $l = $c->{l}) {
1984 while ($l->[0] =~ /^\s*$/) { shift @$l }
1985 $x = $l->[0];
1986 }
1987 $_l_fmt ||= 'A' . length($c->{r});
1988 print 'r',pack($_l_fmt, $c->{r}),' | ';
1989 print "$c->{c} | " if $_show_commit;
1990 print $x;
1991 } else {
1992 show_commit_normal($c);
1993 }
1994}
1995
1996sub show_commit_normal {
1997 my ($c) = @_;
1998 print '-' x72, "\nr$c->{r} | ";
1999 print "$c->{c} | " if $_show_commit;
2000 print "$c->{a} | ", strftime("%Y-%m-%d %H:%M:%S %z (%a, %d %b %Y)",
2001 localtime($c->{t_utc})), ' | ';
2002 my $nr_line = 0;
2003
2004 if (my $l = $c->{l}) {
2005 while ($l->[$#$l] eq "\n" && $l->[($#$l - 1)] eq "\n") {
2006 pop @$l;
2007 }
2008 $nr_line = scalar @$l;
2009 if (!$nr_line) {
2010 print "1 line\n\n\n";
2011 } else {
2012 if ($nr_line == 1) {
2013 $nr_line = '1 line';
2014 } else {
2015 $nr_line .= ' lines';
2016 }
2017 print $nr_line, "\n\n";
2018 print $_ foreach @$l;
2019 }
2020 } else {
2021 print "1 line\n\n";
2022
2023 }
2024 foreach my $x (qw/raw diff/) {
2025 if ($c->{$x}) {
2026 print "\n";
2027 print $_ foreach @{$c->{$x}}
2028 }
2029 }
2030}
2031
2032__END__
2033
2034Data structures:
2035
2036$svn_log hashref (as returned by svn_log_raw)
2037{
2038 fh => file handle of the log file,
2039 state => state of the log file parser (sep/msg/rev/msg_start...)
2040}
2041
2042$log_msg hashref as returned by next_log_entry($svn_log)
2043{
2044 msg => 'whitespace-formatted log entry
2045', # trailing newline is preserved
2046 revision => '8', # integer
2047 date => '2004-02-24T17:01:44.108345Z', # commit date
2048 author => 'committer name'
2049};
2050
2051
2052@mods = array of diff-index line hashes, each element represents one line
2053 of diff-index output
2054
2055diff-index line ($m hash)
2056{
2057 mode_a => first column of diff-index output, no leading ':',
2058 mode_b => second column of diff-index output,
2059 sha1_b => sha1sum of the final blob,
2060 chg => change type [MCRADT],
2061 file_a => original file name of a file (iff chg is 'C' or 'R')
2062 file_b => new/current file name of a file (any chg)
2063}
2064;