03416aeec157e1e364b92fde434ea922ccb16e8e
   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;