1#!/usr/bin/env perl 2# Copyright (C) 2006, Eric Wong <normalperson@yhbt.net> 3# License: GPL v2 or later 4use5.008; 5use warnings; 6use strict; 7use vars qw/$AUTHOR $VERSION 8$sha1 $sha1_short $_revision $_repository 9$_q $_authors $_authors_prog %users/; 10$AUTHOR='Eric Wong <normalperson@yhbt.net>'; 11$VERSION='@@GIT_VERSION@@'; 12 13# From which subdir have we been invoked? 14my$cmd_dir_prefix=eval{ 15 command_oneline([qw/rev-parse --show-prefix/], STDERR =>0) 16} ||''; 17 18my$git_dir_user_set=1ifdefined$ENV{GIT_DIR}; 19$ENV{GIT_DIR} ||='.git'; 20$Git::SVN::default_repo_id ='svn'; 21$Git::SVN::default_ref_id =$ENV{GIT_SVN_ID} ||'git-svn'; 22$Git::SVN::Ra::_log_window_size =100; 23$Git::SVN::_minimize_url ='unset'; 24 25if(!exists$ENV{SVN_SSH}) { 26if(exists$ENV{GIT_SSH}) { 27$ENV{SVN_SSH} =$ENV{GIT_SSH}; 28if($^Oeq'msys') { 29$ENV{SVN_SSH} =~s/\\/\\\\/g; 30$ENV{SVN_SSH} =~s/(.*)/"$1"/; 31} 32} 33} 34 35$Git::SVN::Log::TZ =$ENV{TZ}; 36$ENV{TZ} ='UTC'; 37$| =1;# unbuffer STDOUT 38 39sub fatal (@) {print STDERR "@_\n";exit1} 40sub _req_svn { 41require SVN::Core;# use()-ing this causes segfaults for me... *shrug* 42require SVN::Ra; 43require SVN::Delta; 44if($SVN::Core::VERSION lt'1.1.0') { 45 fatal "Need SVN::Core 1.1.0 or better (got$SVN::Core::VERSION)"; 46} 47} 48my$can_compress=eval{require Compress::Zlib;1}; 49push@Git::SVN::Ra::ISA,'SVN::Ra'; 50push@SVN::Git::Editor::ISA,'SVN::Delta::Editor'; 51push@SVN::Git::Fetcher::ISA,'SVN::Delta::Editor'; 52use Carp qw/croak/; 53use Digest::MD5; 54use IO::File qw//; 55use File::Basename qw/dirname basename/; 56use File::Path qw/mkpath/; 57use File::Spec; 58use File::Find; 59use Getopt::Long qw/:config gnu_getopt no_ignore_case auto_abbrev/; 60use IPC::Open3; 61use Git; 62use Memoize;# core since 5.8.0, Jul 2002 63 64BEGIN{ 65# import functions from Git into our packages, en masse 66no strict 'refs'; 67foreach(qw/command command_oneline command_noisy command_output_pipe 68 command_input_pipe command_close_pipe 69 command_bidi_pipe command_close_bidi_pipe/) { 70formy$package(qw(SVN::Git::Editor SVN::Git::Fetcher 71 Git::SVN::Migration Git::SVN::Log Git::SVN), 72 __PACKAGE__) { 73*{"${package}::$_"} = \&{"Git::$_"}; 74} 75} 76 Memoize::memoize 'Git::config'; 77 Memoize::memoize 'Git::config_bool'; 78} 79 80my($SVN); 81 82$sha1=qr/[a-f\d]{40}/; 83$sha1_short=qr/[a-f\d]{4,40}/; 84my($_stdin,$_help,$_edit, 85$_message,$_file,$_branch_dest, 86$_template,$_shared, 87$_version,$_fetch_all,$_no_rebase,$_fetch_parent, 88$_merge,$_strategy,$_dry_run,$_local, 89$_prefix,$_no_checkout,$_url,$_verbose, 90$_git_format,$_commit_url,$_tag,$_merge_info,$_interactive); 91$Git::SVN::_follow_parent =1; 92$SVN::Git::Fetcher::_placeholder_filename =".gitignore"; 93$_q||=0; 94my%remote_opts= ('username=s'=> \$Git::SVN::Prompt::_username, 95'config-dir=s'=> \$Git::SVN::Ra::config_dir, 96'no-auth-cache'=> \$Git::SVN::Prompt::_no_auth_cache, 97'ignore-paths=s'=> \$SVN::Git::Fetcher::_ignore_regex ); 98my%fc_opts= ('follow-parent|follow!'=> \$Git::SVN::_follow_parent, 99'authors-file|A=s'=> \$_authors, 100'authors-prog=s'=> \$_authors_prog, 101'repack:i'=> \$Git::SVN::_repack, 102'noMetadata'=> \$Git::SVN::_no_metadata, 103'useSvmProps'=> \$Git::SVN::_use_svm_props, 104'useSvnsyncProps'=> \$Git::SVN::_use_svnsync_props, 105'log-window-size=i'=> \$Git::SVN::Ra::_log_window_size, 106'no-checkout'=> \$_no_checkout, 107'quiet|q+'=> \$_q, 108'repack-flags|repack-args|repack-opts=s'=> 109 \$Git::SVN::_repack_flags, 110'use-log-author'=> \$Git::SVN::_use_log_author, 111'add-author-from'=> \$Git::SVN::_add_author_from, 112'localtime'=> \$Git::SVN::_localtime, 113%remote_opts); 114 115my($_trunk,@_tags,@_branches,$_stdlayout); 116my%icv; 117my%init_opts= ('template=s'=> \$_template,'shared:s'=> \$_shared, 118'trunk|T=s'=> \$_trunk,'tags|t=s@'=> \@_tags, 119'branches|b=s@'=> \@_branches,'prefix=s'=> \$_prefix, 120'stdlayout|s'=> \$_stdlayout, 121'minimize-url|m!'=> \$Git::SVN::_minimize_url, 122'no-metadata'=>sub{$icv{noMetadata} =1}, 123'use-svm-props'=>sub{$icv{useSvmProps} =1}, 124'use-svnsync-props'=>sub{$icv{useSvnsyncProps} =1}, 125'rewrite-root=s'=>sub{$icv{rewriteRoot} =$_[1] }, 126'rewrite-uuid=s'=>sub{$icv{rewriteUUID} =$_[1] }, 127%remote_opts); 128my%cmt_opts= ('edit|e'=> \$_edit, 129'rmdir'=> \$SVN::Git::Editor::_rmdir, 130'find-copies-harder'=> \$SVN::Git::Editor::_find_copies_harder, 131'l=i'=> \$SVN::Git::Editor::_rename_limit, 132'copy-similarity|C=i'=> \$SVN::Git::Editor::_cp_similarity 133); 134 135my%cmd= ( 136 fetch => [ \&cmd_fetch,"Download new revisions from SVN", 137{'revision|r=s'=> \$_revision, 138'fetch-all|all'=> \$_fetch_all, 139'parent|p'=> \$_fetch_parent, 140%fc_opts} ], 141 clone => [ \&cmd_clone,"Initialize and fetch revisions", 142{'revision|r=s'=> \$_revision, 143'preserve-empty-dirs'=> 144 \$SVN::Git::Fetcher::_preserve_empty_dirs, 145'placeholder-filename=s'=> 146 \$SVN::Git::Fetcher::_placeholder_filename, 147%fc_opts,%init_opts} ], 148 init => [ \&cmd_init,"Initialize a repo for tracking". 149" (requires URL argument)", 150 \%init_opts], 151'multi-init'=> [ \&cmd_multi_init, 152"Deprecated alias for ". 153"'$0init -T<trunk> -b<branches> -t<tags>'", 154 \%init_opts], 155 dcommit => [ \&cmd_dcommit, 156'Commit several diffs to merge with upstream', 157{'merge|m|M'=> \$_merge, 158'strategy|s=s'=> \$_strategy, 159'verbose|v'=> \$_verbose, 160'dry-run|n'=> \$_dry_run, 161'fetch-all|all'=> \$_fetch_all, 162'commit-url=s'=> \$_commit_url, 163'revision|r=i'=> \$_revision, 164'no-rebase'=> \$_no_rebase, 165'mergeinfo=s'=> \$_merge_info, 166'interactive|i'=> \$_interactive, 167%cmt_opts,%fc_opts} ], 168 branch => [ \&cmd_branch, 169'Create a branch in the SVN repository', 170{'message|m=s'=> \$_message, 171'destination|d=s'=> \$_branch_dest, 172'dry-run|n'=> \$_dry_run, 173'tag|t'=> \$_tag, 174'username=s'=> \$Git::SVN::Prompt::_username, 175'commit-url=s'=> \$_commit_url} ], 176 tag => [sub{$_tag=1; cmd_branch(@_) }, 177'Create a tag in the SVN repository', 178{'message|m=s'=> \$_message, 179'destination|d=s'=> \$_branch_dest, 180'dry-run|n'=> \$_dry_run, 181'username=s'=> \$Git::SVN::Prompt::_username, 182'commit-url=s'=> \$_commit_url} ], 183'set-tree'=> [ \&cmd_set_tree, 184"Set an SVN repository to a git tree-ish", 185{'stdin'=> \$_stdin,%cmt_opts,%fc_opts, } ], 186'create-ignore'=> [ \&cmd_create_ignore, 187'Create a .gitignore per svn:ignore', 188{'revision|r=i'=> \$_revision 189} ], 190'mkdirs'=> [ \&cmd_mkdirs , 191"recreate empty directories after a checkout", 192{'revision|r=i'=> \$_revision} ], 193'propget'=> [ \&cmd_propget, 194'Print the value of a property on a file or directory', 195{'revision|r=i'=> \$_revision} ], 196'proplist'=> [ \&cmd_proplist, 197'List all properties of a file or directory', 198{'revision|r=i'=> \$_revision} ], 199'show-ignore'=> [ \&cmd_show_ignore,"Show svn:ignore listings", 200{'revision|r=i'=> \$_revision 201} ], 202'show-externals'=> [ \&cmd_show_externals,"Show svn:externals listings", 203{'revision|r=i'=> \$_revision 204} ], 205'multi-fetch'=> [ \&cmd_multi_fetch, 206"Deprecated alias for$0fetch --all", 207{'revision|r=s'=> \$_revision,%fc_opts} ], 208'migrate'=> [sub{ }, 209# no-op, we automatically run this anyways, 210'Migrate configuration/metadata/layout from 211 previous versions of git-svn', 212{'minimize'=> \$Git::SVN::Migration::_minimize, 213%remote_opts} ], 214'log'=> [ \&Git::SVN::Log::cmd_show_log,'Show commit logs', 215{'limit=i'=> \$Git::SVN::Log::limit, 216'revision|r=s'=> \$_revision, 217'verbose|v'=> \$Git::SVN::Log::verbose, 218'incremental'=> \$Git::SVN::Log::incremental, 219'oneline'=> \$Git::SVN::Log::oneline, 220'show-commit'=> \$Git::SVN::Log::show_commit, 221'non-recursive'=> \$Git::SVN::Log::non_recursive, 222'authors-file|A=s'=> \$_authors, 223'color'=> \$Git::SVN::Log::color, 224'pager=s'=> \$Git::SVN::Log::pager 225} ], 226'find-rev'=> [ \&cmd_find_rev, 227"Translate between SVN revision numbers and tree-ish", 228{} ], 229'rebase'=> [ \&cmd_rebase,"Fetch and rebase your working directory", 230{'merge|m|M'=> \$_merge, 231'verbose|v'=> \$_verbose, 232'strategy|s=s'=> \$_strategy, 233'local|l'=> \$_local, 234'fetch-all|all'=> \$_fetch_all, 235'dry-run|n'=> \$_dry_run, 236%fc_opts} ], 237'commit-diff'=> [ \&cmd_commit_diff, 238'Commit a diff between two trees', 239{'message|m=s'=> \$_message, 240'file|F=s'=> \$_file, 241'revision|r=s'=> \$_revision, 242%cmt_opts} ], 243'info'=> [ \&cmd_info, 244"Show info about the latest SVN revision 245 on the current branch", 246{'url'=> \$_url, } ], 247'blame'=> [ \&Git::SVN::Log::cmd_blame, 248"Show what revision and author last modified each line of a file", 249{'git-format'=> \$_git_format} ], 250'reset'=> [ \&cmd_reset, 251"Undo fetches back to the specified SVN revision", 252{'revision|r=s'=> \$_revision, 253'parent|p'=> \$_fetch_parent} ], 254'gc'=> [ \&cmd_gc, 255"Compress unhandled.log files in .git/svn and remove ". 256"index files in .git/svn", 257{} ], 258); 259 260use Term::ReadLine; 261package FakeTerm; 262sub new { 263my($class,$reason) =@_; 264returnbless \$reason,shift; 265} 266subreadline{ 267my$self=shift; 268die"Cannot use readline on FakeTerm:$$self"; 269} 270package main; 271 272my$term=eval{ 273$ENV{"GIT_SVN_NOTTY"} 274? new Term::ReadLine 'git-svn', \*STDIN, \*STDOUT 275: new Term::ReadLine 'git-svn'; 276}; 277if($@) { 278$term= new FakeTerm "$@: going non-interactive"; 279} 280 281my$cmd; 282for(my$i=0;$i<@ARGV;$i++) { 283if(defined$cmd{$ARGV[$i]}) { 284$cmd=$ARGV[$i]; 285splice@ARGV,$i,1; 286last; 287}elsif($ARGV[$i]eq'help') { 288$cmd=$ARGV[$i+1]; 289 usage(0); 290} 291}; 292 293# make sure we're always running at the top-level working directory 294unless($cmd&&$cmd=~/(?:clone|init|multi-init)$/) { 295unless(-d $ENV{GIT_DIR}) { 296if($git_dir_user_set) { 297die"GIT_DIR=$ENV{GIT_DIR} explicitly set, ", 298"but it is not a directory\n"; 299} 300my$git_dir=delete$ENV{GIT_DIR}; 301my$cdup=undef; 302 git_cmd_try { 303$cdup= command_oneline(qw/rev-parse --show-cdup/); 304$git_dir='.'unless($cdup); 305chomp$cdupif($cdup); 306$cdup="."unless($cdup&&length$cdup); 307}"Already at toplevel, but$git_dirnot found\n"; 308chdir$cdupor die"Unable to chdir up to '$cdup'\n"; 309unless(-d $git_dir) { 310die"$git_dirstill not found after going to ", 311"'$cdup'\n"; 312} 313$ENV{GIT_DIR} =$git_dir; 314} 315$_repository= Git->repository(Repository =>$ENV{GIT_DIR}); 316} 317 318my%opts= %{$cmd{$cmd}->[2]}if(defined$cmd); 319 320read_git_config(\%opts); 321if($cmd&& ($cmdeq'log'||$cmdeq'blame')) { 322 Getopt::Long::Configure('pass_through'); 323} 324my$rv= GetOptions(%opts,'h|H'=> \$_help,'version|V'=> \$_version, 325'minimize-connections'=> \$Git::SVN::Migration::_minimize, 326'id|i=s'=> \$Git::SVN::default_ref_id, 327'svn-remote|remote|R=s'=>sub{ 328$Git::SVN::no_reuse_existing =1; 329$Git::SVN::default_repo_id =$_[1] }); 330exit1if(!$rv&&$cmd&&$cmdne'log'); 331 332usage(0)if$_help; 333version()if$_version; 334usage(1)unlessdefined$cmd; 335load_authors()if$_authors; 336if(defined$_authors_prog) { 337$_authors_prog="'". File::Spec->rel2abs($_authors_prog) ."'"; 338} 339 340unless($cmd=~/^(?:clone|init|multi-init|commit-diff)$/) { 341 Git::SVN::Migration::migration_check(); 342} 343Git::SVN::init_vars(); 344eval{ 345 Git::SVN::verify_remotes_sanity(); 346$cmd{$cmd}->[0]->(@ARGV); 347}; 348fatal $@if$@; 349post_fetch_checkout(); 350exit0; 351 352####################### primary functions ###################### 353sub usage { 354my$exit=shift||0; 355my$fd=$exit? \*STDERR : \*STDOUT; 356print$fd<<""; 357git-svn - bidirectional operations between a single Subversion tree and git 358Usage: git svn <command> [options] [arguments]\n 359 360print$fd"Available commands:\n"unless$cmd; 361 362foreach(sort keys%cmd) { 363next if$cmd&&$cmdne$_; 364next if/^multi-/;# don't show deprecated commands 365print$fd' ',pack('A17',$_),$cmd{$_}->[1],"\n"; 366foreach(sort keys%{$cmd{$_}->[2]}) { 367# mixed-case options are for .git/config only 368next if/[A-Z]/&&/^[a-z]+$/i; 369# prints out arguments as they should be passed: 370my$x= s#[:=]s$## ? '<arg>' : s#[:=]i$## ? '<num>' : ''; 371print$fd' ' x 21,join(', ',map{length$_>1? 372"--$_":"-$_"} 373split/\|/,$_),"$x\n"; 374} 375} 376print$fd<<""; 377\nGIT_SVN_ID may be set in the environment or via the --id/-i switch to an 378arbitrary identifier if you're tracking multiple SVN branches/repositories in 379one git repository and want to keep them separate. See git-svn(1) for more 380information. 381 382 exit$exit; 383} 384 385sub version { 386 ::_req_svn(); 387 print "git-svn version$VERSION(svn$SVN::Core::VERSION)\n"; 388 exit 0; 389} 390 391sub ask { 392 my ($prompt,%arg) =@_; 393 my$valid_re=$arg{valid_re}; 394 my$default=$arg{default}; 395 my$resp; 396 my$i= 0; 397 398 if ( !( defined($term->IN) 399 && defined( fileno($term->IN) ) 400 && defined($term->OUT ) 401 && defined( fileno($term->OUT) ) ) ){ 402 return defined($default) ?$default: undef; 403 } 404 405 while ($i++< 10) { 406$resp=$term->readline($prompt); 407 if (!defined$resp) { # EOF 408 print "\n"; 409 return defined$default?$default: undef; 410 } 411 if ($respeq '' and defined$default) { 412 return$default; 413 } 414 if (!defined$valid_reor$resp=~ /$valid_re/) { 415 return$resp; 416 } 417 } 418 return undef; 419} 420 421sub do_git_init_db { 422 unless (-d$ENV{GIT_DIR}) { 423 my@init_db= ('init'); 424 push@init_db, "--template=$_template" if defined$_template; 425 if (defined$_shared) { 426 if ($_shared=~ /[a-z]/) { 427 push@init_db, "--shared=$_shared"; 428 } else { 429 push@init_db, "--shared"; 430 } 431 } 432 command_noisy(@init_db); 433$_repository= Git->repository(Repository => ".git"); 434 } 435 my$set; 436 my$pfx= "svn-remote.$Git::SVN::default_repo_id"; 437 foreach my$i(keys%icv) { 438 die "'$set' and '$i' cannot both be set\n" if$set; 439 next unless defined$icv{$i}; 440 command_noisy('config', "$pfx.$i",$icv{$i}); 441$set=$i; 442 } 443 my$ignore_regex= \$SVN::Git::Fetcher::_ignore_regex; 444 command_noisy('config', "$pfx.ignore-paths",$$ignore_regex) 445 if defined$$ignore_regex; 446 447 if (defined$SVN::Git::Fetcher::_preserve_empty_dirs) { 448 my$fname= \$SVN::Git::Fetcher::_placeholder_filename; 449 command_noisy('config', "$pfx.preserve-empty-dirs", 'true'); 450 command_noisy('config', "$pfx.placeholder-filename",$$fname); 451 } 452} 453 454sub init_subdir { 455 my$repo_path= shift or return; 456 mkpath([$repo_path]) unless -d$repo_path; 457 chdir$repo_pathor die "Couldn't chdir to $repo_path:$!\n"; 458$ENV{GIT_DIR} = '.git'; 459$_repository= Git->repository(Repository =>$ENV{GIT_DIR}); 460} 461 462sub cmd_clone { 463 my ($url,$path) =@_; 464 if (!defined$path&& 465 (defined$_trunk||@_branches||@_tags|| 466 defined$_stdlayout) && 467$url!~ m#^[a-z\+]+://#) { 468$path=$url; 469 } 470$path= basename($url) if !defined$path|| !length$path; 471 my$authors_absolute=$_authors? File::Spec->rel2abs($_authors) : ""; 472 cmd_init($url,$path); 473 command_oneline('config', 'svn.authorsfile',$authors_absolute) 474 if$_authors; 475 Git::SVN::fetch_all($Git::SVN::default_repo_id); 476} 477 478sub cmd_init { 479 if (defined$_stdlayout) { 480$_trunk= 'trunk' if (!defined$_trunk); 481@_tags= 'tags' if (!@_tags); 482@_branches= 'branches' if (!@_branches); 483 } 484 if (defined$_trunk||@_branches||@_tags) { 485 return cmd_multi_init(@_); 486 } 487 my$url= shift or die "SVN repository location required ", 488 "as a command-line argument\n"; 489$url= canonicalize_url($url); 490 init_subdir(@_); 491 do_git_init_db(); 492 493 if ($Git::SVN::_minimize_url eq 'unset') { 494$Git::SVN::_minimize_url = 0; 495 } 496 497 Git::SVN->init($url); 498} 499 500sub cmd_fetch { 501 if (grep /^\d+=./,@_) { 502 die "'<rev>=<commit>' fetch arguments are ", 503 "no longer supported.\n"; 504 } 505 my ($remote) =@_; 506 if (@_> 1) { 507 die "Usage:$0 fetch [--all] [--parent] [svn-remote]\n"; 508 } 509$Git::SVN::no_reuse_existing = undef; 510 if ($_fetch_parent) { 511 my ($url,$rev,$uuid,$gs) = working_head_info('HEAD'); 512 unless ($gs) { 513 die "Unable to determine upstream SVN information from ", 514 "working tree history\n"; 515 } 516 # just fetch, don't checkout. 517$_no_checkout= 'true'; 518$_fetch_all?$gs->fetch_all :$gs->fetch; 519 } elsif ($_fetch_all) { 520 cmd_multi_fetch(); 521 } else { 522$remote||=$Git::SVN::default_repo_id; 523 Git::SVN::fetch_all($remote, Git::SVN::read_all_remotes()); 524 } 525} 526 527sub cmd_set_tree { 528 my (@commits) =@_; 529 if ($_stdin|| !@commits) { 530 print "Reading from stdin...\n"; 531@commits= (); 532 while (<STDIN>) { 533 if (/\b($sha1_short)\b/o) { 534 unshift@commits,$1; 535 } 536 } 537 } 538 my@revs; 539 foreach my$c(@commits) { 540 my@tmp= command('rev-parse',$c); 541 if (scalar@tmp== 1) { 542 push@revs,$tmp[0]; 543 } elsif (scalar@tmp> 1) { 544 push@revs, reverse(command('rev-list',@tmp)); 545 } else { 546 fatal "Failed to rev-parse $c"; 547 } 548 } 549 my$gs= Git::SVN->new; 550 my ($r_last,$cmt_last) =$gs->last_rev_commit; 551$gs->fetch; 552 if (defined$gs->{last_rev} &&$r_last!=$gs->{last_rev}) { 553 fatal "There are new revisions that were fetched ", 554 "and need to be merged (or acknowledged)", 555 "before committing.\nlast rev:$r_last\n", 556 " current:$gs->{last_rev}"; 557 } 558$gs->set_tree($_) foreach@revs; 559 print "Done committing ",scalar@revs," revisions to SVN\n"; 560 unlink$gs->{index}; 561} 562 563sub split_merge_info_range { 564 my ($range) =@_; 565 if ($range=~ /(\d+)-(\d+)/) { 566 return (int($1), int($2)); 567 } else { 568 return (int($range), int($range)); 569 } 570} 571 572sub combine_ranges { 573 my ($in) =@_; 574 575 my@fnums= (); 576 my@arr= split(/,/,$in); 577 for my$element(@arr) { 578 my ($start,$end) = split_merge_info_range($element); 579 push@fnums,$start; 580 } 581 582 my@sorted=@arr[ sort { 583$fnums[$a] <=>$fnums[$b] 584 } 0..$#arr]; 585 586 my@return= (); 587 my$last= -1; 588 my$first= -1; 589 for my$element(@sorted) { 590 my ($start,$end) = split_merge_info_range($element); 591 592 if ($last== -1) { 593$first=$start; 594$last=$end; 595 next; 596 } 597 if ($start<=$last+1) { 598 if ($end>$last) { 599$last=$end; 600 } 601 next; 602 } 603 if ($first==$last) { 604 push@return, "$first"; 605 } else { 606 push@return, "$first-$last"; 607 } 608$first=$start; 609$last=$end; 610 } 611 612 if ($first!= -1) { 613 if ($first==$last) { 614 push@return, "$first"; 615 } else { 616 push@return, "$first-$last"; 617 } 618 } 619 620 return join(',',@return); 621} 622 623sub merge_revs_into_hash { 624 my ($hash,$minfo) =@_; 625 my@lines= split(' ',$minfo); 626 627 for my$line(@lines) { 628 my ($branchpath,$revs) = split(/:/,$line); 629 630 if (exists($hash->{$branchpath})) { 631 # Merge the two revision sets 632 my$combined= "$hash->{$branchpath},$revs"; 633$hash->{$branchpath} = combine_ranges($combined); 634 } else { 635 # Just do range combining for consolidation 636$hash->{$branchpath} = combine_ranges($revs); 637 } 638 } 639} 640 641sub merge_merge_info { 642 my ($mergeinfo_one,$mergeinfo_two) =@_; 643 my%result_hash= (); 644 645 merge_revs_into_hash(\%result_hash,$mergeinfo_one); 646 merge_revs_into_hash(\%result_hash,$mergeinfo_two); 647 648 my$result= ''; 649 # Sort below is for consistency's sake 650 for my$branchname(sort keys(%result_hash)) { 651 my$revlist=$result_hash{$branchname}; 652$result.= "$branchname:$revlist\n" 653 } 654 return$result; 655} 656 657sub populate_merge_info { 658 my ($d,$gs,$uuid,$linear_refs,$rewritten_parent) =@_; 659 660 my%parentshash; 661 read_commit_parents(\%parentshash,$d); 662 my@parents= @{$parentshash{$d}}; 663 if ($#parents> 0) { 664 # Merge commit 665 my$all_parents_ok= 1; 666 my$aggregate_mergeinfo= ''; 667 my$rooturl=$gs->repos_root; 668 669 if (defined($rewritten_parent)) { 670 # Replace first parent with newly-rewritten version 671 shift@parents; 672 unshift@parents,$rewritten_parent; 673 } 674 675 foreach my$parent(@parents) { 676 my ($branchurl,$svnrev,$paruuid) = 677 cmt_metadata($parent); 678 679 unless (defined($svnrev)) { 680 # Should have been caught be preflight check 681 fatal "merge commit $dhas ancestor $parent, but that change " 682 ."doesnot have git-svn metadata!"; 683 } 684 unless ($branchurl=~ /^$rooturl(.*)/) { 685 fatal "commit $parent git-svn metadata changed mid-run!"; 686 } 687 my$branchpath=$1; 688 689 my$ra= Git::SVN::Ra->new($branchurl); 690 my (undef, undef,$props) = 691$ra->get_dir(canonicalize_path("."),$svnrev); 692 my$par_mergeinfo=$props->{'svn:mergeinfo'}; 693 unless (defined$par_mergeinfo) { 694$par_mergeinfo= ''; 695 } 696 # Merge previous mergeinfo values 697$aggregate_mergeinfo= 698 merge_merge_info($aggregate_mergeinfo, 699$par_mergeinfo, 0); 700 701 next if$parenteq$parents[0]; # Skip first parent 702 # Add new changes being placed in tree by merge 703 my@cmd= (qw/rev-list --reverse/, 704$parent, qw/--not/); 705 foreach my$par(@parents) { 706 unless ($pareq$parent) { 707 push@cmd,$par; 708 } 709 } 710 my@revsin= (); 711 my ($revlist,$ctx) = command_output_pipe(@cmd); 712 while (<$revlist>) { 713 my$irev=$_; 714 chomp$irev; 715 my (undef,$csvnrev, undef) = 716 cmt_metadata($irev); 717 unless (defined$csvnrev) { 718 # A child is missing SVN annotations... 719 # this might be OK, or might not be. 720 warn "W:child $irevis merged into revision " 721 ."$d but doesnot have git-svn metadata." 722 ."This means git-svn cannot determine the " 723 ."svn revision numbers to place into the " 724 ."svn:mergeinfo property. You must ensure " 725 ."a branch is entirely committed to " 726 ."SVN before merging it in order for" 727 ."svn:mergeinfo population to function " 728 ."properly"; 729 } 730 push@revsin,$csvnrev; 731 } 732 command_close_pipe($revlist,$ctx); 733 734 last unless$all_parents_ok; 735 736 # We now have a list of all SVN revnos which are 737 # merged by this particular parent. Integrate them. 738 next if$#revsin== -1; 739 my$newmergeinfo= "$branchpath:" . join(',',@revsin); 740$aggregate_mergeinfo= 741 merge_merge_info($aggregate_mergeinfo, 742$newmergeinfo, 1); 743 } 744 if ($all_parents_okand$aggregate_mergeinfo) { 745 return$aggregate_mergeinfo; 746 } 747 } 748 749 return undef; 750} 751 752sub cmd_dcommit { 753 my$head= shift; 754 command_noisy(qw/update-index --refresh/); 755 git_cmd_try { command_oneline(qw/diff-index --quiet HEAD/) } 756 'Cannot dcommit with a dirty index. Commit your changes first, ' 757 . "or stash them with `git stash'.\n"; 758$head||= 'HEAD'; 759 760 my$old_head; 761 if ($headne 'HEAD') { 762$old_head= eval { 763 command_oneline([qw/symbolic-ref -q HEAD/]) 764 }; 765 if ($old_head) { 766$old_head=~ s{^refs/heads/}{}; 767 } else { 768$old_head= eval { command_oneline(qw/rev-parse HEAD/) }; 769 } 770 command(['checkout',$head], STDERR => 0); 771 } 772 773 my@refs; 774 my ($url,$rev,$uuid,$gs) = working_head_info('HEAD', \@refs); 775 unless ($gs) { 776 die "Unable to determine upstream SVN information from ", 777 "$headhistory.\nPerhaps the repository is empty."; 778 } 779 780 if (defined$_commit_url) { 781$url=$_commit_url; 782 } else { 783$url= eval { command_oneline('config', '--get', 784 "svn-remote.$gs->{repo_id}.commiturl") }; 785 if (!$url) { 786$url=$gs->full_pushurl 787 } 788 } 789 790 my$last_rev=$_revisionif defined$_revision; 791 if ($url) { 792 print "Committing to$url...\n"; 793 } 794 my ($linear_refs,$parents) = linearize_history($gs, \@refs); 795 if ($_no_rebase&& scalar(@$linear_refs) > 1) { 796 warn "Attempting to commit more than one change while ", 797 "--no-rebase is enabled.\n", 798 "If these changes depend on each other, re-running ", 799 "without --no-rebase may be required." 800 } 801 802 if (defined$_interactive){ 803 my$ask_default= "y"; 804 foreach my$d(@$linear_refs){ 805 my ($fh,$ctx) = command_output_pipe(qw(show --summary),"$d"); 806while(<$fh>){ 807print$_; 808} 809 command_close_pipe($fh,$ctx); 810$_= ask("Commit this patch to SVN? ([y]es (default)|[n]o|[q]uit|[a]ll): ", 811 valid_re =>qr/^(?:yes|y|no|n|quit|q|all|a)/i, 812default=>$ask_default); 813die"Commit this patch reply required"unlessdefined$_; 814if(/^[nq]/i) { 815exit(0); 816}elsif(/^a/i) { 817last; 818} 819} 820} 821 822my$expect_url=$url; 823 824my$push_merge_info=eval{ 825 command_oneline(qw/config --get svn.pushmergeinfo/) 826}; 827if(not defined($push_merge_info) 828or$push_merge_infoeq"false" 829or$push_merge_infoeq"no" 830or$push_merge_infoeq"never") { 831$push_merge_info=0; 832} 833 834unless(defined($_merge_info) || !$push_merge_info) { 835# Preflight check of changes to ensure no issues with mergeinfo 836# This includes check for uncommitted-to-SVN parents 837# (other than the first parent, which we will handle), 838# information from different SVN repos, and paths 839# which are not underneath this repository root. 840my$rooturl=$gs->repos_root; 841foreachmy$d(@$linear_refs) { 842my%parentshash; 843 read_commit_parents(\%parentshash,$d); 844my@realparents= @{$parentshash{$d}}; 845if($#realparents>0) { 846# Merge commit 847shift@realparents;# Remove/ignore first parent 848foreachmy$parent(@realparents) { 849my($branchurl,$svnrev,$paruuid) = cmt_metadata($parent); 850unless(defined$paruuid) { 851# A parent is missing SVN annotations... 852# abort the whole operation. 853 fatal "$parentis merged into revision$d, " 854."but does not have git-svn metadata. " 855."Either dcommit the branch or use a " 856."local cherry-pick, FF merge, or rebase " 857."instead of an explicit merge commit."; 858} 859 860unless($paruuideq$uuid) { 861# Parent has SVN metadata from different repository 862 fatal "merge parent$parentfor change$dhas " 863."git-svn uuid$paruuid, while current change " 864."has uuid$uuid!"; 865} 866 867unless($branchurl=~/^$rooturl(.*)/) { 868# This branch is very strange indeed. 869 fatal "merge parent$parentfor$dis on branch " 870."$branchurl, which is not under the " 871."git-svn root$rooturl!"; 872} 873} 874} 875} 876} 877 878my$rewritten_parent; 879 Git::SVN::remove_username($expect_url); 880if(defined($_merge_info)) { 881$_merge_info=~tr{ }{\n}; 882} 883while(1) { 884my$d=shift@$linear_refsorlast; 885unless(defined$last_rev) { 886(undef,$last_rev,undef) = cmt_metadata("$d~1"); 887unless(defined$last_rev) { 888 fatal "Unable to extract revision information ", 889"from commit$d~1"; 890} 891} 892if($_dry_run) { 893print"diff-tree$d~1$d\n"; 894}else{ 895my$cmt_rev; 896 897unless(defined($_merge_info) || !$push_merge_info) { 898$_merge_info= populate_merge_info($d,$gs, 899$uuid, 900$linear_refs, 901$rewritten_parent); 902} 903 904my%ed_opts= ( r =>$last_rev, 905log=> get_commit_entry($d)->{log}, 906 ra => Git::SVN::Ra->new($url), 907 config => SVN::Core::config_get_config( 908$Git::SVN::Ra::config_dir 909), 910 tree_a =>"$d~1", 911 tree_b =>$d, 912 editor_cb =>sub{ 913print"Committed r$_[0]\n"; 914$cmt_rev=$_[0]; 915}, 916 mergeinfo =>$_merge_info, 917 svn_path =>''); 918if(!SVN::Git::Editor->new(\%ed_opts)->apply_diff) { 919print"No changes\n$d~1==$d\n"; 920}elsif($parents->{$d} && @{$parents->{$d}}) { 921$gs->{inject_parents_dcommit}->{$cmt_rev} = 922$parents->{$d}; 923} 924$_fetch_all?$gs->fetch_all:$gs->fetch; 925$last_rev=$cmt_rev; 926next if$_no_rebase; 927 928# we always want to rebase against the current HEAD, 929# not any head that was passed to us 930my@diff= command('diff-tree',$d, 931$gs->refname,'--'); 932my@finish; 933if(@diff) { 934@finish= rebase_cmd(); 935print STDERR "W:$dand ",$gs->refname, 936" differ, using@finish:\n", 937join("\n",@diff),"\n"; 938}else{ 939print"No changes between current HEAD and ", 940$gs->refname, 941"\nResetting to the latest ", 942$gs->refname,"\n"; 943@finish= qw/reset --mixed/; 944} 945 command_noisy(@finish,$gs->refname); 946 947$rewritten_parent= command_oneline(qw/rev-parse HEAD/); 948 949if(@diff) { 950@refs= (); 951my($url_,$rev_,$uuid_,$gs_) = 952 working_head_info('HEAD', \@refs); 953my($linear_refs_,$parents_) = 954 linearize_history($gs_, \@refs); 955if(scalar(@$linear_refs) != 956scalar(@$linear_refs_)) { 957 fatal "# of revisions changed ", 958"\nbefore:\n", 959join("\n",@$linear_refs), 960"\n\nafter:\n", 961join("\n",@$linear_refs_),"\n", 962'If you are attempting to commit ', 963"merges, try running:\n\t", 964'git rebase --interactive', 965'--preserve-merges ', 966$gs->refname, 967"\nBefore dcommitting"; 968} 969if($url_ne$expect_url) { 970if($url_eq$gs->metadata_url) { 971print 972"Accepting rewritten URL:", 973"$url_\n"; 974}else{ 975 fatal 976"URL mismatch after rebase:", 977"$url_!=$expect_url"; 978} 979} 980if($uuid_ne$uuid) { 981 fatal "uuid mismatch after rebase: ", 982"$uuid_!=$uuid"; 983} 984# remap parents 985my(%p,@l,$i); 986for($i=0;$i<scalar@$linear_refs;$i++) { 987my$new=$linear_refs_->[$i]ornext; 988$p{$new} = 989$parents->{$linear_refs->[$i]}; 990push@l,$new; 991} 992$parents= \%p; 993$linear_refs= \@l; 994} 995} 996} 997 998if($old_head) { 999my$new_head= command_oneline(qw/rev-parse HEAD/);1000my$new_is_symbolic=eval{1001 command_oneline(qw/symbolic-ref -q HEAD/);1002};1003if($new_is_symbolic) {1004print"dcommitted the branch ",$head,"\n";1005}else{1006print"dcommitted on a detached HEAD because you gave ",1007"a revision argument.\n",1008"The rewritten commit is: ",$new_head,"\n";1009}1010 command(['checkout',$old_head], STDERR =>0);1011}10121013unlink$gs->{index};1014}10151016sub cmd_branch {1017my($branch_name,$head) =@_;10181019unless(defined$branch_name&&length$branch_name) {1020die(($_tag?"tag":"branch") ." name required\n");1021}1022$head||='HEAD';10231024my(undef,$rev,undef,$gs) = working_head_info($head);1025my$src=$gs->full_pushurl;10261027my$remote= Git::SVN::read_all_remotes()->{$gs->{repo_id}};1028my$allglobs=$remote->{$_tag?'tags':'branches'};1029my$glob;1030if($#{$allglobs} ==0) {1031$glob=$allglobs->[0];1032}else{1033unless(defined$_branch_dest) {1034die"Multiple ",1035$_tag?"tag":"branch",1036" paths defined for Subversion repository.\n",1037"You must specify where you want to create the ",1038$_tag?"tag":"branch",1039" with the --destination argument.\n";1040}1041foreachmy$g(@{$allglobs}) {1042# SVN::Git::Editor could probably be moved to Git.pm..1043my$re= SVN::Git::Editor::glob2pat($g->{path}->{left});1044if($_branch_dest=~/$re/) {1045$glob=$g;1046last;1047}1048}1049unless(defined$glob) {1050my$dest_re=qr/\b\Q$_branch_dest\E\b/;1051foreachmy$g(@{$allglobs}) {1052$g->{path}->{left} =~/$dest_re/ornext;1053if(defined$glob) {1054die"Ambiguous destination: ",1055$_branch_dest,"\nmatches both '",1056$glob->{path}->{left},"' and '",1057$g->{path}->{left},"'\n";1058}1059$glob=$g;1060}1061unless(defined$glob) {1062die"Unknown ",1063$_tag?"tag":"branch",1064" destination$_branch_dest\n";1065}1066}1067}1068my($lft,$rgt) = @{$glob->{path} }{qw/left right/};1069my$url;1070if(defined$_commit_url) {1071$url=$_commit_url;1072}else{1073$url=eval{ command_oneline('config','--get',1074"svn-remote.$gs->{repo_id}.commiturl") };1075if(!$url) {1076$url=$remote->{pushurl} ||$remote->{url};1077}1078}1079my$dst=join'/',$url,$lft,$branch_name, ($rgt|| ());10801081if($dst=~/^https:/&&$src=~/^http:/) {1082$src=~s/^http:/https:/;1083}10841085::_req_svn();10861087my$ctx= SVN::Client->new(1088 auth => Git::SVN::Ra::_auth_providers(),1089 log_msg =>sub{1090${$_[0] } =defined$_message1091?$_message1092:'Create '. ($_tag?'tag ':'branch ')1093.$branch_name;1094},1095);10961097eval{1098$ctx->ls($dst,'HEAD',0);1099}and die"branch ${branch_name} already exists\n";11001101print"Copying ${src} at r${rev} to ${dst}...\n";1102$ctx->copy($src,$rev,$dst)1103unless$_dry_run;11041105$gs->fetch_all;1106}11071108sub cmd_find_rev {1109my$revision_or_hash=shift or die"SVN or git revision required ",1110"as a command-line argument\n";1111my$result;1112if($revision_or_hash=~/^r\d+$/) {1113my$head=shift;1114$head||='HEAD';1115my@refs;1116my(undef,undef,$uuid,$gs) = working_head_info($head, \@refs);1117unless($gs) {1118die"Unable to determine upstream SVN information from ",1119"$headhistory\n";1120}1121my$desired_revision=substr($revision_or_hash,1);1122$result=$gs->rev_map_get($desired_revision,$uuid);1123}else{1124my(undef,$rev,undef) = cmt_metadata($revision_or_hash);1125$result=$rev;1126}1127print"$result\n"if$result;1128}11291130sub auto_create_empty_directories {1131my($gs) =@_;1132my$var=eval{ command_oneline('config','--get','--bool',1133"svn-remote.$gs->{repo_id}.automkdirs") };1134# By default, create empty directories by consulting the unhandled log,1135# but allow setting it to 'false' to skip it.1136return!($var&&$vareq'false');1137}11381139sub cmd_rebase {1140 command_noisy(qw/update-index --refresh/);1141my($url,$rev,$uuid,$gs) = working_head_info('HEAD');1142unless($gs) {1143die"Unable to determine upstream SVN information from ",1144"working tree history\n";1145}1146if($_dry_run) {1147print"Remote Branch: ".$gs->refname."\n";1148print"SVN URL: ".$url."\n";1149return;1150}1151if(command(qw/diff-index HEAD --/)) {1152print STDERR "Cannot rebase with uncommited changes:\n";1153 command_noisy('status');1154exit1;1155}1156unless($_local) {1157# rebase will checkout for us, so no need to do it explicitly1158$_no_checkout='true';1159$_fetch_all?$gs->fetch_all:$gs->fetch;1160}1161 command_noisy(rebase_cmd(),$gs->refname);1162if(auto_create_empty_directories($gs)) {1163$gs->mkemptydirs;1164}1165}11661167sub cmd_show_ignore {1168my($url,$rev,$uuid,$gs) = working_head_info('HEAD');1169$gs||= Git::SVN->new;1170my$r= (defined$_revision?$_revision:$gs->ra->get_latest_revnum);1171$gs->prop_walk($gs->{path},$r,sub{1172my($gs,$path,$props) =@_;1173print STDOUT "\n#$path\n";1174my$s=$props->{'svn:ignore'}orreturn;1175$s=~s/[\r\n]+/\n/g;1176$s=~s/^\n+//;1177chomp$s;1178$s=~ s#^#$path#gm;1179print STDOUT "$s\n";1180});1181}11821183sub cmd_show_externals {1184my($url,$rev,$uuid,$gs) = working_head_info('HEAD');1185$gs||= Git::SVN->new;1186my$r= (defined$_revision?$_revision:$gs->ra->get_latest_revnum);1187$gs->prop_walk($gs->{path},$r,sub{1188my($gs,$path,$props) =@_;1189print STDOUT "\n#$path\n";1190my$s=$props->{'svn:externals'}orreturn;1191$s=~s/[\r\n]+/\n/g;1192chomp$s;1193$s=~ s#^#$path#gm;1194print STDOUT "$s\n";1195});1196}11971198sub cmd_create_ignore {1199my($url,$rev,$uuid,$gs) = working_head_info('HEAD');1200$gs||= Git::SVN->new;1201my$r= (defined$_revision?$_revision:$gs->ra->get_latest_revnum);1202$gs->prop_walk($gs->{path},$r,sub{1203my($gs,$path,$props) =@_;1204# $path is of the form /path/to/dir/1205$path='.'.$path;1206# SVN can have attributes on empty directories,1207# which git won't track1208 mkpath([$path])unless-d $path;1209my$ignore=$path.'.gitignore';1210my$s=$props->{'svn:ignore'}orreturn;1211open(GITIGNORE,'>',$ignore)1212or fatal("Failed to open `$ignore' for writing:$!");1213$s=~s/[\r\n]+/\n/g;1214$s=~s/^\n+//;1215chomp$s;1216# Prefix all patterns so that the ignore doesn't apply1217# to sub-directories.1218$s=~ s#^#/#gm;1219print GITIGNORE "$s\n";1220close(GITIGNORE)1221or fatal("Failed to close `$ignore':$!");1222 command_noisy('add','-f',$ignore);1223});1224}12251226sub cmd_mkdirs {1227my($url,$rev,$uuid,$gs) = working_head_info('HEAD');1228$gs||= Git::SVN->new;1229$gs->mkemptydirs($_revision);1230}12311232sub canonicalize_path {1233my($path) =@_;1234my$dot_slash_added=0;1235if(substr($path,0,1)ne"/") {1236$path="./".$path;1237$dot_slash_added=1;1238}1239# File::Spec->canonpath doesn't collapse x/../y into y (for a1240# good reason), so let's do this manually.1241$path=~ s#/+#/#g;1242$path=~ s#/\.(?:/|$)#/#g;1243$path=~ s#/[^/]+/\.\.##g;1244$path=~ s#/$##g;1245$path=~ s#^\./## if $dot_slash_added;1246$path=~ s#^/##;1247$path=~ s#^\.$##;1248return$path;1249}12501251sub canonicalize_url {1252my($url) =@_;1253$url=~ s#^([^:]+://[^/]*/)(.*)$#$1 . canonicalize_path($2)#e;1254return$url;1255}12561257# get_svnprops(PATH)1258# ------------------1259# Helper for cmd_propget and cmd_proplist below.1260sub get_svnprops {1261my$path=shift;1262my($url,$rev,$uuid,$gs) = working_head_info('HEAD');1263$gs||= Git::SVN->new;12641265# prefix THE PATH by the sub-directory from which the user1266# invoked us.1267$path=$cmd_dir_prefix.$path;1268 fatal("No such file or directory:$path")unless-e $path;1269my$is_dir= -d $path?1:0;1270$path=$gs->{path} .'/'.$path;12711272# canonicalize the path (otherwise libsvn will abort or fail to1273# find the file)1274$path= canonicalize_path($path);12751276my$r= (defined$_revision?$_revision:$gs->ra->get_latest_revnum);1277my$props;1278if($is_dir) {1279(undef,undef,$props) =$gs->ra->get_dir($path,$r);1280}1281else{1282(undef,$props) =$gs->ra->get_file($path,$r,undef);1283}1284return$props;1285}12861287# cmd_propget (PROP, PATH)1288# ------------------------1289# Print the SVN property PROP for PATH.1290sub cmd_propget {1291my($prop,$path) =@_;1292$path='.'ifnot defined$path;1293 usage(1)ifnot defined$prop;1294my$props= get_svnprops($path);1295if(not defined$props->{$prop}) {1296 fatal("`$path' does not have a `$prop' SVN property.");1297}1298print$props->{$prop} ."\n";1299}13001301# cmd_proplist (PATH)1302# -------------------1303# Print the list of SVN properties for PATH.1304sub cmd_proplist {1305my$path=shift;1306$path='.'ifnot defined$path;1307my$props= get_svnprops($path);1308print"Properties on '$path':\n";1309foreach(sort keys%{$props}) {1310print"$_\n";1311}1312}13131314sub cmd_multi_init {1315my$url=shift;1316unless(defined$_trunk||@_branches||@_tags) {1317 usage(1);1318}13191320$_prefix=''unlessdefined$_prefix;1321if(defined$url) {1322$url= canonicalize_url($url);1323 init_subdir(@_);1324}1325 do_git_init_db();1326if(defined$_trunk) {1327$_trunk=~ s#^/+##;1328my$trunk_ref='refs/remotes/'.$_prefix.'trunk';1329# try both old-style and new-style lookups:1330my$gs_trunk=eval{ Git::SVN->new($trunk_ref) };1331unless($gs_trunk) {1332my($trunk_url,$trunk_path) =1333 complete_svn_url($url,$_trunk);1334$gs_trunk= Git::SVN->init($trunk_url,$trunk_path,1335undef,$trunk_ref);1336}1337}1338return unless@_branches||@_tags;1339my$ra=$url? Git::SVN::Ra->new($url) :undef;1340foreachmy$path(@_branches) {1341 complete_url_ls_init($ra,$path,'--branches/-b',$_prefix);1342}1343foreachmy$path(@_tags) {1344 complete_url_ls_init($ra,$path,'--tags/-t',$_prefix.'tags/');1345}1346}13471348sub cmd_multi_fetch {1349$Git::SVN::no_reuse_existing =undef;1350my$remotes= Git::SVN::read_all_remotes();1351foreachmy$repo_id(sort keys%$remotes) {1352if($remotes->{$repo_id}->{url}) {1353 Git::SVN::fetch_all($repo_id,$remotes);1354}1355}1356}13571358# this command is special because it requires no metadata1359sub cmd_commit_diff {1360my($ta,$tb,$url) =@_;1361my$usage="Usage:$0commit-diff -r<revision> ".1362"<tree-ish> <tree-ish> [<URL>]";1363 fatal($usage)if(!defined$ta|| !defined$tb);1364my$svn_path='';1365if(!defined$url) {1366my$gs=eval{ Git::SVN->new};1367if(!$gs) {1368 fatal("Needed URL or usable git-svn --id in ",1369"the command-line\n",$usage);1370}1371$url=$gs->{url};1372$svn_path=$gs->{path};1373}1374unless(defined$_revision) {1375 fatal("-r|--revision is a required argument\n",$usage);1376}1377if(defined$_message&&defined$_file) {1378 fatal("Both --message/-m and --file/-F specified ",1379"for the commit message.\n",1380"I have no idea what you mean");1381}1382if(defined$_file) {1383$_message= file_to_s($_file);1384}else{1385$_message||= get_commit_entry($tb)->{log};1386}1387my$ra||= Git::SVN::Ra->new($url);1388my$r=$_revision;1389if($req'HEAD') {1390$r=$ra->get_latest_revnum;1391}elsif($r!~/^\d+$/) {1392die"revision argument:$rnot understood by git-svn\n";1393}1394my%ed_opts= ( r =>$r,1395log=>$_message,1396 ra =>$ra,1397 tree_a =>$ta,1398 tree_b =>$tb,1399 editor_cb =>sub{print"Committed r$_[0]\n"},1400 svn_path =>$svn_path);1401if(!SVN::Git::Editor->new(\%ed_opts)->apply_diff) {1402print"No changes\n$ta==$tb\n";1403}1404}14051406sub escape_uri_only {1407my($uri) =@_;1408my@tmp;1409foreach(splitm{/},$uri) {1410s/([^~\w.%+-]|%(?![a-fA-F0-9]{2}))/sprintf("%%%02X",ord($1))/eg;1411push@tmp,$_;1412}1413join('/',@tmp);1414}14151416sub escape_url {1417my($url) =@_;1418if($url=~ m#^([^:]+)://([^/]*)(.*)$#) {1419my($scheme,$domain,$uri) = ($1,$2, escape_uri_only($3));1420$url="$scheme://$domain$uri";1421}1422$url;1423}14241425sub cmd_info {1426my$path= canonicalize_path(defined($_[0]) ?$_[0] :".");1427my$fullpath= canonicalize_path($cmd_dir_prefix.$path);1428if(exists$_[1]) {1429die"Too many arguments specified\n";1430}14311432my($file_type,$diff_status) = find_file_type_and_diff_status($path);14331434if(!$file_type&& !$diff_status) {1435print STDERR "svn: '$path' is not under version control\n";1436exit1;1437}14381439my($url,$rev,$uuid,$gs) = working_head_info('HEAD');1440unless($gs) {1441die"Unable to determine upstream SVN information from ",1442"working tree history\n";1443}14441445# canonicalize_path() will return "" to make libsvn 1.5.x happy,1446$path="."if$patheq"";14471448my$full_url=$url. ($fullpatheq""?"":"/$fullpath");14491450if($_url) {1451print escape_url($full_url),"\n";1452return;1453}14541455my$result="Path:$path\n";1456$result.="Name: ". basename($path) ."\n"if$file_typene"dir";1457$result.="URL: ". escape_url($full_url) ."\n";14581459eval{1460my$repos_root=$gs->repos_root;1461 Git::SVN::remove_username($repos_root);1462$result.="Repository Root: ". escape_url($repos_root) ."\n";1463};1464if($@) {1465$result.="Repository Root: (offline)\n";1466}1467::_req_svn();1468$result.="Repository UUID:$uuid\n"unless$diff_statuseq"A"&&1469($SVN::Core::VERSION le'1.5.4'||$file_typene"dir");1470$result.="Revision: ". ($diff_statuseq"A"?0:$rev) ."\n";14711472$result.="Node Kind: ".1473($file_typeeq"dir"?"directory":"file") ."\n";14741475my$schedule=$diff_statuseq"A"1476?"add"1477: ($diff_statuseq"D"?"delete":"normal");1478$result.="Schedule:$schedule\n";14791480if($diff_statuseq"A") {1481print$result,"\n";1482return;1483}14841485my($lc_author,$lc_rev,$lc_date_utc);1486my@args= Git::SVN::Log::git_svn_log_cmd($rev,$rev,"--",$fullpath);1487my$log= command_output_pipe(@args);1488my$esc_color=qr/(?:\033\[(?:(?:\d+;)*\d*)?m)*/;1489while(<$log>) {1490if(/^${esc_color}author (.+) <[^>]+> (\d+) ([\-\+]?\d+)$/o) {1491$lc_author=$1;1492$lc_date_utc= Git::SVN::Log::parse_git_date($2,$3);1493}elsif(/^${esc_color} (git-svn-id:.+)$/o) {1494(undef,$lc_rev,undef) = ::extract_metadata($1);1495}1496}1497close$log;14981499 Git::SVN::Log::set_local_timezone();15001501$result.="Last Changed Author:$lc_author\n";1502$result.="Last Changed Rev:$lc_rev\n";1503$result.="Last Changed Date: ".1504 Git::SVN::Log::format_svn_date($lc_date_utc) ."\n";15051506if($file_typene"dir") {1507my$text_last_updated_date=1508($diff_statuseq"D"?$lc_date_utc: (stat$path)[9]);1509$result.=1510"Text Last Updated: ".1511 Git::SVN::Log::format_svn_date($text_last_updated_date) .1512"\n";1513my$checksum;1514if($diff_statuseq"D") {1515my($fh,$ctx) =1516 command_output_pipe(qw(cat-file blob),"HEAD:$path");1517if($file_typeeq"link") {1518my$file_name= <$fh>;1519$checksum= md5sum("link$file_name");1520}else{1521$checksum= md5sum($fh);1522}1523 command_close_pipe($fh,$ctx);1524}elsif($file_typeeq"link") {1525my$file_name=1526 command(qw(cat-file blob),"HEAD:$path");1527$checksum=1528 md5sum("link ".$file_name);1529}else{1530open FILE,"<",$pathor die$!;1531$checksum= md5sum(\*FILE);1532close FILE or die$!;1533}1534$result.="Checksum: ".$checksum."\n";1535}15361537print$result,"\n";1538}15391540sub cmd_reset {1541my$target=shift||$_revisionor die"SVN revision required\n";1542$target=$1if$target=~/^r(\d+)$/;1543$target=~/^\d+$/or die"Numeric SVN revision expected\n";1544my($url,$rev,$uuid,$gs) = working_head_info('HEAD');1545unless($gs) {1546die"Unable to determine upstream SVN information from ".1547"history\n";1548}1549my($r,$c) =$gs->find_rev_before($target,not$_fetch_parent);1550die"Cannot find SVN revision$target\n"unlessdefined($c);1551$gs->rev_map_set($r,$c,'reset',$uuid);1552print"r$r=$c($gs->{ref_id})\n";1553}15541555sub cmd_gc {1556if(!$can_compress) {1557warn"Compress::Zlib could not be found; unhandled.log ".1558"files will not be compressed.\n";1559}1560 find({ wanted => \&gc_directory, no_chdir =>1},"$ENV{GIT_DIR}/svn");1561}15621563########################### utility functions #########################15641565sub rebase_cmd {1566my@cmd= qw/rebase/;1567push@cmd,'-v'if$_verbose;1568push@cmd, qw/--merge/if$_merge;1569push@cmd,"--strategy=$_strategy"if$_strategy;1570@cmd;1571}15721573sub post_fetch_checkout {1574return if$_no_checkout;1575my$gs=$Git::SVN::_head orreturn;1576return if verify_ref('refs/heads/master^0');15771578# look for "trunk" ref if it exists1579my$remote= Git::SVN::read_all_remotes()->{$gs->{repo_id}};1580my$fetch=$remote->{fetch};1581if($fetch) {1582foreachmy$p(keys%$fetch) {1583 basename($fetch->{$p})eq'trunk'ornext;1584$gs= Git::SVN->new($fetch->{$p},$gs->{repo_id},$p);1585last;1586}1587}15881589my$valid_head= verify_ref('HEAD^0');1590 command_noisy(qw(update-ref refs/heads/master),$gs->refname);1591return if($valid_head|| !verify_ref('HEAD^0'));15921593return if$ENV{GIT_DIR} !~ m#^(?:.*/)?\.git$#;1594my$index=$ENV{GIT_INDEX_FILE} ||"$ENV{GIT_DIR}/index";1595return if-f $index;15961597return if command_oneline(qw/rev-parse --is-inside-work-tree/)eq'false';1598return if command_oneline(qw/rev-parse --is-inside-git-dir/)eq'true';1599 command_noisy(qw/read-tree -m -u -v HEAD HEAD/);1600print STDERR "Checked out HEAD:\n",1601$gs->full_url," r",$gs->last_rev,"\n";1602if(auto_create_empty_directories($gs)) {1603$gs->mkemptydirs($gs->last_rev);1604}1605}16061607sub complete_svn_url {1608my($url,$path) =@_;1609$path=~ s#/+$##;1610if($path!~ m#^[a-z\+]+://#) {1611if(!defined$url||$url!~ m#^[a-z\+]+://#) {1612 fatal("E: '$path' is not a complete URL ",1613"and a separate URL is not specified");1614}1615return($url,$path);1616}1617return($path,'');1618}16191620sub complete_url_ls_init {1621my($ra,$repo_path,$switch,$pfx) =@_;1622unless($repo_path) {1623print STDERR "W:$switchnot specified\n";1624return;1625}1626$repo_path=~ s#/+$##;1627if($repo_path=~ m#^[a-z\+]+://#) {1628$ra= Git::SVN::Ra->new($repo_path);1629$repo_path='';1630}else{1631$repo_path=~ s#^/+##;1632unless($ra) {1633 fatal("E: '$repo_path' is not a complete URL ",1634"and a separate URL is not specified");1635}1636}1637my$url=$ra->{url};1638my$gs= Git::SVN->init($url,undef,undef,undef,1);1639my$k="svn-remote.$gs->{repo_id}.url";1640my$orig_url=eval{ command_oneline(qw/config --get/,$k) };1641if($orig_url&& ($orig_urlne$gs->{url})) {1642die"$kalready set:$orig_url\n",1643"wanted to set to:$gs->{url}\n";1644}1645 command_oneline('config',$k,$gs->{url})unless$orig_url;1646my$remote_path="$gs->{path}/$repo_path";1647$remote_path=~s{%([0-9A-F]{2})}{chr hex($1)}ieg;1648$remote_path=~ s#/+#/#g;1649$remote_path=~ s#^/##g;1650$remote_path.="/*"if$remote_path!~ /\*/;1651my($n) = ($switch=~/^--(\w+)/);1652if(length$pfx&&$pfx!~ m#/$#) {1653die"--prefix='$pfx' must have a trailing slash '/'\n";1654}1655 command_noisy('config',1656'--add',1657"svn-remote.$gs->{repo_id}.$n",1658"$remote_path:refs/remotes/$pfx*".1659('/*' x (($remote_path=~ tr/*/*/) -1)) );1660}16611662sub verify_ref {1663my($ref) =@_;1664eval{ command_oneline(['rev-parse','--verify',$ref],1665{ STDERR =>0}); };1666}16671668sub get_tree_from_treeish {1669my($treeish) =@_;1670# $treeish can be a symbolic ref, too:1671my$type= command_oneline(qw/cat-file -t/,$treeish);1672my$expected;1673while($typeeq'tag') {1674($treeish,$type) = command(qw/cat-file tag/,$treeish);1675}1676if($typeeq'commit') {1677$expected= (grep/^tree /, command(qw/cat-file commit/,1678$treeish))[0];1679($expected) = ($expected=~/^tree ($sha1)$/o);1680die"Unable to get tree from$treeish\n"unless$expected;1681}elsif($typeeq'tree') {1682$expected=$treeish;1683}else{1684die"$treeishis a$type, expected tree, tag or commit\n";1685}1686return$expected;1687}16881689sub get_commit_entry {1690my($treeish) =shift;1691my%log_entry= (log=>'', tree => get_tree_from_treeish($treeish) );1692my$commit_editmsg="$ENV{GIT_DIR}/COMMIT_EDITMSG";1693my$commit_msg="$ENV{GIT_DIR}/COMMIT_MSG";1694open my$log_fh,'>',$commit_editmsgor croak $!;16951696my$type= command_oneline(qw/cat-file -t/,$treeish);1697if($typeeq'commit'||$typeeq'tag') {1698my($msg_fh,$ctx) = command_output_pipe('cat-file',1699$type,$treeish);1700my$in_msg=0;1701my$author;1702my$saw_from=0;1703my$msgbuf="";1704while(<$msg_fh>) {1705if(!$in_msg) {1706$in_msg=1if(/^\s*$/);1707$author=$1if(/^author (.*>)/);1708}elsif(/^git-svn-id: /) {1709# skip this for now, we regenerate the1710# correct one on re-fetch anyways1711# TODO: set *:merge properties or like...1712}else{1713if(/^From:/||/^Signed-off-by:/) {1714$saw_from=1;1715}1716$msgbuf.=$_;1717}1718}1719$msgbuf=~s/\s+$//s;1720if($Git::SVN::_add_author_from &&defined($author)1721&& !$saw_from) {1722$msgbuf.="\n\nFrom:$author";1723}1724print$log_fh $msgbufor croak $!;1725 command_close_pipe($msg_fh,$ctx);1726}1727close$log_fhor croak $!;17281729if($_edit|| ($typeeq'tree')) {1730chomp(my$editor= command_oneline(qw(var GIT_EDITOR)));1731system('sh','-c',$editor.' "$@"',$editor,$commit_editmsg);1732}1733rename$commit_editmsg,$commit_msgor croak $!;1734{1735require Encode;1736# SVN requires messages to be UTF-8 when entering the repo1737local$/;1738open$log_fh,'<',$commit_msgor croak $!;1739binmode$log_fh;1740chomp($log_entry{log} = <$log_fh>);17411742my$enc= Git::config('i18n.commitencoding') ||'UTF-8';1743my$msg=$log_entry{log};17441745eval{$msg= Encode::decode($enc,$msg,1) };1746if($@) {1747die"Could not decode as$enc:\n",$msg,1748"\nPerhaps you need to set i18n.commitencoding\n";1749}17501751eval{$msg= Encode::encode('UTF-8',$msg,1) };1752die"Could not encode as UTF-8:\n$msg\n"if$@;17531754$log_entry{log} =$msg;17551756close$log_fhor croak $!;1757}1758unlink$commit_msg;1759 \%log_entry;1760}17611762sub s_to_file {1763my($str,$file,$mode) =@_;1764open my$fd,'>',$fileor croak $!;1765print$fd $str,"\n"or croak $!;1766close$fdor croak $!;1767chmod($mode&~umask,$file)if(defined$mode);1768}17691770sub file_to_s {1771my$file=shift;1772open my$fd,'<',$fileor croak "$!: file:$file\n";1773local$/;1774my$ret= <$fd>;1775close$fdor croak $!;1776$ret=~s/\s*$//s;1777return$ret;1778}17791780# '<svn username> = real-name <email address>' mapping based on git-svnimport:1781sub load_authors {1782open my$authors,'<',$_authorsor die"Can't open$_authors$!\n";1783my$log=$cmdeq'log';1784while(<$authors>) {1785chomp;1786next unless/^(.+?|\(no author\))\s*=\s*(.+?)\s*<(.+)>\s*$/;1787my($user,$name,$email) = ($1,$2,$3);1788if($log) {1789$Git::SVN::Log::rusers{"$name<$email>"} =$user;1790}else{1791$users{$user} = [$name,$email];1792}1793}1794close$authorsor croak $!;1795}17961797# convert GetOpt::Long specs for use by git-config1798sub read_git_config {1799my$opts=shift;1800my@config_only;1801foreachmy$o(keys%$opts) {1802# if we have mixedCase and a long option-only, then1803# it's a config-only variable that we don't need for1804# the command-line.1805push@config_only,$oif($o=~/[A-Z]/&&$o=~/^[a-z]+$/i);1806my$v=$opts->{$o};1807my($key) = ($o=~/^([a-zA-Z\-]+)/);1808$key=~s/-//g;1809my$arg='git config';1810$arg.=' --int'if($o=~/[:=]i$/);1811$arg.=' --bool'if($o!~/[:=][sfi]$/);1812if(ref$veq'ARRAY') {1813chomp(my@tmp=`$arg--get-all svn.$key`);1814@$v=@tmpif@tmp;1815 } else {1816 chomp(my$tmp= `$arg--get svn.$key`);1817if($tmp&& !($arg=~/ --bool/&&$tmpeq'false')) {1818$$v=$tmp;1819}1820}1821}1822delete@$opts{@config_only}if@config_only;1823}18241825sub extract_metadata {1826my$id=shift orreturn(undef,undef,undef);1827my($url,$rev,$uuid) = ($id=~ /^\s*git-svn-id:\s+(.*)\@(\d+)1828 \s([a-f\d\-]+)$/ix);1829if(!defined$rev|| !$uuid|| !$url) {1830# some of the original repositories I made had1831# identifiers like this:1832($rev,$uuid) = ($id=~/^\s*git-svn-id:\s(\d+)\@([a-f\d\-]+)/i);1833}1834return($url,$rev,$uuid);1835}18361837sub cmt_metadata {1838return extract_metadata((grep(/^git-svn-id: /,1839 command(qw/cat-file commit/,shift)))[-1]);1840}18411842sub cmt_sha2rev_batch {1843my%s2r;1844my($pid,$in,$out,$ctx) = command_bidi_pipe(qw/cat-file --batch/);1845my$list=shift;18461847foreachmy$sha(@{$list}) {1848my$first=1;1849my$size=0;1850print$out $sha,"\n";18511852while(my$line= <$in>) {1853if($first&&$line=~/^[[:xdigit:]]{40}\smissing$/) {1854last;1855}elsif($first&&1856$line=~/^[[:xdigit:]]{40}\scommit\s(\d+)$/) {1857$first=0;1858$size=$1;1859next;1860}elsif($line=~/^(git-svn-id: )/) {1861my(undef,$rev,undef) =1862 extract_metadata($line);1863$s2r{$sha} =$rev;1864}18651866$size-=length($line);1867last if($size==0);1868}1869}18701871 command_close_bidi_pipe($pid,$in,$out,$ctx);18721873return \%s2r;1874}18751876sub working_head_info {1877my($head,$refs) =@_;1878my@args= qw/log--no-color --no-decorate --first-parent1879--pretty=medium/;1880my($fh,$ctx) = command_output_pipe(@args,$head);1881my$hash;1882my%max;1883while(<$fh>) {1884if(m{^commit ($::sha1)$}) {1885unshift@$refs,$hashif$hashand$refs;1886$hash=$1;1887next;1888}1889next unlesss{^\s*(git-svn-id:)}{$1};1890my($url,$rev,$uuid) = extract_metadata($_);1891if(defined$url&&defined$rev) {1892next if$max{$url}and$max{$url} <$rev;1893if(my$gs= Git::SVN->find_by_url($url)) {1894my$c=$gs->rev_map_get($rev,$uuid);1895if($c&&$ceq$hash) {1896close$fh;# break the pipe1897return($url,$rev,$uuid,$gs);1898}else{1899$max{$url} ||=$gs->rev_map_max;1900}1901}1902}1903}1904 command_close_pipe($fh,$ctx);1905(undef,undef,undef,undef);1906}19071908sub read_commit_parents {1909my($parents,$c) =@_;1910chomp(my$p= command_oneline(qw/rev-list --parents -1/,$c));1911$p=~s/^($c)\s*//or die"rev-list --parents -1$cfailed!\n";1912@{$parents->{$c}} =split(/ /,$p);1913}19141915sub linearize_history {1916my($gs,$refs) =@_;1917my%parents;1918foreachmy$c(@$refs) {1919 read_commit_parents(\%parents,$c);1920}19211922my@linear_refs;1923my%skip= ();1924my$last_svn_commit=$gs->last_commit;1925foreachmy$c(reverse@$refs) {1926next if$ceq$last_svn_commit;1927last if$skip{$c};19281929unshift@linear_refs,$c;1930$skip{$c} =1;19311932# we only want the first parent to diff against for linear1933# history, we save the rest to inject when we finalize the1934# svn commit1935my$fp_a= verify_ref("$c~1");1936my$fp_b=shift@{$parents{$c}}if$parents{$c};1937if(!$fp_a|| !$fp_b) {1938die"Commit$c\n",1939"has no parent commit, and therefore ",1940"nothing to diff against.\n",1941"You should be working from a repository ",1942"originally created by git-svn\n";1943}1944if($fp_ane$fp_b) {1945die"$c~1=$fp_a, however parsing commit$c",1946"revealed that:\n$c~1=$fp_b\nBUG!\n";1947}19481949foreachmy$p(@{$parents{$c}}) {1950$skip{$p} =1;1951}1952}1953(\@linear_refs, \%parents);1954}19551956sub find_file_type_and_diff_status {1957my($path) =@_;1958return('dir','')if$patheq'';19591960my$diff_output=1961 command_oneline(qw(diff --cached --name-status --),$path) ||"";1962my$diff_status= (split(' ',$diff_output))[0] ||"";19631964my$ls_tree= command_oneline(qw(ls-tree HEAD),$path) ||"";19651966return(undef,undef)if!$diff_status&& !$ls_tree;19671968if($diff_statuseq"A") {1969return("link",$diff_status)if-l $path;1970return("dir",$diff_status)if-d $path;1971return("file",$diff_status);1972}19731974my$mode= (split(' ',$ls_tree))[0] ||"";19751976return("link",$diff_status)if$modeeq"120000";1977return("dir",$diff_status)if$modeeq"040000";1978return("file",$diff_status);1979}19801981sub md5sum {1982my$arg=shift;1983my$ref=ref$arg;1984my$md5= Digest::MD5->new();1985if($refeq'GLOB'||$refeq'IO::File'||$refeq'File::Temp') {1986$md5->addfile($arg)or croak $!;1987}elsif($refeq'SCALAR') {1988$md5->add($$arg)or croak $!;1989}elsif(!$ref) {1990$md5->add($arg)or croak $!;1991}else{1992::fatal "Can't provide MD5 hash for unknown ref type: '",$ref,"'";1993}1994return$md5->hexdigest();1995}19961997sub gc_directory {1998if($can_compress&& -f $_&& basename($_)eq"unhandled.log") {1999my$out_filename=$_.".gz";2000open my$in_fh,"<",$_or die"Unable to open$_:$!\n";2001binmode$in_fh;2002my$gz= Compress::Zlib::gzopen($out_filename,"ab")or2003die"Unable to open$out_filename:$!\n";20042005my$res;2006while($res=sysread($in_fh,my$str,1024)) {2007$gz->gzwrite($str)or2008die"Unable to write: ".$gz->gzerror()."!\n";2009}2010unlink$_or die"unlink$File::Find::name:$!\n";2011}elsif(-f $_&& basename($_)eq"index") {2012unlink$_or die"unlink$_:$!\n";2013}2014}20152016package Git::SVN;2017use strict;2018use warnings;2019use Fcntl qw/:DEFAULT :seek/;2020useconstant rev_map_fmt =>'NH40';2021use vars qw/$default_repo_id $default_ref_id $_no_metadata $_follow_parent2022$_repack $_repack_flags $_use_svm_props $_head2023$_use_svnsync_props $no_reuse_existing $_minimize_url2024$_use_log_author $_add_author_from $_localtime/;2025use Carp qw/croak/;2026use File::Path qw/mkpath/;2027use File::Copy qw/copy/;2028use IPC::Open3;2029use Memoize;# core since 5.8.0, Jul 20022030use Memoize::Storable;20312032my($_gc_nr,$_gc_period);20332034# properties that we do not log:2035my%SKIP_PROP;2036BEGIN{2037%SKIP_PROP=map{$_=>1} qw/svn:wc:ra_dav:version-url2038 svn:special svn:executable2039 svn:entry:committed-rev2040 svn:entry:last-author2041 svn:entry:uuid2042 svn:entry:committed-date/;20432044# some options are read globally, but can be overridden locally2045# per [svn-remote "..."] section. Command-line options will *NOT*2046# override options set in an [svn-remote "..."] section2047no strict 'refs';2048formy$option(qw/follow_parent no_metadata use_svm_props2049 use_svnsync_props/) {2050my$key=$option;2051$key=~tr/_//d;2052my$prop="-$option";2053*$option=sub{2054my($self) =@_;2055return$self->{$prop}ifexists$self->{$prop};2056my$k="svn-remote.$self->{repo_id}.$key";2057eval{ command_oneline(qw/config --get/,$k) };2058if($@) {2059$self->{$prop} = ${"Git::SVN::_$option"};2060}else{2061my$v= command_oneline(qw/config --bool/,$k);2062$self->{$prop} =$veq'false'?0:1;2063}2064return$self->{$prop};2065}2066}2067}206820692070my(%LOCKFILES,%INDEX_FILES);2071END{2072unlink keys%LOCKFILESif%LOCKFILES;2073unlink keys%INDEX_FILESif%INDEX_FILES;2074}20752076sub resolve_local_globs {2077my($url,$fetch,$glob_spec) =@_;2078return unlessdefined$glob_spec;2079my$ref=$glob_spec->{ref};2080my$path=$glob_spec->{path};2081foreach(command(qw#for-each-ref --format=%(refname) refs/#)) {2082next unless m#^$ref->{regex}$#;2083my$p=$1;2084my$pathname= desanitize_refname($path->full_path($p));2085my$refname= desanitize_refname($ref->full_path($p));2086if(my$existing=$fetch->{$pathname}) {2087if($existingne$refname) {2088die"Refspec conflict:\n",2089"existing:$existing\n",2090" globbed:$refname\n";2091}2092my$u= (::cmt_metadata("$refname"))[0];2093$u=~s!^\Q$url\E(/|$)!!or die2094"$refname: '$url' not found in '$u'\n";2095if($pathnamene$u) {2096warn"W: Refspec glob conflict ",2097"(ref:$refname):\n",2098"expected path:$pathname\n",2099" real path:$u\n",2100"Continuing ahead with$u\n";2101next;2102}2103}else{2104$fetch->{$pathname} =$refname;2105}2106}2107}21082109sub parse_revision_argument {2110my($base,$head) =@_;2111if(!defined$::_revision || $::_revision eq'BASE:HEAD') {2112return($base,$head);2113}2114return($1,$2)if($::_revision =~/^(\d+):(\d+)$/);2115return($::_revision, $::_revision)if($::_revision =~/^\d+$/);2116return($head,$head)if($::_revision eq'HEAD');2117return($base,$1)if($::_revision =~/^BASE:(\d+)$/);2118return($1,$head)if($::_revision =~/^(\d+):HEAD$/);2119die"revision argument: $::_revision not understood by git-svn\n";2120}21212122sub fetch_all {2123my($repo_id,$remotes) =@_;2124if(ref$repo_id) {2125my$gs=$repo_id;2126$repo_id=undef;2127$repo_id=$gs->{repo_id};2128}2129$remotes||= read_all_remotes();2130my$remote=$remotes->{$repo_id}or2131die"[svn-remote\"$repo_id\"] unknown\n";2132my$fetch=$remote->{fetch};2133my$url=$remote->{url}or die"svn-remote.$repo_id.url not defined\n";2134my(@gs,@globs);2135my$ra= Git::SVN::Ra->new($url);2136my$uuid=$ra->get_uuid;2137my$head=$ra->get_latest_revnum;21382139# ignore errors, $head revision may not even exist anymore2140eval{$ra->get_log("",$head,0,1,0,1,sub{$head=$_[1] }) };2141warn"W:$@\n"if$@;21422143my$base=defined$fetch?$head:0;21442145# read the max revs for wildcard expansion (branches/*, tags/*)2146foreachmy$t(qw/branches tags/) {2147defined$remote->{$t}ornext;2148push@globs, @{$remote->{$t}};21492150my$max_rev=eval{ tmp_config(qw/--int --get/,2151"svn-remote.$repo_id.${t}-maxRev") };2152if(defined$max_rev&& ($max_rev<$base)) {2153$base=$max_rev;2154}elsif(!defined$max_rev) {2155$base=0;2156}2157}21582159if($fetch) {2160foreachmy$p(sort keys%$fetch) {2161my$gs= Git::SVN->new($fetch->{$p},$repo_id,$p);2162my$lr=$gs->rev_map_max;2163if(defined$lr) {2164$base=$lrif($lr<$base);2165}2166push@gs,$gs;2167}2168}21692170($base,$head) = parse_revision_argument($base,$head);2171$ra->gs_fetch_loop_common($base,$head, \@gs, \@globs);2172}21732174sub read_all_remotes {2175my$r= {};2176my$use_svm_props=eval{ command_oneline(qw/config --bool2177 svn.useSvmProps/) };2178$use_svm_props=$use_svm_propseq'true'if$use_svm_props;2179my$svn_refspec=qr{\s*(.*?)\s*:\s*(.+?)\s*};2180foreach(grep{s/^svn-remote\.//} command(qw/config -l/)) {2181if(m!^(.+)\.fetch=$svn_refspec$!) {2182my($remote,$local_ref,$remote_ref) = ($1,$2,$3);2183die("svn-remote.$remote: remote ref '$remote_ref' "2184."must start with 'refs/'\n")2185unless$remote_ref=~m{^refs/};2186$local_ref= uri_decode($local_ref);2187$r->{$remote}->{fetch}->{$local_ref} =$remote_ref;2188$r->{$remote}->{svm} = {}if$use_svm_props;2189}elsif(m!^(.+)\.usesvmprops=\s*(.*)\s*$!) {2190$r->{$1}->{svm} = {};2191}elsif(m!^(.+)\.url=\s*(.*)\s*$!) {2192$r->{$1}->{url} =$2;2193}elsif(m!^(.+)\.pushurl=\s*(.*)\s*$!) {2194$r->{$1}->{pushurl} =$2;2195}elsif(m!^(.+)\.(branches|tags)=$svn_refspec$!) {2196my($remote,$t,$local_ref,$remote_ref) =2197($1,$2,$3,$4);2198die("svn-remote.$remote: remote ref '$remote_ref' ($t) "2199."must start with 'refs/'\n")2200unless$remote_ref=~m{^refs/};2201$local_ref= uri_decode($local_ref);2202my$rs= {2203 t =>$t,2204 remote =>$remote,2205 path => Git::SVN::GlobSpec->new($local_ref,1),2206ref=> Git::SVN::GlobSpec->new($remote_ref,0) };2207if(length($rs->{ref}->{right}) !=0) {2208die"The '*' glob character must be the last ",2209"character of '$remote_ref'\n";2210}2211push@{$r->{$remote}->{$t} },$rs;2212}2213}22142215map{2216if(defined$r->{$_}->{svm}) {2217my$svm;2218eval{2219my$section="svn-remote.$_";2220$svm= {2221 source => tmp_config('--get',2222"$section.svm-source"),2223 replace => tmp_config('--get',2224"$section.svm-replace"),2225}2226};2227$r->{$_}->{svm} =$svm;2228}2229}keys%$r;22302231$r;2232}22332234sub init_vars {2235$_gc_nr=$_gc_period=1000;2236if(defined$_repack||defined$_repack_flags) {2237warn"Repack options are obsolete; they have no effect.\n";2238}2239}22402241sub verify_remotes_sanity {2242return unless-d $ENV{GIT_DIR};2243my%seen;2244foreach(command(qw/config -l/)) {2245if(m!^svn-remote\.(?:.+)\.fetch=.*:refs/remotes/(\S+)\s*$!) {2246if($seen{$1}) {2247die"Remote ref refs/remote/$1is tracked by",2248"\n \"$_\"\nand\n \"$seen{$1}\"\n",2249"Please resolve this ambiguity in ",2250"your git configuration file before ",2251"continuing\n";2252}2253$seen{$1} =$_;2254}2255}2256}22572258sub find_existing_remote {2259my($url,$remotes) =@_;2260returnundefif$no_reuse_existing;2261my$existing;2262foreachmy$repo_id(keys%$remotes) {2263my$u=$remotes->{$repo_id}->{url}ornext;2264next if$une$url;2265$existing=$repo_id;2266last;2267}2268$existing;2269}22702271sub init_remote_config {2272my($self,$url,$no_write) =@_;2273$url=~s!/+$!!;# strip trailing slash2274my$r= read_all_remotes();2275my$existing= find_existing_remote($url,$r);2276if($existing) {2277unless($no_write) {2278print STDERR "Using existing ",2279"[svn-remote\"$existing\"]\n";2280}2281$self->{repo_id} =$existing;2282}elsif($_minimize_url) {2283my$min_url= Git::SVN::Ra->new($url)->minimize_url;2284$existing= find_existing_remote($min_url,$r);2285if($existing) {2286unless($no_write) {2287print STDERR "Using existing ",2288"[svn-remote\"$existing\"]\n";2289}2290$self->{repo_id} =$existing;2291}2292if($min_urlne$url) {2293unless($no_write) {2294print STDERR "Using higher level of URL: ",2295"$url=>$min_url\n";2296}2297my$old_path=$self->{path};2298$self->{path} =$url;2299$self->{path} =~s!^\Q$min_url\E(/|$)!!;2300if(length$old_path) {2301$self->{path} .="/$old_path";2302}2303$url=$min_url;2304}2305}2306my$orig_url;2307if(!$existing) {2308# verify that we aren't overwriting anything:2309$orig_url=eval{2310 command_oneline('config','--get',2311"svn-remote.$self->{repo_id}.url")2312};2313if($orig_url&& ($orig_urlne$url)) {2314die"svn-remote.$self->{repo_id}.url already set: ",2315"$orig_url\nwanted to set to:$url\n";2316}2317}2318my($xrepo_id,$xpath) = find_ref($self->refname);2319if(!$no_write&&defined$xpath) {2320die"svn-remote.$xrepo_id.fetch already set to track ",2321"$xpath:",$self->refname,"\n";2322}2323unless($no_write) {2324 command_noisy('config',2325"svn-remote.$self->{repo_id}.url",$url);2326$self->{path} =~s{^/}{};2327$self->{path} =~s{%([0-9A-F]{2})}{chr hex($1)}ieg;2328 command_noisy('config','--add',2329"svn-remote.$self->{repo_id}.fetch",2330"$self->{path}:".$self->refname);2331}2332$self->{url} =$url;2333}23342335sub find_by_url {# repos_root and, path are optional2336my($class,$full_url,$repos_root,$path) =@_;23372338returnundefunlessdefined$full_url;2339 remove_username($full_url);2340 remove_username($repos_root)ifdefined$repos_root;2341my$remotes= read_all_remotes();2342if(defined$full_url&&defined$repos_root&& !defined$path) {2343$path=$full_url;2344$path=~ s#^\Q$repos_root\E(?:/|$)##;2345}2346foreachmy$repo_id(keys%$remotes) {2347my$u=$remotes->{$repo_id}->{url}ornext;2348 remove_username($u);2349next ifdefined$repos_root&&$repos_rootne$u;23502351my$fetch=$remotes->{$repo_id}->{fetch} || {};2352foreachmy$t(qw/branches tags/) {2353foreachmy$globspec(@{$remotes->{$repo_id}->{$t}}) {2354 resolve_local_globs($u,$fetch,$globspec);2355}2356}2357my$p=$path;2358my$rwr= rewrite_root({repo_id =>$repo_id});2359my$svm=$remotes->{$repo_id}->{svm}2360ifdefined$remotes->{$repo_id}->{svm};2361unless(defined$p) {2362$p=$full_url;2363my$z=$u;2364my$prefix='';2365if($rwr) {2366$z=$rwr;2367 remove_username($z);2368}elsif(defined$svm) {2369$z=$svm->{source};2370$prefix=$svm->{replace};2371$prefix=~ s#^\Q$u\E(?:/|$)##;2372$prefix=~ s#/$##;2373}2374$p=~ s#^\Q$z\E(?:/|$)#$prefix# or next;2375}2376foreachmy$f(keys%$fetch) {2377next if$fne$p;2378return Git::SVN->new($fetch->{$f},$repo_id,$f);2379}2380}2381undef;2382}23832384sub init {2385my($class,$url,$path,$repo_id,$ref_id,$no_write) =@_;2386my$self= _new($class,$repo_id,$ref_id,$path);2387if(defined$url) {2388$self->init_remote_config($url,$no_write);2389}2390$self;2391}23922393sub find_ref {2394my($ref_id) =@_;2395foreach(command(qw/config -l/)) {2396next unless m!^svn-remote\.(.+)\.fetch=2397 \s*(.*?)\s*:\s*(.+?)\s*$!x;2398my($repo_id,$path,$ref) = ($1,$2,$3);2399if($refeq$ref_id) {2400$path=''if($path=~ m#^\./?#);2401return($repo_id,$path);2402}2403}2404(undef,undef,undef);2405}24062407sub new {2408my($class,$ref_id,$repo_id,$path) =@_;2409if(defined$ref_id&& !defined$repo_id&& !defined$path) {2410($repo_id,$path) = find_ref($ref_id);2411if(!defined$repo_id) {2412die"Could not find a\"svn-remote.*.fetch\"key ",2413"in the repository configuration matching: ",2414"$ref_id\n";2415}2416}2417my$self= _new($class,$repo_id,$ref_id,$path);2418if(!defined$self->{path} || !length$self->{path}) {2419my$fetch= command_oneline('config','--get',2420"svn-remote.$repo_id.fetch",2421":$ref_id\$")or2422die"Failed to read\"svn-remote.$repo_id.fetch\"",2423"\":$ref_id\$\"in config\n";2424($self->{path},undef) =split(/\s*:\s*/,$fetch);2425}2426$self->{path} =~s{/+}{/}g;2427$self->{path} =~s{\A/}{};2428$self->{path} =~s{/\z}{};2429$self->{url} = command_oneline('config','--get',2430"svn-remote.$repo_id.url")or2431die"Failed to read\"svn-remote.$repo_id.url\"in config\n";2432$self->{pushurl} =eval{ command_oneline('config','--get',2433"svn-remote.$repo_id.pushurl") };2434$self->rebuild;2435$self;2436}24372438sub refname {2439my($refname) =$_[0]->{ref_id} ;24402441# It cannot end with a slash /, we'll throw up on this because2442# SVN can't have directories with a slash in their name, either:2443if($refname=~m{/$}) {2444die"ref: '$refname' ends with a trailing slash, this is ",2445"not permitted by git nor Subversion\n";2446}24472448# It cannot have ASCII control character space, tilde ~, caret ^,2449# colon :, question-mark ?, asterisk *, space, or open bracket [2450# anywhere.2451#2452# Additionally, % must be escaped because it is used for escaping2453# and we want our escaped refname to be reversible2454$refname=~s{([ \%~\^:\?\*\[\t])}{uc sprintf('%%%02x',ord($1))}eg;24552456# no slash-separated component can begin with a dot .2457# /.* becomes /%2E*2458$refname=~s{/\.}{/%2E}g;24592460# It cannot have two consecutive dots .. anywhere2461# .. becomes %2E%2E2462$refname=~s{\.\.}{%2E%2E}g;24632464# trailing dots and .lock are not allowed2465# .$ becomes %2E and .lock becomes %2Elock2466$refname=~s{\.(?=$|lock$)}{%2E};24672468# the sequence @{ is used to access the reflog2469# @{ becomes %40{2470$refname=~s{\@\{}{%40\{}g;24712472return$refname;2473}24742475sub desanitize_refname {2476my($refname) =@_;2477$refname=~s{%(?:([0-9A-F]{2}))}{chr hex($1)}eg;2478return$refname;2479}24802481sub svm_uuid {2482my($self) =@_;2483return$self->{svm}->{uuid}if$self->svm;2484$self->ra;2485unless($self->{svm}) {2486die"SVM UUID not cached, and reading remotely failed\n";2487}2488$self->{svm}->{uuid};2489}24902491sub svm {2492my($self) =@_;2493return$self->{svm}if$self->{svm};2494my$svm;2495# see if we have it in our config, first:2496eval{2497my$section="svn-remote.$self->{repo_id}";2498$svm= {2499 source => tmp_config('--get',"$section.svm-source"),2500 uuid => tmp_config('--get',"$section.svm-uuid"),2501 replace => tmp_config('--get',"$section.svm-replace"),2502}2503};2504if($svm&&$svm->{source} &&$svm->{uuid} &&$svm->{replace}) {2505$self->{svm} =$svm;2506}2507$self->{svm};2508}25092510sub _set_svm_vars {2511my($self,$ra) =@_;2512return$raif$self->svm;25132514my@err= ("useSvmProps set, but failed to read SVM properties\n",2515"(svm:source, svm:uuid) ",2516"from the following URLs:\n");2517sub read_svm_props {2518my($self,$ra,$path,$r) =@_;2519my$props= ($ra->get_dir($path,$r))[2];2520my$src=$props->{'svm:source'};2521my$uuid=$props->{'svm:uuid'};2522returnundefif(!$src|| !$uuid);25232524chomp($src,$uuid);25252526$uuid=~m{^[0-9a-f\-]{30,}$}i2527or die"doesn't look right - svm:uuid is '$uuid'\n";25282529# the '!' is used to mark the repos_root!/relative/path2530$src=~s{/?!/?}{/};2531$src=~s{/+$}{};# no trailing slashes please2532# username is of no interest2533$src=~s{(^[a-z\+]*://)[^/@]*@}{$1};25342535my$replace=$ra->{url};2536$replace.="/$path"iflength$path;25372538my$section="svn-remote.$self->{repo_id}";2539 tmp_config("$section.svm-source",$src);2540 tmp_config("$section.svm-replace",$replace);2541 tmp_config("$section.svm-uuid",$uuid);2542$self->{svm} = {2543 source =>$src,2544 uuid =>$uuid,2545 replace =>$replace2546};2547}25482549my$r=$ra->get_latest_revnum;2550my$path=$self->{path};2551my%tried;2552while(length$path) {2553unless($tried{"$self->{url}/$path"}) {2554return$raif$self->read_svm_props($ra,$path,$r);2555$tried{"$self->{url}/$path"} =1;2556}2557$path=~ s#/?[^/]+$##;2558}2559die"Path: '$path' should be ''\n"if$pathne'';2560return$raif$self->read_svm_props($ra,$path,$r);2561$tried{"$self->{url}/$path"} =1;25622563if($ra->{repos_root}eq$self->{url}) {2564die@err, (map{"$_\n"}keys%tried),"\n";2565}25662567# nope, make sure we're connected to the repository root:2568my$ok;2569my@tried_b;2570$path=$ra->{svn_path};2571$ra= Git::SVN::Ra->new($ra->{repos_root});2572while(length$path) {2573unless($tried{"$ra->{url}/$path"}) {2574$ok=$self->read_svm_props($ra,$path,$r);2575last if$ok;2576$tried{"$ra->{url}/$path"} =1;2577}2578$path=~ s#/?[^/]+$##;2579}2580die"Path: '$path' should be ''\n"if$pathne'';2581$ok||=$self->read_svm_props($ra,$path,$r);2582$tried{"$ra->{url}/$path"} =1;2583if(!$ok) {2584die@err, (map{"$_\n"}keys%tried),"\n";2585}2586 Git::SVN::Ra->new($self->{url});2587}25882589sub svnsync {2590my($self) =@_;2591return$self->{svnsync}if$self->{svnsync};25922593if($self->no_metadata) {2594die"Can't have both 'noMetadata' and ",2595"'useSvnsyncProps' options set!\n";2596}2597if($self->rewrite_root) {2598die"Can't have both 'useSvnsyncProps' and 'rewriteRoot' ",2599"options set!\n";2600}2601if($self->rewrite_uuid) {2602die"Can't have both 'useSvnsyncProps' and 'rewriteUUID' ",2603"options set!\n";2604}26052606my$svnsync;2607# see if we have it in our config, first:2608eval{2609my$section="svn-remote.$self->{repo_id}";26102611my$url= tmp_config('--get',"$section.svnsync-url");2612($url) = ($url=~m{^([a-z\+]+://\S+)$})or2613die"doesn't look right - svn:sync-from-url is '$url'\n";26142615my$uuid= tmp_config('--get',"$section.svnsync-uuid");2616($uuid) = ($uuid=~m{^([0-9a-f\-]{30,})$}i)or2617die"doesn't look right - svn:sync-from-uuid is '$uuid'\n";26182619$svnsync= { url =>$url, uuid =>$uuid}2620};2621if($svnsync&&$svnsync->{url} &&$svnsync->{uuid}) {2622return$self->{svnsync} =$svnsync;2623}26242625my$err="useSvnsyncProps set, but failed to read ".2626"svnsync property: svn:sync-from-";2627my$rp=$self->ra->rev_proplist(0);26282629my$url=$rp->{'svn:sync-from-url'}or die$err."url\n";2630($url) = ($url=~m{^([a-z\+]+://\S+)$})or2631die"doesn't look right - svn:sync-from-url is '$url'\n";26322633my$uuid=$rp->{'svn:sync-from-uuid'}or die$err."uuid\n";2634($uuid) = ($uuid=~m{^([0-9a-f\-]{30,})$}i)or2635die"doesn't look right - svn:sync-from-uuid is '$uuid'\n";26362637my$section="svn-remote.$self->{repo_id}";2638 tmp_config('--add',"$section.svnsync-uuid",$uuid);2639 tmp_config('--add',"$section.svnsync-url",$url);2640return$self->{svnsync} = { url =>$url, uuid =>$uuid};2641}26422643# this allows us to memoize our SVN::Ra UUID locally and avoid a2644# remote lookup (useful for 'git svn log').2645sub ra_uuid {2646my($self) =@_;2647unless($self->{ra_uuid}) {2648my$key="svn-remote.$self->{repo_id}.uuid";2649my$uuid=eval{ tmp_config('--get',$key) };2650if(!$@&&$uuid&&$uuid=~/^([a-f\d\-]{30,})$/i) {2651$self->{ra_uuid} =$uuid;2652}else{2653die"ra_uuid called without URL\n"unless$self->{url};2654$self->{ra_uuid} =$self->ra->get_uuid;2655 tmp_config('--add',$key,$self->{ra_uuid});2656}2657}2658$self->{ra_uuid};2659}26602661sub _set_repos_root {2662my($self,$repos_root) =@_;2663my$k="svn-remote.$self->{repo_id}.reposRoot";2664$repos_root||=$self->ra->{repos_root};2665 tmp_config($k,$repos_root);2666$repos_root;2667}26682669sub repos_root {2670my($self) =@_;2671my$k="svn-remote.$self->{repo_id}.reposRoot";2672eval{ tmp_config('--get',$k) } ||$self->_set_repos_root;2673}26742675sub ra {2676my($self) =shift;2677my$ra= Git::SVN::Ra->new($self->{url});2678$self->_set_repos_root($ra->{repos_root});2679if($self->use_svm_props&& !$self->{svm}) {2680if($self->no_metadata) {2681die"Can't have both 'noMetadata' and ",2682"'useSvmProps' options set!\n";2683}elsif($self->use_svnsync_props) {2684die"Can't have both 'useSvnsyncProps' and ",2685"'useSvmProps' options set!\n";2686}2687$ra=$self->_set_svm_vars($ra);2688$self->{-want_revprops} =1;2689}2690$ra;2691}26922693# prop_walk(PATH, REV, SUB)2694# -------------------------2695# Recursively traverse PATH at revision REV and invoke SUB for each2696# directory that contains a SVN property. SUB will be invoked as2697# follows: &SUB(gs, path, props); where `gs' is this instance of2698# Git::SVN, `path' the path to the directory where the properties2699# `props' were found. The `path' will be relative to point of checkout,2700# that is, if url://repo/trunk is the current Git branch, and that2701# directory contains a sub-directory `d', SUB will be invoked with `/d/'2702# as `path' (note the trailing `/').2703sub prop_walk {2704my($self,$path,$rev,$sub) =@_;27052706$path=~ s#^/##;2707my($dirent,undef,$props) =$self->ra->get_dir($path,$rev);2708$path=~ s#^/*#/#g;2709my$p=$path;2710# Strip the irrelevant part of the path.2711$p=~ s#^/+\Q$self->{path}\E(/|$)#/#;2712# Ensure the path is terminated by a `/'.2713$p=~ s#/*$#/#;27142715# The properties contain all the internal SVN stuff nobody2716# (usually) cares about.2717my$interesting_props=0;2718foreach(keys%{$props}) {2719# If it doesn't start with `svn:', it must be a2720# user-defined property.2721++$interesting_propsandnext if$_!~/^svn:/;2722# FIXME: Fragile, if SVN adds new public properties,2723# this needs to be updated.2724++$interesting_propsif/^svn:(?:ignore|keywords|executable2725|eol-style|mime-type2726|externals|needs-lock)$/x;2727}2728&$sub($self,$p,$props)if$interesting_props;27292730foreach(sort keys%$dirent) {2731next if$dirent->{$_}->{kind} !=$SVN::Node::dir;2732$self->prop_walk($self->{path} .$p.$_,$rev,$sub);2733}2734}27352736sub last_rev { ($_[0]->last_rev_commit)[0] }2737sub last_commit { ($_[0]->last_rev_commit)[1] }27382739# returns the newest SVN revision number and newest commit SHA12740sub last_rev_commit {2741my($self) =@_;2742if(defined$self->{last_rev} &&defined$self->{last_commit}) {2743return($self->{last_rev},$self->{last_commit});2744}2745my$c= ::verify_ref($self->refname.'^0');2746if($c&& !$self->use_svm_props&& !$self->no_metadata) {2747my$rev= (::cmt_metadata($c))[1];2748if(defined$rev) {2749($self->{last_rev},$self->{last_commit}) = ($rev,$c);2750return($rev,$c);2751}2752}2753my$map_path=$self->map_path;2754unless(-e $map_path) {2755($self->{last_rev},$self->{last_commit}) = (undef,undef);2756return(undef,undef);2757}2758my($rev,$commit) =$self->rev_map_max(1);2759($self->{last_rev},$self->{last_commit}) = ($rev,$commit);2760return($rev,$commit);2761}27622763sub get_fetch_range {2764my($self,$min,$max) =@_;2765$max||=$self->ra->get_latest_revnum;2766$min||=$self->rev_map_max;2767(++$min,$max);2768}27692770sub tmp_config {2771my(@args) =@_;2772my$old_def_config="$ENV{GIT_DIR}/svn/config";2773my$config="$ENV{GIT_DIR}/svn/.metadata";2774if(! -f $config&& -f $old_def_config) {2775rename$old_def_config,$configor2776die"Failed rename$old_def_config=>$config:$!\n";2777}2778my$old_config=$ENV{GIT_CONFIG};2779$ENV{GIT_CONFIG} =$config;2780$@=undef;2781my@ret=eval{2782unless(-f $config) {2783 mkfile($config);2784open my$fh,'>',$configor2785die"Can't open$config:$!\n";2786print$fh"; This file is used internally by ",2787"git-svn\n"or die2788"Couldn't write to$config:$!\n";2789print$fh"; You should not have to edit it\n"or2790die"Couldn't write to$config:$!\n";2791close$fhor die"Couldn't close$config:$!\n";2792}2793 command('config',@args);2794};2795my$err=$@;2796if(defined$old_config) {2797$ENV{GIT_CONFIG} =$old_config;2798}else{2799delete$ENV{GIT_CONFIG};2800}2801die$errif$err;2802wantarray?@ret:$ret[0];2803}28042805sub tmp_index_do {2806my($self,$sub) =@_;2807my$old_index=$ENV{GIT_INDEX_FILE};2808$ENV{GIT_INDEX_FILE} =$self->{index};2809$@=undef;2810my@ret=eval{2811my($dir,$base) = ($self->{index} =~ m#^(.*?)/?([^/]+)$#);2812 mkpath([$dir])unless-d $dir;2813&$sub;2814};2815my$err=$@;2816if(defined$old_index) {2817$ENV{GIT_INDEX_FILE} =$old_index;2818}else{2819delete$ENV{GIT_INDEX_FILE};2820}2821die$errif$err;2822wantarray?@ret:$ret[0];2823}28242825sub assert_index_clean {2826my($self,$treeish) =@_;28272828$self->tmp_index_do(sub{2829 command_noisy('read-tree',$treeish)unless-e $self->{index};2830my$x= command_oneline('write-tree');2831my($y) = (command(qw/cat-file commit/,$treeish) =~2832/^tree ($::sha1)/mo);2833return if$yeq$x;28342835warn"Index mismatch:$y!=$x\nrereading$treeish\n";2836unlink$self->{index}or die"unlink$self->{index}:$!\n";2837 command_noisy('read-tree',$treeish);2838$x= command_oneline('write-tree');2839if($yne$x) {2840::fatal "trees ($treeish)$y!=$x\n",2841"Something is seriously wrong...";2842}2843});2844}28452846sub get_commit_parents {2847my($self,$log_entry) =@_;2848my(%seen,@ret,@tmp);2849# legacy support for 'set-tree'; this is only used by set_tree_cb:2850if(my$ip=$self->{inject_parents}) {2851if(my$commit=delete$ip->{$log_entry->{revision}}) {2852push@tmp,$commit;2853}2854}2855if(my$cur= ::verify_ref($self->refname.'^0')) {2856push@tmp,$cur;2857}2858if(my$ipd=$self->{inject_parents_dcommit}) {2859if(my$commit=delete$ipd->{$log_entry->{revision}}) {2860push@tmp,@$commit;2861}2862}2863push@tmp,$_foreach(@{$log_entry->{parents}},@tmp);2864while(my$p=shift@tmp) {2865next if$seen{$p};2866$seen{$p} =1;2867push@ret,$p;2868}2869@ret;2870}28712872sub rewrite_root {2873my($self) =@_;2874return$self->{-rewrite_root}ifexists$self->{-rewrite_root};2875my$k="svn-remote.$self->{repo_id}.rewriteRoot";2876my$rwr=eval{ command_oneline(qw/config --get/,$k) };2877if($rwr) {2878$rwr=~ s#/+$##;2879if($rwr!~ m#^[a-z\+]+://#) {2880die"$rwris not a valid URL (key:$k)\n";2881}2882}2883$self->{-rewrite_root} =$rwr;2884}28852886sub rewrite_uuid {2887my($self) =@_;2888return$self->{-rewrite_uuid}ifexists$self->{-rewrite_uuid};2889my$k="svn-remote.$self->{repo_id}.rewriteUUID";2890my$rwid=eval{ command_oneline(qw/config --get/,$k) };2891if($rwid) {2892$rwid=~ s#/+$##;2893if($rwid!~ m#^[a-f0-9]{8}-(?:[a-f0-9]{4}-){3}[a-f0-9]{12}$#) {2894die"$rwidis not a valid UUID (key:$k)\n";2895}2896}2897$self->{-rewrite_uuid} =$rwid;2898}28992900sub metadata_url {2901my($self) =@_;2902($self->rewrite_root||$self->{url}) .2903(length$self->{path} ?'/'.$self->{path} :'');2904}29052906sub full_url {2907my($self) =@_;2908$self->{url} . (length$self->{path} ?'/'.$self->{path} :'');2909}29102911sub full_pushurl {2912my($self) =@_;2913if($self->{pushurl}) {2914return$self->{pushurl} . (length$self->{path} ?'/'.2915$self->{path} :'');2916}else{2917return$self->full_url;2918}2919}29202921sub set_commit_header_env {2922my($log_entry) =@_;2923my%env;2924foreachmy$ned(qw/NAME EMAIL DATE/) {2925foreachmy$ac(qw/AUTHOR COMMITTER/) {2926$env{"GIT_${ac}_${ned}"} =$ENV{"GIT_${ac}_${ned}"};2927}2928}29292930$ENV{GIT_AUTHOR_NAME} =$log_entry->{name};2931$ENV{GIT_AUTHOR_EMAIL} =$log_entry->{email};2932$ENV{GIT_AUTHOR_DATE} =$ENV{GIT_COMMITTER_DATE} =$log_entry->{date};29332934$ENV{GIT_COMMITTER_NAME} = (defined$log_entry->{commit_name})2935?$log_entry->{commit_name}2936:$log_entry->{name};2937$ENV{GIT_COMMITTER_EMAIL} = (defined$log_entry->{commit_email})2938?$log_entry->{commit_email}2939:$log_entry->{email};2940 \%env;2941}29422943sub restore_commit_header_env {2944my($env) =@_;2945foreachmy$ned(qw/NAME EMAIL DATE/) {2946foreachmy$ac(qw/AUTHOR COMMITTER/) {2947my$k="GIT_${ac}_${ned}";2948if(defined$env->{$k}) {2949$ENV{$k} =$env->{$k};2950}else{2951delete$ENV{$k};2952}2953}2954}2955}29562957sub gc {2958 command_noisy('gc','--auto');2959};29602961sub do_git_commit {2962my($self,$log_entry) =@_;2963my$lr=$self->last_rev;2964if(defined$lr&&$lr>=$log_entry->{revision}) {2965die"Last fetched revision of ",$self->refname,2966" was r$lr, but we are about to fetch: ",2967"r$log_entry->{revision}!\n";2968}2969if(my$c=$self->rev_map_get($log_entry->{revision})) {2970 croak "$log_entry->{revision} =$calready exists! ",2971"Why are we refetching it?\n";2972}2973my$old_env= set_commit_header_env($log_entry);2974my$tree=$log_entry->{tree};2975if(!defined$tree) {2976$tree=$self->tmp_index_do(sub{2977 command_oneline('write-tree') });2978}2979die"Tree is not a valid sha1:$tree\n"if$tree!~/^$::sha1$/o;29802981my@exec= ('git','commit-tree',$tree);2982foreach($self->get_commit_parents($log_entry)) {2983push@exec,'-p',$_;2984}2985defined(my$pid= open3(my$msg_fh,my$out_fh,'>&STDERR',@exec))2986or croak $!;2987binmode$msg_fh;29882989# we always get UTF-8 from SVN, but we may want our commits in2990# a different encoding.2991if(my$enc= Git::config('i18n.commitencoding')) {2992require Encode;2993 Encode::from_to($log_entry->{log},'UTF-8',$enc);2994}2995print$msg_fh $log_entry->{log}or croak $!;2996 restore_commit_header_env($old_env);2997unless($self->no_metadata) {2998print$msg_fh"\ngit-svn-id:$log_entry->{metadata}\n"2999or croak $!;3000}3001$msg_fh->flush==0or croak $!;3002close$msg_fhor croak $!;3003chomp(my$commit=do{local$/; <$out_fh> });3004close$out_fhor croak $!;3005waitpid$pid,0;3006 croak $?if$?;3007if($commit!~/^$::sha1$/o) {3008die"Failed to commit, invalid sha1:$commit\n";3009}30103011$self->rev_map_set($log_entry->{revision},$commit,1);30123013$self->{last_rev} =$log_entry->{revision};3014$self->{last_commit} =$commit;3015print"r$log_entry->{revision}"unless$::_q >1;3016if(defined$log_entry->{svm_revision}) {3017print" (\@$log_entry->{svm_revision})"unless$::_q >1;3018$self->rev_map_set($log_entry->{svm_revision},$commit,30190,$self->svm_uuid);3020}3021print" =$commit($self->{ref_id})\n"unless$::_q >1;3022if(--$_gc_nr==0) {3023$_gc_nr=$_gc_period;3024 gc();3025}3026return$commit;3027}30283029sub match_paths {3030my($self,$paths,$r) =@_;3031return1if$self->{path}eq'';3032if(my$path=$paths->{"/$self->{path}"}) {3033return($path->{action}eq'D') ?0:1;3034}3035$self->{path_regex} ||=qr/^\/\Q$self->{path}\E\//;3036if(grep/$self->{path_regex}/,keys%$paths) {3037return1;3038}3039my$c='';3040foreach(split m#/#, $self->{path}) {3041$c.="/$_";3042next unless($paths->{$c} &&3043($paths->{$c}->{action} =~/^[AR]$/));3044if($self->ra->check_path($self->{path},$r) ==3045$SVN::Node::dir) {3046return1;3047}3048}3049return0;3050}30513052sub find_parent_branch {3053my($self,$paths,$rev) =@_;3054returnundefunless$self->follow_parent;3055unless(defined$paths) {3056my$err_handler=$SVN::Error::handler;3057$SVN::Error::handler = \&Git::SVN::Ra::skip_unknown_revs;3058$self->ra->get_log([$self->{path}],$rev,$rev,0,1,1,3059sub{$paths=$_[0] });3060$SVN::Error::handler =$err_handler;3061}3062returnundefunlessdefined$paths;30633064# look for a parent from another branch:3065my@b_path_components=split m#/#, $self->{path};3066my@a_path_components;3067my$i;3068while(@b_path_components) {3069$i=$paths->{'/'.join('/',@b_path_components)};3070last if$i&&defined$i->{copyfrom_path};3071unshift(@a_path_components,pop(@b_path_components));3072}3073returnundefunlessdefined$i&&defined$i->{copyfrom_path};3074my$branch_from=$i->{copyfrom_path};3075if(@a_path_components) {3076print STDERR "branch_from:$branch_from=> ";3077$branch_from.='/'.join('/',@a_path_components);3078print STDERR $branch_from,"\n";3079}3080my$r=$i->{copyfrom_rev};3081my$repos_root=$self->ra->{repos_root};3082my$url=$self->ra->{url};3083my$new_url=$url.$branch_from;3084print STDERR "Found possible branch point: ",3085"$new_url=> ",$self->full_url,",$r\n"3086unless$::_q >1;3087$branch_from=~ s#^/##;3088my$gs=$self->other_gs($new_url,$url,3089$branch_from,$r,$self->{ref_id});3090my($r0,$parent) =$gs->find_rev_before($r,1);3091{3092my($base,$head);3093if(!defined$r0|| !defined$parent) {3094($base,$head) = parse_revision_argument(0,$r);3095}else{3096if($r0<$r) {3097$gs->ra->get_log([$gs->{path}],$r0+1,$r,1,30980,1,sub{$base=$_[1] -1});3099}3100}3101if(defined$base&&$base<=$r) {3102$gs->fetch($base,$r);3103}3104($r0,$parent) =$gs->find_rev_before($r,1);3105}3106if(defined$r0&&defined$parent) {3107print STDERR "Found branch parent: ($self->{ref_id})$parent\n"3108unless$::_q >1;3109my$ed;3110if($self->ra->can_do_switch) {3111$self->assert_index_clean($parent);3112print STDERR "Following parent with do_switch\n"3113unless$::_q >1;3114# do_switch works with svn/trunk >= r22312, but that3115# is not included with SVN 1.4.3 (the latest version3116# at the moment), so we can't rely on it3117$self->{last_rev} =$r0;3118$self->{last_commit} =$parent;3119$ed= SVN::Git::Fetcher->new($self,$gs->{path});3120$gs->ra->gs_do_switch($r0,$rev,$gs,3121$self->full_url,$ed)3122or die"SVN connection failed somewhere...\n";3123}elsif($self->ra->trees_match($new_url,$r0,3124$self->full_url,$rev)) {3125print STDERR "Trees match:\n",3126"$new_url\@$r0\n",3127" ${\$self->full_url}\@$rev\n",3128"Following parent with no changes\n"3129unless$::_q >1;3130$self->tmp_index_do(sub{3131 command_noisy('read-tree',$parent);3132});3133$self->{last_commit} =$parent;3134}else{3135print STDERR "Following parent with do_update\n"3136unless$::_q >1;3137$ed= SVN::Git::Fetcher->new($self);3138$self->ra->gs_do_update($rev,$rev,$self,$ed)3139or die"SVN connection failed somewhere...\n";3140}3141print STDERR "Successfully followed parent\n"unless$::_q >1;3142return$self->make_log_entry($rev, [$parent],$ed);3143}3144returnundef;3145}31463147sub do_fetch {3148my($self,$paths,$rev) =@_;3149my$ed;3150my($last_rev,@parents);3151if(my$lc=$self->last_commit) {3152# we can have a branch that was deleted, then re-added3153# under the same name but copied from another path, in3154# which case we'll have multiple parents (we don't3155# want to break the original ref, nor lose copypath info):3156if(my$log_entry=$self->find_parent_branch($paths,$rev)) {3157push@{$log_entry->{parents}},$lc;3158return$log_entry;3159}3160$ed= SVN::Git::Fetcher->new($self);3161$last_rev=$self->{last_rev};3162$ed->{c} =$lc;3163@parents= ($lc);3164}else{3165$last_rev=$rev;3166if(my$log_entry=$self->find_parent_branch($paths,$rev)) {3167return$log_entry;3168}3169$ed= SVN::Git::Fetcher->new($self);3170}3171unless($self->ra->gs_do_update($last_rev,$rev,$self,$ed)) {3172die"SVN connection failed somewhere...\n";3173}3174$self->make_log_entry($rev, \@parents,$ed);3175}31763177sub mkemptydirs {3178my($self,$r) =@_;31793180sub scan {3181my($r,$empty_dirs,$line) =@_;3182if(defined$r&&$line=~/^r(\d+)$/) {3183return0if$1>$r;3184}elsif($line=~/^ \+empty_dir: (.+)$/) {3185$empty_dirs->{$1} =1;3186}elsif($line=~/^ \-empty_dir: (.+)$/) {3187my@d=grep{m[^\Q$1\E(/|$)]} (keys%$empty_dirs);3188delete@$empty_dirs{@d};3189}31901;# continue3191};31923193my%empty_dirs= ();3194my$gz_file="$self->{dir}/unhandled.log.gz";3195if(-f $gz_file) {3196if(!$can_compress) {3197warn"Compress::Zlib could not be found; ",3198"empty directories in$gz_filewill not be read\n";3199}else{3200my$gz= Compress::Zlib::gzopen($gz_file,"rb")or3201die"Unable to open$gz_file:$!\n";3202my$line;3203while($gz->gzreadline($line) >0) {3204 scan($r, \%empty_dirs,$line)orlast;3205}3206$gz->gzclose;3207}3208}32093210if(open my$fh,'<',"$self->{dir}/unhandled.log") {3211binmode$fhor croak "binmode:$!";3212while(<$fh>) {3213 scan($r, \%empty_dirs,$_)orlast;3214}3215close$fh;3216}32173218my$strip=qr/\A\Q$self->{path}\E(?:\/|$)/;3219foreachmy$d(sort keys%empty_dirs) {3220$d= uri_decode($d);3221$d=~s/$strip//;3222next unlesslength($d);3223next if-d $d;3224if(-e $d) {3225warn"$dexists but is not a directory\n";3226}else{3227print"creating empty directory:$d\n";3228 mkpath([$d]);3229}3230}3231}32323233sub get_untracked {3234my($self,$ed) =@_;3235my@out;3236my$h=$ed->{empty};3237foreach(sort keys%$h) {3238my$act=$h->{$_} ?'+empty_dir':'-empty_dir';3239push@out,"$act: ". uri_encode($_);3240warn"W:$act:$_\n";3241}3242foreachmy$t(qw/dir_prop file_prop/) {3243$h=$ed->{$t}ornext;3244foreachmy$path(sort keys%$h) {3245my$ppath=$patheq''?'.':$path;3246foreachmy$prop(sort keys%{$h->{$path}}) {3247next if$SKIP_PROP{$prop};3248my$v=$h->{$path}->{$prop};3249my$t_ppath_prop="$t: ".3250 uri_encode($ppath) .' '.3251 uri_encode($prop);3252if(defined$v) {3253push@out," +$t_ppath_prop".3254 uri_encode($v);3255}else{3256push@out," -$t_ppath_prop";3257}3258}3259}3260}3261foreachmy$t(qw/absent_file absent_directory/) {3262$h=$ed->{$t}ornext;3263foreachmy$parent(sort keys%$h) {3264foreachmy$path(sort@{$h->{$parent}}) {3265push@out,"$t: ".3266 uri_encode("$parent/$path");3267warn"W:$t:$parent/$path",3268"Insufficient permissions?\n";3269}3270}3271}3272 \@out;3273}32743275# parse_svn_date(DATE)3276# --------------------3277# Given a date (in UTC) from Subversion, return a string in the format3278# "<TZ Offset> <local date/time>" that Git will use.3279#3280# By default the parsed date will be in UTC; if $Git::SVN::_localtime3281# is true we'll convert it to the local timezone instead.3282sub parse_svn_date {3283my$date=shift||return'+0000 1970-01-01 00:00:00';3284my($Y,$m,$d,$H,$M,$S) = ($date=~ /^(\d{4})\-(\d\d)\-(\d\d)T3285(\d\d)\:(\d\d)\:(\d\d)\.\d*Z$/x)or3286 croak "Unable to parse date:$date\n";3287my$parsed_date;# Set next.32883289if($Git::SVN::_localtime) {3290# Translate the Subversion datetime to an epoch time.3291# Begin by switching ourselves to $date's timezone, UTC.3292my$old_env_TZ=$ENV{TZ};3293$ENV{TZ} ='UTC';32943295my$epoch_in_UTC=3296 POSIX::strftime('%s',$S,$M,$H,$d,$m-1,$Y-1900);32973298# Determine our local timezone (including DST) at the3299# time of $epoch_in_UTC. $Git::SVN::Log::TZ stored the3300# value of TZ, if any, at the time we were run.3301if(defined$Git::SVN::Log::TZ) {3302$ENV{TZ} =$Git::SVN::Log::TZ;3303}else{3304delete$ENV{TZ};3305}33063307my$our_TZ=3308 POSIX::strftime('%Z',$S,$M,$H,$d,$m-1,$Y-1900);33093310# This converts $epoch_in_UTC into our local timezone.3311my($sec,$min,$hour,$mday,$mon,$year,3312$wday,$yday,$isdst) =localtime($epoch_in_UTC);33133314$parsed_date=sprintf('%s%04d-%02d-%02d%02d:%02d:%02d',3315$our_TZ,$year+1900,$mon+1,3316$mday,$hour,$min,$sec);33173318# Reset us to the timezone in effect when we entered3319# this routine.3320if(defined$old_env_TZ) {3321$ENV{TZ} =$old_env_TZ;3322}else{3323delete$ENV{TZ};3324}3325}else{3326$parsed_date="+0000$Y-$m-$d$H:$M:$S";3327}33283329return$parsed_date;3330}33313332sub other_gs {3333my($self,$new_url,$url,3334$branch_from,$r,$old_ref_id) =@_;3335my$gs= Git::SVN->find_by_url($new_url,$url,$branch_from);3336unless($gs) {3337my$ref_id=$old_ref_id;3338$ref_id=~s/\@\d+-*$//;3339$ref_id.="\@$r";3340# just grow a tail if we're not unique enough :x3341$ref_id.='-'while find_ref($ref_id);3342my($u,$p,$repo_id) = ($new_url,'',$ref_id);3343if($u=~ s#^\Q$url\E(/|$)##) {3344$p=$u;3345$u=$url;3346$repo_id=$self->{repo_id};3347}3348while(1) {3349# It is possible to tag two different subdirectories at3350# the same revision. If the url for an existing ref3351# does not match, we must either find a ref with a3352# matching url or create a new ref by growing a tail.3353$gs= Git::SVN->init($u,$p,$repo_id,$ref_id,1);3354my(undef,$max_commit) =$gs->rev_map_max(1);3355last if(!$max_commit);3356my($url) = ::cmt_metadata($max_commit);3357last if($urleq$gs->metadata_url);3358$ref_id.='-';3359}3360print STDERR "Initializing parent:$ref_id\n"unless$::_q >1;3361}3362$gs3363}33643365sub call_authors_prog {3366my($orig_author) =@_;3367$orig_author= command_oneline('rev-parse','--sq-quote',$orig_author);3368my$author=`$::_authors_prog$orig_author`;3369 if ($?!= 0) {3370 die "$::_authors_prog failed with exit code$?\n"3371 }3372 if ($author=~ /^\s*(.+?)\s*<(.*)>\s*$/) {3373 my ($name,$email) = ($1,$2);3374$email= undef if length$2== 0;3375 return [$name,$email];3376 } else {3377 die "Author:$orig_author: $::_authors_prog returned "3378 . "invalid author format:$author\n";3379 }3380}33813382sub check_author {3383 my ($author) =@_;3384 if (!defined$author|| length$author== 0) {3385$author= '(no author)';3386 }3387 if (!defined $::users{$author}) {3388 if (defined $::_authors_prog) {3389 $::users{$author} = call_authors_prog($author);3390 } elsif (defined $::_authors) {3391 die "Author:$authornot defined in $::_authors file\n";3392 }3393 }3394$author;3395}33963397sub find_extra_svk_parents {3398 my ($self,$ed,$tickets,$parents) =@_;3399 # aha! svk:merge property changed...3400 my@tickets= split "\n",$tickets;3401 my@known_parents;3402 for my$ticket(@tickets) {3403 my ($uuid,$path,$rev) = split /:/,$ticket;3404 if ($uuideq$self->ra_uuid ) {3405 my$url=$self->{url};3406 my$repos_root=$url;3407 my$branch_from=$path;3408$branch_from=~ s{^/}{};3409 my$gs=$self->other_gs($repos_root."/".$branch_from,3410$url,3411$branch_from,3412$rev,3413$self->{ref_id});3414 if ( my$commit=$gs->rev_map_get($rev,$uuid) ) {3415 # wahey! we found it, but it might be3416 # an old one (!)3417 push@known_parents, [$rev,$commit];3418 }3419 }3420 }3421 # Ordering matters; highest-numbered commit merge tickets3422 # first, as they may account for later merge ticket additions3423 # or changes.3424@known_parents= map {$_->[1]} sort {$b->[0] <=>$a->[0]}@known_parents;3425 for my$parent(@known_parents) {3426 my@cmd= ('rev-list',$parent, map { "^$_" }@$parents);3427 my ($msg_fh,$ctx) = command_output_pipe(@cmd);3428 my$new;3429 while ( <$msg_fh> ) {3430$new=1;last;3431 }3432 command_close_pipe($msg_fh,$ctx);3433 if ($new) {3434 print STDERR3435 "Found merge parent (svk:merge ticket):$parent\n";3436 push@$parents,$parent;3437 }3438 }3439}34403441sub lookup_svn_merge {3442 my$uuid= shift;3443 my$url= shift;3444 my$merge= shift;34453446 my ($source,$revs) = split ":",$merge;3447 my$path=$source;3448$path=~ s{^/}{};3449 my$gs= Git::SVN->find_by_url($url.$source,$url,$path);3450 if ( !$gs) {3451 warn "Couldn't find revmap for$url$source\n";3452 return;3453 }3454 my@ranges= split ",",$revs;3455 my ($tip,$tip_commit);3456 my@merged_commit_ranges;3457 # find the tip3458 for my$range(@ranges) {3459 my ($bottom,$top) = split "-",$range;3460$top||=$bottom;3461 my$bottom_commit=$gs->find_rev_after($bottom, 1,$top);3462 my$top_commit=$gs->find_rev_before($top, 1,$bottom);34633464 unless ($top_commitand$bottom_commit) {3465 warn "W:unknown path/rev in svn:mergeinfo "3466 ."dirprop:$source:$range\n";3467 next;3468 }34693470 if (scalar(command('rev-parse', "$bottom_commit^@"))) {3471 push@merged_commit_ranges,3472 "$bottom_commit^..$top_commit";3473 } else {3474 push@merged_commit_ranges, "$top_commit";3475 }34763477 if ( !defined$tipor$top>$tip) {3478$tip=$top;3479$tip_commit=$top_commit;3480 }3481 }3482 return ($tip_commit,@merged_commit_ranges);3483}34843485sub _rev_list {3486 my ($msg_fh,$ctx) = command_output_pipe(3487 "rev-list",@_,3488 );3489 my@rv;3490 while ( <$msg_fh> ) {3491 chomp;3492 push@rv,$_;3493 }3494 command_close_pipe($msg_fh,$ctx);3495@rv;3496}34973498sub check_cherry_pick {3499 my$base= shift;3500 my$tip= shift;3501 my$parents= shift;3502 my@ranges=@_;3503 my%commits= map {$_=> 1 }3504 _rev_list("--no-merges",$tip, "--not",$base,@$parents, "--");3505 for my$range(@ranges) {3506 delete@commits{_rev_list($range, "--")};3507 }3508 for my$commit(keys%commits) {3509 if (has_no_changes($commit)) {3510 delete$commits{$commit};3511 }3512 }3513 return (keys%commits);3514}35153516sub has_no_changes {3517 my$commit= shift;35183519 my@revs= split / /, command_oneline(3520 qw(rev-list --parents -1 -m),$commit);35213522# Commits with no parents, e.g. the start of a partial branch,3523# have changes by definition.3524return1if(@revs<2);35253526# Commits with multiple parents, e.g a merge, have no changes3527# by definition.3528return0if(@revs>2);35293530return(command_oneline("rev-parse","$commit^{tree}")eq3531 command_oneline("rev-parse","$commit~1^{tree}"));3532}35333534# The GIT_DIR environment variable is not always set until after the command3535# line arguments are processed, so we can't memoize in a BEGIN block.3536{3537my$memoized=0;35383539sub memoize_svn_mergeinfo_functions {3540return if$memoized;3541$memoized=1;35423543my$cache_path="$ENV{GIT_DIR}/svn/.caches/";3544 mkpath([$cache_path])unless-d $cache_path;35453546 tie my%lookup_svn_merge_cache=>'Memoize::Storable',3547"$cache_path/lookup_svn_merge.db",'nstore';3548 memoize 'lookup_svn_merge',3549 SCALAR_CACHE =>'FAULT',3550 LIST_CACHE => ['HASH'=> \%lookup_svn_merge_cache],3551;35523553 tie my%check_cherry_pick_cache=>'Memoize::Storable',3554"$cache_path/check_cherry_pick.db",'nstore';3555 memoize 'check_cherry_pick',3556 SCALAR_CACHE =>'FAULT',3557 LIST_CACHE => ['HASH'=> \%check_cherry_pick_cache],3558;35593560 tie my%has_no_changes_cache=>'Memoize::Storable',3561"$cache_path/has_no_changes.db",'nstore';3562 memoize 'has_no_changes',3563 SCALAR_CACHE => ['HASH'=> \%has_no_changes_cache],3564 LIST_CACHE =>'FAULT',3565;3566}35673568sub unmemoize_svn_mergeinfo_functions {3569return ifnot$memoized;3570$memoized=0;35713572 Memoize::unmemoize 'lookup_svn_merge';3573 Memoize::unmemoize 'check_cherry_pick';3574 Memoize::unmemoize 'has_no_changes';3575}35763577 Memoize::memoize 'Git::SVN::repos_root';3578}35793580END{3581# Force cache writeout explicitly instead of waiting for3582# global destruction to avoid segfault in Storable:3583# http://rt.cpan.org/Public/Bug/Display.html?id=360873584 unmemoize_svn_mergeinfo_functions();3585}35863587sub parents_exclude {3588my$parents=shift;3589my@commits=@_;3590return unless@commits;35913592my@excluded;3593my$excluded;3594do{3595my@cmd= ('rev-list',"-1",@commits,"--not",@$parents);3596$excluded= command_oneline(@cmd);3597if($excluded) {3598my@new;3599my$found;3600formy$commit(@commits) {3601if($commiteq$excluded) {3602push@excluded,$commit;3603$found++;3604last;3605}3606else{3607push@new,$commit;3608}3609}3610die"saw commit '$excluded' in rev-list output, "3611."but we didn't ask for that commit (wanted:@commits--not@$parents)"3612unless$found;3613@commits=@new;3614}3615}3616while($excludedand@commits);36173618return@excluded;3619}362036213622# note: this function should only be called if the various dirprops3623# have actually changed3624sub find_extra_svn_parents {3625my($self,$ed,$mergeinfo,$parents) =@_;3626# aha! svk:merge property changed...36273628 memoize_svn_mergeinfo_functions();36293630# We first search for merged tips which are not in our3631# history. Then, we figure out which git revisions are in3632# that tip, but not this revision. If all of those revisions3633# are now marked as merge, we can add the tip as a parent.3634my@merges=split"\n",$mergeinfo;3635my@merge_tips;3636my$url=$self->{url};3637my$uuid=$self->ra_uuid;3638my%ranges;3639formy$merge(@merges) {3640my($tip_commit,@ranges) =3641 lookup_svn_merge($uuid,$url,$merge);3642unless(!$tip_commitor3643grep{$_eq$tip_commit}@$parents) {3644push@merge_tips,$tip_commit;3645$ranges{$tip_commit} = \@ranges;3646}else{3647push@merge_tips,undef;3648}3649}36503651my%excluded=map{$_=>1}3652 parents_exclude($parents,grep{defined}@merge_tips);36533654# check merge tips for new parents3655my@new_parents;3656formy$merge_tip(@merge_tips) {3657my$spec=shift@merges;3658next unless$merge_tipand$excluded{$merge_tip};36593660my$ranges=$ranges{$merge_tip};36613662# check out 'new' tips3663my$merge_base;3664eval{3665$merge_base= command_oneline(3666"merge-base",3667@$parents,$merge_tip,3668);3669};3670if($@) {3671die"An error occurred during merge-base"3672unless$@->isa("Git::Error::Command");36733674warn"W: Cannot find common ancestor between ".3675"@$parentsand$merge_tip. Ignoring merge info.\n";3676next;3677}36783679# double check that there are no missing non-merge commits3680my(@incomplete) = check_cherry_pick(3681$merge_base,$merge_tip,3682$parents,3683@$ranges,3684);36853686if(@incomplete) {3687warn"W:svn cherry-pick ignored ($spec) - missing "3688.@incomplete." commit(s) (eg$incomplete[0])\n";3689}else{3690warn3691"Found merge parent (svn:mergeinfo prop): ",3692$merge_tip,"\n";3693push@new_parents,$merge_tip;3694}3695}36963697# cater for merges which merge commits from multiple branches3698if(@new_parents>1) {3699for(my$i=0;$i<=$#new_parents;$i++) {3700for(my$j=0;$j<=$#new_parents;$j++) {3701next if$i==$j;3702next unless$new_parents[$i];3703next unless$new_parents[$j];3704my$revs= command_oneline(3705"rev-list","-1",3706"$new_parents[$i]..$new_parents[$j]",3707);3708if( !$revs) {3709undef($new_parents[$j]);3710}3711}3712}3713}3714push@$parents,grep{defined}@new_parents;3715}37163717sub make_log_entry {3718my($self,$rev,$parents,$ed) =@_;3719my$untracked=$self->get_untracked($ed);37203721my@parents=@$parents;3722my$ps=$ed->{path_strip} ||"";3723formy$path(grep{m/$ps/} %{$ed->{dir_prop}} ) {3724my$props=$ed->{dir_prop}{$path};3725if($props->{"svk:merge"} ) {3726$self->find_extra_svk_parents3727($ed,$props->{"svk:merge"}, \@parents);3728}3729if($props->{"svn:mergeinfo"} ) {3730$self->find_extra_svn_parents3731($ed,3732$props->{"svn:mergeinfo"},3733 \@parents);3734}3735}37363737open my$un,'>>',"$self->{dir}/unhandled.log"or croak $!;3738print$un"r$rev\n"or croak $!;3739print$un $_,"\n"foreach@$untracked;3740my%log_entry= ( parents => \@parents, revision =>$rev,3741log=>'');37423743my$headrev;3744my$logged=delete$self->{logged_rev_props};3745if(!$logged||$self->{-want_revprops}) {3746my$rp=$self->ra->rev_proplist($rev);3747foreach(sort keys%$rp) {3748my$v=$rp->{$_};3749if(/^svn:(author|date|log)$/) {3750$log_entry{$1} =$v;3751}elsif($_eq'svm:headrev') {3752$headrev=$v;3753}else{3754print$un" rev_prop: ", uri_encode($_),' ',3755 uri_encode($v),"\n";3756}3757}3758}else{3759map{$log_entry{$_} =$logged->{$_} }keys%$logged;3760}3761close$unor croak $!;37623763$log_entry{date} = parse_svn_date($log_entry{date});3764$log_entry{log} .="\n";3765my$author=$log_entry{author} = check_author($log_entry{author});3766my($name,$email) =defined$::users{$author} ? @{$::users{$author}}3767: ($author,undef);37683769my($commit_name,$commit_email) = ($name,$email);3770if($_use_log_author) {3771my$name_field;3772if($log_entry{log} =~/From:\s+(.*\S)\s*\n/i) {3773$name_field=$1;3774}elsif($log_entry{log} =~/Signed-off-by:\s+(.*\S)\s*\n/i) {3775$name_field=$1;3776}3777if(!defined$name_field) {3778if(!defined$email) {3779$email=$name;3780}3781}elsif($name_field=~/(.*?)\s+<(.*)>/) {3782($name,$email) = ($1,$2);3783}elsif($name_field=~/(.*)@/) {3784($name,$email) = ($1,$name_field);3785}else{3786($name,$email) = ($name_field,$name_field);3787}3788}3789if(defined$headrev&&$self->use_svm_props) {3790if($self->rewrite_root) {3791die"Can't have both 'useSvmProps' and 'rewriteRoot' ",3792"options set!\n";3793}3794if($self->rewrite_uuid) {3795die"Can't have both 'useSvmProps' and 'rewriteUUID' ",3796"options set!\n";3797}3798my($uuid,$r) =$headrev=~m{^([a-f\d\-]{30,}):(\d+)$}i;3799# we don't want "SVM: initializing mirror for junk" ...3800returnundefif$r==0;3801my$svm=$self->svm;3802if($uuidne$svm->{uuid}) {3803die"UUID mismatch on SVM path:\n",3804"expected:$svm->{uuid}\n",3805" got:$uuid\n";3806}3807my$full_url=$self->full_url;3808$full_url=~ s#^\Q$svm->{replace}\E(/|$)#$svm->{source}$1# or3809die"Failed to replace '$svm->{replace}' with ",3810"'$svm->{source}' in$full_url\n";3811# throw away username for storing in records3812 remove_username($full_url);3813$log_entry{metadata} ="$full_url\@$r$uuid";3814$log_entry{svm_revision} =$r;3815$email||="$author\@$uuid";3816$commit_email||="$author\@$uuid";3817}elsif($self->use_svnsync_props) {3818my$full_url=$self->svnsync->{url};3819$full_url.="/$self->{path}"iflength$self->{path};3820 remove_username($full_url);3821my$uuid=$self->svnsync->{uuid};3822$log_entry{metadata} ="$full_url\@$rev$uuid";3823$email||="$author\@$uuid";3824$commit_email||="$author\@$uuid";3825}else{3826my$url=$self->metadata_url;3827 remove_username($url);3828my$uuid=$self->rewrite_uuid||$self->ra->get_uuid;3829$log_entry{metadata} ="$url\@$rev".$uuid;3830$email||="$author\@".$uuid;3831$commit_email||="$author\@".$uuid;3832}3833$log_entry{name} =$name;3834$log_entry{email} =$email;3835$log_entry{commit_name} =$commit_name;3836$log_entry{commit_email} =$commit_email;3837 \%log_entry;3838}38393840sub fetch {3841my($self,$min_rev,$max_rev,@parents) =@_;3842my($last_rev,$last_commit) =$self->last_rev_commit;3843my($base,$head) =$self->get_fetch_range($min_rev,$max_rev);3844$self->ra->gs_fetch_loop_common($base,$head, [$self]);3845}38463847sub set_tree_cb {3848my($self,$log_entry,$tree,$rev,$date,$author) =@_;3849$self->{inject_parents} = {$rev=>$tree};3850$self->fetch(undef,undef);3851}38523853sub set_tree {3854my($self,$tree) = (shift,shift);3855my$log_entry= ::get_commit_entry($tree);3856unless($self->{last_rev}) {3857::fatal("Must have an existing revision to commit");3858}3859my%ed_opts= ( r =>$self->{last_rev},3860log=>$log_entry->{log},3861 ra =>$self->ra,3862 tree_a =>$self->{last_commit},3863 tree_b =>$tree,3864 editor_cb =>sub{3865$self->set_tree_cb($log_entry,$tree,@_) },3866 svn_path =>$self->{path} );3867if(!SVN::Git::Editor->new(\%ed_opts)->apply_diff) {3868print"No changes\nr$self->{last_rev} =$tree\n";3869}3870}38713872sub rebuild_from_rev_db {3873my($self,$path) =@_;3874my$r= -1;3875open my$fh,'<',$pathor croak "open:$!";3876binmode$fhor croak "binmode:$!";3877while(<$fh>) {3878length($_) ==41or croak "inconsistent size in ($_) != 41";3879chomp($_);3880++$r;3881next if$_eq('0' x 40);3882$self->rev_map_set($r,$_);3883print"r$r=$_\n";3884}3885close$fhor croak "close:$!";3886unlink$pathor croak "unlink:$!";3887}38883889sub rebuild {3890my($self) =@_;3891my$map_path=$self->map_path;3892my$partial= (-e $map_path&& ! -z $map_path);3893return unless::verify_ref($self->refname.'^0');3894if(!$partial&& ($self->use_svm_props||$self->no_metadata)) {3895my$rev_db=$self->rev_db_path;3896$self->rebuild_from_rev_db($rev_db);3897if($self->use_svm_props) {3898my$svm_rev_db=$self->rev_db_path($self->svm_uuid);3899$self->rebuild_from_rev_db($svm_rev_db);3900}3901$self->unlink_rev_db_symlink;3902return;3903}3904print"Rebuilding$map_path...\n"if(!$partial);3905my($base_rev,$head) = ($partial?$self->rev_map_max_norebuild(1) :3906(undef,undef));3907my($log,$ctx) =3908 command_output_pipe(qw/rev-list --pretty=raw --no-color --reverse/,3909($head?"$head..":"") .$self->refname,3910'--');3911my$metadata_url=$self->metadata_url;3912 remove_username($metadata_url);3913my$svn_uuid=$self->rewrite_uuid||$self->ra_uuid;3914my$c;3915while(<$log>) {3916if(m{^commit ($::sha1)$}) {3917$c=$1;3918next;3919}3920next unlesss{^\s*(git-svn-id:)}{$1};3921my($url,$rev,$uuid) = ::extract_metadata($_);3922 remove_username($url);39233924# ignore merges (from set-tree)3925next if(!defined$rev|| !$uuid);39263927# if we merged or otherwise started elsewhere, this is3928# how we break out of it3929if(($uuidne$svn_uuid) ||3930($metadata_url&&$url&& ($urlne$metadata_url))) {3931next;3932}3933if($partial&&$head) {3934print"Partial-rebuilding$map_path...\n";3935print"Currently at$base_rev=$head\n";3936$head=undef;3937}39383939$self->rev_map_set($rev,$c);3940print"r$rev=$c\n";3941}3942 command_close_pipe($log,$ctx);3943print"Done rebuilding$map_path\n"if(!$partial|| !$head);3944my$rev_db_path=$self->rev_db_path;3945if(-f $self->rev_db_path) {3946unlink$self->rev_db_pathor croak "unlink:$!";3947}3948$self->unlink_rev_db_symlink;3949}39503951# rev_map:3952# Tie::File seems to be prone to offset errors if revisions get sparse,3953# it's not that fast, either. Tie::File is also not in Perl 5.6. So3954# one of my favorite modules is out :< Next up would be one of the DBM3955# modules, but I'm not sure which is most portable...3956#3957# This is the replacement for the rev_db format, which was too big3958# and inefficient for large repositories with a lot of sparse history3959# (mainly tags)3960#3961# The format is this:3962# - 24 bytes for every record,3963# * 4 bytes for the integer representing an SVN revision number3964# * 20 bytes representing the sha1 of a git commit3965# - No empty padding records like the old format3966# (except the last record, which can be overwritten)3967# - new records are written append-only since SVN revision numbers3968# increase monotonically3969# - lookups on SVN revision number are done via a binary search3970# - Piping the file to xxd -c24 is a good way of dumping it for3971# viewing or editing (piped back through xxd -r), should the need3972# ever arise.3973# - The last record can be padding revision with an all-zero sha13974# This is used to optimize fetch performance when using multiple3975# "fetch" directives in .git/config3976#3977# These files are disposable unless noMetadata or useSvmProps is set39783979sub _rev_map_set {3980my($fh,$rev,$commit) =@_;39813982binmode$fhor croak "binmode:$!";3983my$size= (stat($fh))[7];3984($size%24) ==0or croak "inconsistent size:$size";39853986my$wr_offset=0;3987if($size>0) {3988sysseek($fh, -24, SEEK_END)or croak "seek:$!";3989my$read=sysread($fh,my$buf,24)or croak "read:$!";3990$read==24or croak "read only$readbytes (!= 24)";3991my($last_rev,$last_commit) =unpack(rev_map_fmt,$buf);3992if($last_commiteq('0' x40)) {3993if($size>=48) {3994sysseek($fh, -48, SEEK_END)or croak "seek:$!";3995$read=sysread($fh,$buf,24)or3996 croak "read:$!";3997$read==24or3998 croak "read only$readbytes (!= 24)";3999($last_rev,$last_commit) =4000unpack(rev_map_fmt,$buf);4001if($last_commiteq('0' x40)) {4002 croak "inconsistent .rev_map\n";4003}4004}4005if($last_rev>=$rev) {4006 croak "last_rev is higher!:$last_rev>=$rev";4007}4008$wr_offset= -24;4009}4010}4011sysseek($fh,$wr_offset, SEEK_END)or croak "seek:$!";4012syswrite($fh,pack(rev_map_fmt,$rev,$commit),24) ==24or4013 croak "write:$!";4014}40154016sub _rev_map_reset {4017my($fh,$rev,$commit) =@_;4018my$c= _rev_map_get($fh,$rev);4019$ceq$commitor die"_rev_map_reset(@_) commit$cdoes not match!\n";4020my$offset=sysseek($fh,0, SEEK_CUR)or croak "seek:$!";4021truncate$fh,$offsetor croak "truncate:$!";4022}40234024sub mkfile {4025my($path) =@_;4026unless(-e $path) {4027my($dir,$base) = ($path=~ m#^(.*?)/?([^/]+)$#);4028 mkpath([$dir])unless-d $dir;4029open my$fh,'>>',$pathor die"Couldn't create$path:$!\n";4030close$fhor die"Couldn't close (create)$path:$!\n";4031}4032}40334034sub rev_map_set {4035my($self,$rev,$commit,$update_ref,$uuid) =@_;4036defined$commitor die"missing arg3\n";4037length$commit==40or die"arg3 must be a full SHA1 hexsum\n";4038my$db=$self->map_path($uuid);4039my$db_lock="$db.lock";4040my$sig;4041$update_ref||=0;4042if($update_ref) {4043$SIG{INT} =$SIG{HUP} =$SIG{TERM} =$SIG{ALRM} =$SIG{PIPE} =4044$SIG{USR1} =$SIG{USR2} =sub{$sig=$_[0] };4045}4046 mkfile($db);40474048$LOCKFILES{$db_lock} =1;4049my$sync;4050# both of these options make our .rev_db file very, very important4051# and we can't afford to lose it because rebuild() won't work4052if($self->use_svm_props||$self->no_metadata) {4053$sync=1;4054 copy($db,$db_lock)or die"rev_map_set(@_): ",4055"Failed to copy: ",4056"$db=>$db_lock($!)\n";4057}else{4058rename$db,$db_lockor die"rev_map_set(@_): ",4059"Failed to rename: ",4060"$db=>$db_lock($!)\n";4061}40624063sysopen(my$fh,$db_lock, O_RDWR | O_CREAT)4064or croak "Couldn't open$db_lock:$!\n";4065$update_refeq'reset'? _rev_map_reset($fh,$rev,$commit) :4066 _rev_map_set($fh,$rev,$commit);4067if($sync) {4068$fh->flushor die"Couldn't flush$db_lock:$!\n";4069$fh->syncor die"Couldn't sync$db_lock:$!\n";4070}4071close$fhor croak $!;4072if($update_ref) {4073$_head=$self;4074my$note="";4075$note=" ($update_ref)"if($update_ref!~/^\d*$/);4076 command_noisy('update-ref','-m',"r$rev$note",4077$self->refname,$commit);4078}4079rename$db_lock,$dbor die"rev_map_set(@_): ","Failed to rename: ",4080"$db_lock=>$db($!)\n";4081delete$LOCKFILES{$db_lock};4082if($update_ref) {4083$SIG{INT} =$SIG{HUP} =$SIG{TERM} =$SIG{ALRM} =$SIG{PIPE} =4084$SIG{USR1} =$SIG{USR2} ='DEFAULT';4085kill$sig,$$ifdefined$sig;4086}4087}40884089# If want_commit, this will return an array of (rev, commit) where4090# commit _must_ be a valid commit in the archive.4091# Otherwise, it'll return the max revision (whether or not the4092# commit is valid or just a 0x40 placeholder).4093sub rev_map_max {4094my($self,$want_commit) =@_;4095$self->rebuild;4096my($r,$c) =$self->rev_map_max_norebuild($want_commit);4097$want_commit? ($r,$c) :$r;4098}40994100sub rev_map_max_norebuild {4101my($self,$want_commit) =@_;4102my$map_path=$self->map_path;4103stat$map_pathorreturn$want_commit? (0,undef) :0;4104sysopen(my$fh,$map_path, O_RDONLY)or croak "open:$!";4105binmode$fhor croak "binmode:$!";4106my$size= (stat($fh))[7];4107($size%24) ==0or croak "inconsistent size:$size";41084109if($size==0) {4110close$fhor croak "close:$!";4111return$want_commit? (0,undef) :0;4112}41134114sysseek($fh, -24, SEEK_END)or croak "seek:$!";4115sysread($fh,my$buf,24) ==24or croak "read:$!";4116my($r,$c) =unpack(rev_map_fmt,$buf);4117if($want_commit&&$ceq('0' x40)) {4118if($size<48) {4119return$want_commit? (0,undef) :0;4120}4121sysseek($fh, -48, SEEK_END)or croak "seek:$!";4122sysread($fh,$buf,24) ==24or croak "read:$!";4123($r,$c) =unpack(rev_map_fmt,$buf);4124if($ceq('0'x40)) {4125 croak "Penultimate record is all-zeroes in$map_path";4126}4127}4128close$fhor croak "close:$!";4129$want_commit? ($r,$c) :$r;4130}41314132sub rev_map_get {4133my($self,$rev,$uuid) =@_;4134my$map_path=$self->map_path($uuid);4135returnundefunless-e $map_path;41364137sysopen(my$fh,$map_path, O_RDONLY)or croak "open:$!";4138my$c= _rev_map_get($fh,$rev);4139close($fh)or croak "close:$!";4140$c4141}41424143sub _rev_map_get {4144my($fh,$rev) =@_;41454146binmode$fhor croak "binmode:$!";4147my$size= (stat($fh))[7];4148($size%24) ==0or croak "inconsistent size:$size";41494150if($size==0) {4151returnundef;4152}41534154my($l,$u) = (0,$size-24);4155my($r,$c,$buf);41564157while($l<=$u) {4158my$i=int(($l/24+$u/24) /2) *24;4159sysseek($fh,$i, SEEK_SET)or croak "seek:$!";4160sysread($fh,my$buf,24) ==24or croak "read:$!";4161my($r,$c) =unpack(rev_map_fmt,$buf);41624163if($r<$rev) {4164$l=$i+24;4165}elsif($r>$rev) {4166$u=$i-24;4167}else{# $r == $rev4168return$ceq('0' x 40) ?undef:$c;4169}4170}4171undef;4172}41734174# Finds the first svn revision that exists on (if $eq_ok is true) or4175# before $rev for the current branch. It will not search any lower4176# than $min_rev. Returns the git commit hash and svn revision number4177# if found, else (undef, undef).4178sub find_rev_before {4179my($self,$rev,$eq_ok,$min_rev) =@_;4180--$revunless$eq_ok;4181$min_rev||=1;4182my$max_rev=$self->rev_map_max;4183$rev=$max_revif($rev>$max_rev);4184while($rev>=$min_rev) {4185if(my$c=$self->rev_map_get($rev)) {4186return($rev,$c);4187}4188--$rev;4189}4190return(undef,undef);4191}41924193# Finds the first svn revision that exists on (if $eq_ok is true) or4194# after $rev for the current branch. It will not search any higher4195# than $max_rev. Returns the git commit hash and svn revision number4196# if found, else (undef, undef).4197sub find_rev_after {4198my($self,$rev,$eq_ok,$max_rev) =@_;4199++$revunless$eq_ok;4200$max_rev||=$self->rev_map_max;4201while($rev<=$max_rev) {4202if(my$c=$self->rev_map_get($rev)) {4203return($rev,$c);4204}4205++$rev;4206}4207return(undef,undef);4208}42094210sub _new {4211my($class,$repo_id,$ref_id,$path) =@_;4212unless(defined$repo_id&&length$repo_id) {4213$repo_id=$Git::SVN::default_repo_id;4214}4215unless(defined$ref_id&&length$ref_id) {4216$_prefix=''unlessdefined($_prefix);4217$_[2] =$ref_id=4218"refs/remotes/$_prefix$Git::SVN::default_ref_id";4219}4220$_[1] =$repo_id;4221my$dir="$ENV{GIT_DIR}/svn/$ref_id";42224223# Older repos imported by us used $GIT_DIR/svn/foo instead of4224# $GIT_DIR/svn/refs/remotes/foo when tracking refs/remotes/foo4225if($ref_id=~m{^refs/remotes/(.*)}) {4226my$old_dir="$ENV{GIT_DIR}/svn/$1";4227if(-d $old_dir&& ! -d $dir) {4228$dir=$old_dir;4229}4230}42314232$_[3] =$path=''unless(defined$path);4233 mkpath([$dir]);4234bless{4235 ref_id =>$ref_id, dir =>$dir,index=>"$dir/index",4236 path =>$path, config =>"$ENV{GIT_DIR}/svn/config",4237 map_root =>"$dir/.rev_map", repo_id =>$repo_id},$class;4238}42394240# for read-only access of old .rev_db formats4241sub unlink_rev_db_symlink {4242my($self) =@_;4243my$link=$self->rev_db_path;4244$link=~s/\.[\w-]+$//or croak "missing UUID at the end of$link";4245if(-l $link) {4246unlink$linkor croak "unlink:$linkfailed!";4247}4248}42494250sub rev_db_path {4251my($self,$uuid) =@_;4252my$db_path=$self->map_path($uuid);4253$db_path=~s{/\.rev_map\.}{/\.rev_db\.}4254or croak "map_path:$db_pathdoes not contain '/.rev_map.' !";4255$db_path;4256}42574258# the new replacement for .rev_db4259sub map_path {4260my($self,$uuid) =@_;4261$uuid||=$self->ra_uuid;4262"$self->{map_root}.$uuid";4263}42644265sub uri_encode {4266my($f) =@_;4267$f=~ s#([^a-zA-Z0-9\*!\:_\./\-])#uc sprintf("%%%02x",ord($1))#eg;4268$f4269}42704271sub uri_decode {4272my($f) =@_;4273$f=~ s#%([0-9a-fA-F]{2})#chr(hex($1))#eg;4274$f4275}42764277sub remove_username {4278$_[0] =~s{^([^:]*://)[^@]+@}{$1};4279}42804281package Git::SVN::Prompt;4282use strict;4283use warnings;4284require SVN::Core;4285use vars qw/$_no_auth_cache $_username/;42864287sub simple {4288my($cred,$realm,$default_username,$may_save,$pool) =@_;4289$may_save=undefif$_no_auth_cache;4290$default_username=$_usernameifdefined$_username;4291if(defined$default_username&&length$default_username) {4292if(defined$realm&&length$realm) {4293print STDERR "Authentication realm:$realm\n";4294 STDERR->flush;4295}4296$cred->username($default_username);4297}else{4298 username($cred,$realm,$may_save,$pool);4299}4300$cred->password(_read_password("Password for '".4301$cred->username."': ",$realm));4302$cred->may_save($may_save);4303$SVN::_Core::SVN_NO_ERROR;4304}43054306sub ssl_server_trust {4307my($cred,$realm,$failures,$cert_info,$may_save,$pool) =@_;4308$may_save=undefif$_no_auth_cache;4309print STDERR "Error validating server certificate for '$realm':\n";4310{4311no warnings 'once';4312# All variables SVN::Auth::SSL::* are used only once,4313# so we're shutting up Perl warnings about this.4314if($failures&$SVN::Auth::SSL::UNKNOWNCA) {4315print STDERR " - The certificate is not issued ",4316"by a trusted authority. Use the\n",4317" fingerprint to validate ",4318"the certificate manually!\n";4319}4320if($failures&$SVN::Auth::SSL::CNMISMATCH) {4321print STDERR " - The certificate hostname ",4322"does not match.\n";4323}4324if($failures&$SVN::Auth::SSL::NOTYETVALID) {4325print STDERR " - The certificate is not yet valid.\n";4326}4327if($failures&$SVN::Auth::SSL::EXPIRED) {4328print STDERR " - The certificate has expired.\n";4329}4330if($failures&$SVN::Auth::SSL::OTHER) {4331print STDERR " - The certificate has ",4332"an unknown error.\n";4333}4334}# no warnings 'once'4335printf STDERR4336"Certificate information:\n".4337" - Hostname:%s\n".4338" - Valid: from%suntil%s\n".4339" - Issuer:%s\n".4340" - Fingerprint:%s\n",4341map$cert_info->$_,qw(hostname valid_from valid_until4342 issuer_dname fingerprint);4343my$choice;4344prompt:4345print STDERR $may_save?4346"(R)eject, accept (t)emporarily or accept (p)ermanently? ":4347"(R)eject or accept (t)emporarily? ";4348 STDERR->flush;4349$choice=lc(substr(<STDIN> ||'R',0,1));4350if($choice=~/^t$/i) {4351$cred->may_save(undef);4352}elsif($choice=~/^r$/i) {4353return-1;4354}elsif($may_save&&$choice=~/^p$/i) {4355$cred->may_save($may_save);4356}else{4357goto prompt;4358}4359$cred->accepted_failures($failures);4360$SVN::_Core::SVN_NO_ERROR;4361}43624363sub ssl_client_cert {4364my($cred,$realm,$may_save,$pool) =@_;4365$may_save=undefif$_no_auth_cache;4366print STDERR "Client certificate filename: ";4367 STDERR->flush;4368chomp(my$filename= <STDIN>);4369$cred->cert_file($filename);4370$cred->may_save($may_save);4371$SVN::_Core::SVN_NO_ERROR;4372}43734374sub ssl_client_cert_pw {4375my($cred,$realm,$may_save,$pool) =@_;4376$may_save=undefif$_no_auth_cache;4377$cred->password(_read_password("Password: ",$realm));4378$cred->may_save($may_save);4379$SVN::_Core::SVN_NO_ERROR;4380}43814382sub username {4383my($cred,$realm,$may_save,$pool) =@_;4384$may_save=undefif$_no_auth_cache;4385if(defined$realm&&length$realm) {4386print STDERR "Authentication realm:$realm\n";4387}4388my$username;4389if(defined$_username) {4390$username=$_username;4391}else{4392print STDERR "Username: ";4393 STDERR->flush;4394chomp($username= <STDIN>);4395}4396$cred->username($username);4397$cred->may_save($may_save);4398$SVN::_Core::SVN_NO_ERROR;4399}44004401sub _read_password {4402my($prompt,$realm) =@_;4403my$password='';4404if(exists$ENV{GIT_ASKPASS}) {4405open(PH,"-|",$ENV{GIT_ASKPASS},$prompt);4406$password= <PH>;4407$password=~s/[\012\015]//;# \n\r4408close(PH);4409}else{4410print STDERR $prompt;4411 STDERR->flush;4412require Term::ReadKey;4413 Term::ReadKey::ReadMode('noecho');4414while(defined(my$key= Term::ReadKey::ReadKey(0))) {4415last if$key=~/[\012\015]/;# \n\r4416$password.=$key;4417}4418 Term::ReadKey::ReadMode('restore');4419print STDERR "\n";4420 STDERR->flush;4421}4422$password;4423}44244425package SVN::Git::Fetcher;4426use vars qw/@ISA $_ignore_regex $_preserve_empty_dirs $_placeholder_filename4427@deleted_gpath %added_placeholder $repo_id/;4428use strict;4429use warnings;4430use Carp qw/croak/;4431use File::Basename qw/dirname/;4432use IO::File qw//;44334434# file baton members: path, mode_a, mode_b, pool, fh, blob, base4435sub new {4436my($class,$git_svn,$switch_path) =@_;4437my$self= SVN::Delta::Editor->new;4438bless$self,$class;4439if(exists$git_svn->{last_commit}) {4440$self->{c} =$git_svn->{last_commit};4441$self->{empty_symlinks} =4442 _mark_empty_symlinks($git_svn,$switch_path);4443}44444445# some options are read globally, but can be overridden locally4446# per [svn-remote "..."] section. Command-line options will *NOT*4447# override options set in an [svn-remote "..."] section4448$repo_id=$git_svn->{repo_id};4449my$k="svn-remote.$repo_id.ignore-paths";4450my$v=eval{ command_oneline('config','--get',$k) };4451$self->{ignore_regex} =$v;44524453$k="svn-remote.$repo_id.preserve-empty-dirs";4454$v=eval{ command_oneline('config','--get','--bool',$k) };4455if($v&&$veq'true') {4456$_preserve_empty_dirs=1;4457$k="svn-remote.$repo_id.placeholder-filename";4458$v=eval{ command_oneline('config','--get',$k) };4459$_placeholder_filename=$v;4460}44614462# Load the list of placeholder files added during previous invocations.4463$k="svn-remote.$repo_id.added-placeholder";4464$v=eval{ command_oneline('config','--get-all',$k) };4465if($_preserve_empty_dirs&&$v) {4466# command() prints errors to stderr, so we only call it if4467# command_oneline() succeeded.4468my@v= command('config','--get-all',$k);4469$added_placeholder{ dirname($_) } =$_foreach@v;4470}44714472$self->{empty} = {};4473$self->{dir_prop} = {};4474$self->{file_prop} = {};4475$self->{absent_dir} = {};4476$self->{absent_file} = {};4477$self->{gii} =$git_svn->tmp_index_do(sub{ Git::IndexInfo->new});4478$self->{pathnameencoding} = Git::config('svn.pathnameencoding');4479$self;4480}44814482# this uses the Ra object, so it must be called before do_{switch,update},4483# not inside them (when the Git::SVN::Fetcher object is passed) to4484# do_{switch,update}4485sub _mark_empty_symlinks {4486my($git_svn,$switch_path) =@_;4487my$bool= Git::config_bool('svn.brokenSymlinkWorkaround');4488return{}if(!defined($bool)) || (defined($bool) && !$bool);44894490my%ret;4491my($rev,$cmt) =$git_svn->last_rev_commit;4492return{}unless($rev&&$cmt);44934494# allow the warning to be printed for each revision we fetch to4495# ensure the user sees it. The user can also disable the workaround4496# on the repository even while git svn is running and the next4497# revision fetched will skip this expensive function.4498my$printed_warning;4499chomp(my$empty_blob=`git hash-object -t blob --stdin < /dev/null`);4500my($ls,$ctx) = command_output_pipe(qw/ls-tree -r -z/,$cmt);4501local$/="\0";4502my$pfx=defined($switch_path) ?$switch_path:$git_svn->{path};4503$pfx.='/'iflength($pfx);4504while(<$ls>) {4505chomp;4506s/\A100644 blob $empty_blob\t//oornext;4507unless($printed_warning) {4508print STDERR "Scanning for empty symlinks, ",4509"this may take a while if you have ",4510"many empty files\n",4511"You may disable this with `",4512"git config svn.brokenSymlinkWorkaround ",4513"false'.\n",4514"This may be done in a different ",4515"terminal without restarting ",4516"git svn\n";4517$printed_warning=1;4518}4519my$path=$_;4520my(undef,$props) =4521$git_svn->ra->get_file($pfx.$path,$rev,undef);4522if($props->{'svn:special'}) {4523$ret{$path} =1;4524}4525}4526 command_close_pipe($ls,$ctx);4527 \%ret;4528}45294530# returns true if a given path is inside a ".git" directory4531sub in_dot_git {4532$_[0] =~m{(?:^|/)\.git(?:/|$)};4533}45344535# return value: 0 -- don't ignore, 1 -- ignore4536sub is_path_ignored {4537my($self,$path) =@_;4538return1if in_dot_git($path);4539return1ifdefined($self->{ignore_regex}) &&4540$path=~m!$self->{ignore_regex}!;4541return0unlessdefined($_ignore_regex);4542return1if$path=~m!$_ignore_regex!o;4543return0;4544}45454546sub set_path_strip {4547my($self,$path) =@_;4548$self->{path_strip} =qr/^\Q$path\E(\/|$)/iflength$path;4549}45504551sub open_root {4552{ path =>''};4553}45544555sub open_directory {4556my($self,$path,$pb,$rev) =@_;4557{ path =>$path};4558}45594560sub git_path {4561my($self,$path) =@_;4562if(my$enc=$self->{pathnameencoding}) {4563require Encode;4564 Encode::from_to($path,'UTF-8',$enc);4565}4566if($self->{path_strip}) {4567$path=~s!$self->{path_strip}!!or4568die"Failed to strip path '$path' ($self->{path_strip})\n";4569}4570$path;4571}45724573sub delete_entry {4574my($self,$path,$rev,$pb) =@_;4575returnundefif$self->is_path_ignored($path);45764577my$gpath=$self->git_path($path);4578returnundefif($gpatheq'');45794580# remove entire directories.4581my($tree) = (command('ls-tree','-z',$self->{c},"./$gpath")4582=~/\A040000 tree ([a-f\d]{40})\t\Q$gpath\E\0/);4583if($tree) {4584my($ls,$ctx) = command_output_pipe(qw/ls-tree4585-r --name-only -z/,4586$tree);4587local$/="\0";4588while(<$ls>) {4589chomp;4590my$rmpath="$gpath/$_";4591$self->{gii}->remove($rmpath);4592print"\tD\t$rmpath\n"unless$::_q;4593}4594print"\tD\t$gpath/\n"unless$::_q;4595 command_close_pipe($ls,$ctx);4596}else{4597$self->{gii}->remove($gpath);4598print"\tD\t$gpath\n"unless$::_q;4599}4600# Don't add to @deleted_gpath if we're deleting a placeholder file.4601push@deleted_gpath,$gpathunless$added_placeholder{dirname($path)};4602$self->{empty}->{$path} =0;4603undef;4604}46054606sub open_file {4607my($self,$path,$pb,$rev) =@_;4608my($mode,$blob);46094610goto out if$self->is_path_ignored($path);46114612my$gpath=$self->git_path($path);4613($mode,$blob) = (command('ls-tree','-z',$self->{c},"./$gpath")4614=~/\A(\d{6}) blob ([a-f\d]{40})\t\Q$gpath\E\0/);4615unless(defined$mode&&defined$blob) {4616die"$pathwas not found in commit$self->{c} (r$rev)\n";4617}4618if($modeeq'100644'&&$self->{empty_symlinks}->{$path}) {4619$mode='120000';4620}4621out:4622{ path =>$path, mode_a =>$mode, mode_b =>$mode, blob =>$blob,4623 pool => SVN::Pool->new, action =>'M'};4624}46254626sub add_file {4627my($self,$path,$pb,$cp_path,$cp_rev) =@_;4628my$mode;46294630if(!$self->is_path_ignored($path)) {4631my($dir,$file) = ($path=~ m#^(.*?)/?([^/]+)$#);4632delete$self->{empty}->{$dir};4633$mode='100644';46344635if($added_placeholder{$dir}) {4636# Remove our placeholder file, if we created one.4637 delete_entry($self,$added_placeholder{$dir})4638unless$patheq$added_placeholder{$dir};4639delete$added_placeholder{$dir}4640}4641}46424643{ path =>$path, mode_a =>$mode, mode_b =>$mode,4644 pool => SVN::Pool->new, action =>'A'};4645}46464647sub add_directory {4648my($self,$path,$cp_path,$cp_rev) =@_;4649goto out if$self->is_path_ignored($path);4650my$gpath=$self->git_path($path);4651if($gpatheq'') {4652my($ls,$ctx) = command_output_pipe(qw/ls-tree4653-r --name-only -z/,4654$self->{c});4655local$/="\0";4656while(<$ls>) {4657chomp;4658$self->{gii}->remove($_);4659print"\tD\t$_\n"unless$::_q;4660push@deleted_gpath,$gpath;4661}4662 command_close_pipe($ls,$ctx);4663$self->{empty}->{$path} =0;4664}4665my($dir,$file) = ($path=~ m#^(.*?)/?([^/]+)$#);4666delete$self->{empty}->{$dir};4667$self->{empty}->{$path} =1;46684669if($added_placeholder{$dir}) {4670# Remove our placeholder file, if we created one.4671 delete_entry($self,$added_placeholder{$dir});4672delete$added_placeholder{$dir}4673}46744675out:4676{ path =>$path};4677}46784679sub change_dir_prop {4680my($self,$db,$prop,$value) =@_;4681returnundefif$self->is_path_ignored($db->{path});4682$self->{dir_prop}->{$db->{path}} ||= {};4683$self->{dir_prop}->{$db->{path}}->{$prop} =$value;4684undef;4685}46864687sub absent_directory {4688my($self,$path,$pb) =@_;4689returnundefif$self->is_path_ignored($path);4690$self->{absent_dir}->{$pb->{path}} ||= [];4691push@{$self->{absent_dir}->{$pb->{path}}},$path;4692undef;4693}46944695sub absent_file {4696my($self,$path,$pb) =@_;4697returnundefif$self->is_path_ignored($path);4698$self->{absent_file}->{$pb->{path}} ||= [];4699push@{$self->{absent_file}->{$pb->{path}}},$path;4700undef;4701}47024703sub change_file_prop {4704my($self,$fb,$prop,$value) =@_;4705returnundefif$self->is_path_ignored($fb->{path});4706if($propeq'svn:executable') {4707if($fb->{mode_b} !=120000) {4708$fb->{mode_b} =defined$value?100755:100644;4709}4710}elsif($propeq'svn:special') {4711$fb->{mode_b} =defined$value?120000:100644;4712}else{4713$self->{file_prop}->{$fb->{path}} ||= {};4714$self->{file_prop}->{$fb->{path}}->{$prop} =$value;4715}4716undef;4717}47184719sub apply_textdelta {4720my($self,$fb,$exp) =@_;4721returnundefif$self->is_path_ignored($fb->{path});4722my$fh= $::_repository->temp_acquire('svn_delta');4723# $fh gets auto-closed() by SVN::TxDelta::apply(),4724# (but $base does not,) so dup() it for reading in close_file4725open my$dup,'<&',$fhor croak $!;4726my$base= $::_repository->temp_acquire('git_blob');47274728if($fb->{blob}) {4729my($base_is_link,$size);47304731if($fb->{mode_a}eq'120000'&&4732!$self->{empty_symlinks}->{$fb->{path}}) {4733print$base'link 'or die"print$!\n";4734$base_is_link=1;4735}4736 retry:4737$size= $::_repository->cat_blob($fb->{blob},$base);4738die"Failed to read object$fb->{blob}"if($size<0);47394740if(defined$exp) {4741seek$base,0,0or croak $!;4742my$got= ::md5sum($base);4743if($gotne$exp) {4744my$err="Checksum mismatch: ".4745"$fb->{path}$fb->{blob}\n".4746"expected:$exp\n".4747" got:$got\n";4748if($base_is_link) {4749warn$err,4750"Retrying... (possibly ",4751"a bad symlink from SVN)\n";4752$::_repository->temp_reset($base);4753$base_is_link=0;4754goto retry;4755}4756die$err;4757}4758}4759}4760seek$base,0,0or croak $!;4761$fb->{fh} =$fh;4762$fb->{base} =$base;4763[ SVN::TxDelta::apply($base,$dup,undef,$fb->{path},$fb->{pool}) ];4764}47654766sub close_file {4767my($self,$fb,$exp) =@_;4768returnundefif$self->is_path_ignored($fb->{path});47694770my$hash;4771my$path=$self->git_path($fb->{path});4772if(my$fh=$fb->{fh}) {4773if(defined$exp) {4774seek($fh,0,0)or croak $!;4775my$got= ::md5sum($fh);4776if($gotne$exp) {4777die"Checksum mismatch:$path\n",4778"expected:$exp\ngot:$got\n";4779}4780}4781if($fb->{mode_b} ==120000) {4782sysseek($fh,0,0)or croak $!;4783my$rd=sysread($fh,my$buf,5);47844785if(!defined$rd) {4786 croak "sysread:$!\n";4787}elsif($rd==0) {4788warn"$pathhas mode 120000",4789" but it points to nothing\n",4790"converting to an empty file with mode",4791" 100644\n";4792$fb->{mode_b} ='100644';4793}elsif($bufne'link ') {4794warn"$pathhas mode 120000",4795" but is not a link\n";4796}else{4797my$tmp_fh= $::_repository->temp_acquire(4798'svn_hash');4799my$res;4800while($res=sysread($fh,my$str,1024)) {4801my$out=syswrite($tmp_fh,$str,$res);4802defined($out) &&$out==$res4803or croak("write ",4804 Git::temp_path($tmp_fh),4805":$!\n");4806}4807defined$resor croak $!;48084809($fh,$tmp_fh) = ($tmp_fh,$fh);4810 Git::temp_release($tmp_fh,1);4811}4812}48134814$hash= $::_repository->hash_and_insert_object(4815 Git::temp_path($fh));4816$hash=~/^[a-f\d]{40}$/or die"not a sha1:$hash\n";48174818 Git::temp_release($fb->{base},1);4819 Git::temp_release($fh,1);4820}else{4821$hash=$fb->{blob}or die"no blob information\n";4822}4823$fb->{pool}->clear;4824$self->{gii}->update($fb->{mode_b},$hash,$path)or croak $!;4825print"\t$fb->{action}\t$path\n"if$fb->{action} && ! $::_q;4826undef;4827}48284829sub abort_edit {4830my$self=shift;4831$self->{nr} =$self->{gii}->{nr};4832delete$self->{gii};4833$self->SUPER::abort_edit(@_);4834}48354836sub close_edit {4837my$self=shift;48384839if($_preserve_empty_dirs) {4840my@empty_dirs;48414842# Any entry flagged as empty that also has an associated4843# dir_prop represents a newly created empty directory.4844foreachmy$i(keys%{$self->{empty}}) {4845push@empty_dirs,$iifexists$self->{dir_prop}->{$i};4846}48474848# Search for directories that have become empty due subsequent4849# file deletes.4850push@empty_dirs,$self->find_empty_directories();48514852# Finally, add a placeholder file to each empty directory.4853$self->add_placeholder_file($_)foreach(@empty_dirs);48544855$self->stash_placeholder_list();4856}48574858$self->{git_commit_ok} =1;4859$self->{nr} =$self->{gii}->{nr};4860delete$self->{gii};4861$self->SUPER::close_edit(@_);4862}48634864sub find_empty_directories {4865my($self) =@_;4866my@empty_dirs;4867my%dirs=map{ dirname($_) =>1}@deleted_gpath;48684869foreachmy$dir(sort keys%dirs) {4870next if$direq".";48714872# If there have been any additions to this directory, there is4873# no reason to check if it is empty.4874my$skip_added=0;4875foreachmy$t(qw/dir_prop file_prop/) {4876foreachmy$path(keys%{$self->{$t} }) {4877if(exists$self->{$t}->{dirname($path)}) {4878$skip_added=1;4879last;4880}4881}4882last if$skip_added;4883}4884next if$skip_added;48854886# Use `git ls-tree` to get the filenames of this directory4887# that existed prior to this particular commit.4888my$ls= command('ls-tree','-z','--name-only',4889$self->{c},"$dir/");4890my%files=map{$_=>1}split(/\0/,$ls);48914892# Remove the filenames that were deleted during this commit.4893delete$files{$_}foreach(@deleted_gpath);48944895# Report the directory if there are no filenames left.4896push@empty_dirs,$dirunless(scalar%files);4897}4898@empty_dirs;4899}49004901sub add_placeholder_file {4902my($self,$dir) =@_;4903my$path="$dir/$_placeholder_filename";4904my$gpath=$self->git_path($path);49054906my$fh= $::_repository->temp_acquire($gpath);4907my$hash= $::_repository->hash_and_insert_object(Git::temp_path($fh));4908 Git::temp_release($fh,1);4909$self->{gii}->update('100644',$hash,$gpath)or croak $!;49104911# The directory should no longer be considered empty.4912delete$self->{empty}->{$dir}ifexists$self->{empty}->{$dir};49134914# Keep track of any placeholder files we create.4915$added_placeholder{$dir} =$path;4916}49174918sub stash_placeholder_list {4919my($self) =@_;4920my$k="svn-remote.$repo_id.added-placeholder";4921my$v=eval{ command_oneline('config','--get-all',$k) };4922 command_noisy('config','--unset-all',$k)if$v;4923foreach(values%added_placeholder) {4924 command_noisy('config','--add',$k,$_);4925}4926}49274928package SVN::Git::Editor;4929use vars qw/@ISA $_rmdir $_cp_similarity $_find_copies_harder $_rename_limit/;4930use strict;4931use warnings;4932use Carp qw/croak/;4933use IO::File;49344935sub new {4936my($class,$opts) =@_;4937foreach(qw/svn_path r ra tree_a tree_b log editor_cb/) {4938die"$_required!\n"unless(defined$opts->{$_});4939}49404941my$pool= SVN::Pool->new;4942my$mods= generate_diff($opts->{tree_a},$opts->{tree_b});4943my$types= check_diff_paths($opts->{ra},$opts->{svn_path},4944$opts->{r},$mods);49454946# $opts->{ra} functions should not be used after this:4947my@ce=$opts->{ra}->get_commit_editor($opts->{log},4948$opts->{editor_cb},$pool);4949my$self= SVN::Delta::Editor->new(@ce,$pool);4950bless$self,$class;4951foreach(qw/svn_path r tree_a tree_b/) {4952$self->{$_} =$opts->{$_};4953}4954$self->{url} =$opts->{ra}->{url};4955$self->{mods} =$mods;4956$self->{types} =$types;4957$self->{pool} =$pool;4958$self->{bat} = {''=>$self->open_root($self->{r},$self->{pool}) };4959$self->{rm} = { };4960$self->{path_prefix} =length$self->{svn_path} ?4961"$self->{svn_path}/":'';4962$self->{config} =$opts->{config};4963$self->{mergeinfo} =$opts->{mergeinfo};4964return$self;4965}49664967sub generate_diff {4968my($tree_a,$tree_b) =@_;4969my@diff_tree=qw(diff-tree -z -r);4970if($_cp_similarity) {4971push@diff_tree,"-C$_cp_similarity";4972}else{4973push@diff_tree,'-C';4974}4975push@diff_tree,'--find-copies-harder'if$_find_copies_harder;4976push@diff_tree,"-l$_rename_limit"ifdefined$_rename_limit;4977push@diff_tree,$tree_a,$tree_b;4978my($diff_fh,$ctx) = command_output_pipe(@diff_tree);4979local$/="\0";4980my$state='meta';4981my@mods;4982while(<$diff_fh>) {4983chomp$_;# this gets rid of the trailing "\0"4984if($stateeq'meta'&& /^:(\d{6})\s(\d{6})\s4985($::sha1)\s($::sha1)\s4986([MTCRAD])\d*$/xo) {4987push@mods, { mode_a =>$1, mode_b =>$2,4988 sha1_a =>$3, sha1_b =>$4,4989 chg =>$5};4990if($5=~/^(?:C|R)$/) {4991$state='file_a';4992}else{4993$state='file_b';4994}4995}elsif($stateeq'file_a') {4996my$x=$mods[$#mods]or croak "Empty array\n";4997if($x->{chg} !~/^(?:C|R)$/) {4998 croak "Error parsing$_,$x->{chg}\n";4999}5000$x->{file_a} =$_;5001$state='file_b';5002}elsif($stateeq'file_b') {5003my$x=$mods[$#mods]or croak "Empty array\n";5004if(exists$x->{file_a} &&$x->{chg} !~/^(?:C|R)$/) {5005 croak "Error parsing$_,$x->{chg}\n";5006}5007if(!exists$x->{file_a} &&$x->{chg} =~/^(?:C|R)$/) {5008 croak "Error parsing$_,$x->{chg}\n";5009}5010$x->{file_b} =$_;5011$state='meta';5012}else{5013 croak "Error parsing$_\n";5014}5015}5016 command_close_pipe($diff_fh,$ctx);5017 \@mods;5018}50195020sub check_diff_paths {5021my($ra,$pfx,$rev,$mods) =@_;5022my%types;5023$pfx.='/'iflength$pfx;50245025sub type_diff_paths {5026my($ra,$types,$path,$rev) =@_;5027my@p=split m#/+#, $path;5028my$c=shift@p;5029unless(defined$types->{$c}) {5030$types->{$c} =$ra->check_path($c,$rev);5031}5032while(@p) {5033$c.='/'.shift@p;5034next ifdefined$types->{$c};5035$types->{$c} =$ra->check_path($c,$rev);5036}5037}50385039foreachmy$m(@$mods) {5040foreachmy$f(qw/file_a file_b/) {5041next unlessdefined$m->{$f};5042my($dir) = ($m->{$f} =~ m#^(.*?)/?(?:[^/]+)$#);5043if(length$pfx.$dir&& !defined$types{$dir}) {5044 type_diff_paths($ra, \%types,$pfx.$dir,$rev);5045}5046}5047}5048 \%types;5049}50505051sub split_path {5052return($_[0] =~ m#^(.*?)/?([^/]+)$#);5053}50545055sub repo_path {5056my($self,$path) =@_;5057if(my$enc=$self->{pathnameencoding}) {5058require Encode;5059 Encode::from_to($path,$enc,'UTF-8');5060}5061$self->{path_prefix}.(defined$path?$path:'');5062}50635064sub url_path {5065my($self,$path) =@_;5066if($self->{url} =~ m#^https?://#) {5067$path=~s!([^~a-zA-Z0-9_./-])!uc sprintf("%%%02x",ord($1))!eg;5068}5069$self->{url} .'/'.$self->repo_path($path);5070}50715072sub rmdirs {5073my($self) =@_;5074my$rm=$self->{rm};5075delete$rm->{''};# we never delete the url we're tracking5076return unless%$rm;50775078foreach(keys%$rm) {5079my@d=split m#/#, $_;5080my$c=shift@d;5081$rm->{$c} =1;5082while(@d) {5083$c.='/'.shift@d;5084$rm->{$c} =1;5085}5086}5087delete$rm->{$self->{svn_path}};5088delete$rm->{''};# we never delete the url we're tracking5089return unless%$rm;50905091my($fh,$ctx) = command_output_pipe(qw/ls-tree --name-only -r -z/,5092$self->{tree_b});5093local$/="\0";5094while(<$fh>) {5095chomp;5096my@dn=split m#/#, $_;5097while(pop@dn) {5098delete$rm->{join'/',@dn};5099}5100unless(%$rm) {5101close$fh;5102return;5103}5104}5105 command_close_pipe($fh,$ctx);51065107my($r,$p,$bat) = ($self->{r},$self->{pool},$self->{bat});5108foreachmy$d(sort{$b=~ tr#/#/# <=> $a =~ tr#/#/# } keys %$rm) {5109$self->close_directory($bat->{$d},$p);5110my($dn) = ($d=~ m#^(.*?)/?(?:[^/]+)$#);5111print"\tD+\t$d/\n"unless$::_q;5112$self->SUPER::delete_entry($d,$r,$bat->{$dn},$p);5113delete$bat->{$d};5114}5115}51165117sub open_or_add_dir {5118my($self,$full_path,$baton) =@_;5119my$t=$self->{types}->{$full_path};5120if(!defined$t) {5121die"$full_pathnot known in r$self->{r} or we have a bug!\n";5122}5123{5124no warnings 'once';5125# SVN::Node::none and SVN::Node::file are used only once,5126# so we're shutting up Perl's warnings about them.5127if($t==$SVN::Node::none) {5128return$self->add_directory($full_path,$baton,5129undef, -1,$self->{pool});5130}elsif($t==$SVN::Node::dir) {5131return$self->open_directory($full_path,$baton,5132$self->{r},$self->{pool});5133}# no warnings 'once'5134print STDERR "$full_pathalready exists in repository at ",5135"r$self->{r} and it is not a directory (",5136($t==$SVN::Node::file ?'file':'unknown'),"/$t)\n";5137}# no warnings 'once'5138exit1;5139}51405141sub ensure_path {5142my($self,$path) =@_;5143my$bat=$self->{bat};5144my$repo_path=$self->repo_path($path);5145return$bat->{''}unless(length$repo_path);5146my@p=split m#/+#, $repo_path;5147my$c=shift@p;5148$bat->{$c} ||=$self->open_or_add_dir($c,$bat->{''});5149while(@p) {5150my$c0=$c;5151$c.='/'.shift@p;5152$bat->{$c} ||=$self->open_or_add_dir($c,$bat->{$c0});5153}5154return$bat->{$c};5155}51565157# Subroutine to convert a globbing pattern to a regular expression.5158# From perl cookbook.5159sub glob2pat {5160my$globstr=shift;5161my%patmap= ('*'=>'.*','?'=>'.','['=>'[',']'=>']');5162$globstr=~s{(.)} { $patmap{$1}||"\Q$1"}ge;5163return'^'.$globstr.'$';5164}51655166sub check_autoprop {5167my($self,$pattern,$properties,$file,$fbat) =@_;5168# Convert the globbing pattern to a regular expression.5169my$regex= glob2pat($pattern);5170# Check if the pattern matches the file name.5171if($file=~m/($regex)/) {5172# Parse the list of properties to set.5173my@props=split(/;/,$properties);5174foreachmy$prop(@props) {5175# Parse 'name=value' syntax and set the property.5176if($prop=~/([^=]+)=(.*)/) {5177my($n,$v) = ($1,$2);5178for($n,$v) {5179s/^\s+//;s/\s+$//;5180}5181$self->change_file_prop($fbat,$n,$v);5182}5183}5184}5185}51865187sub apply_autoprops {5188my($self,$file,$fbat) =@_;5189my$conf_t= ${$self->{config}}{'config'};5190no warnings 'once';5191# Check [miscellany]/enable-auto-props in svn configuration.5192if(SVN::_Core::svn_config_get_bool(5193$conf_t,5194$SVN::_Core::SVN_CONFIG_SECTION_MISCELLANY,5195$SVN::_Core::SVN_CONFIG_OPTION_ENABLE_AUTO_PROPS,51960)) {5197# Auto-props are enabled. Enumerate them to look for matches.5198my$callback=sub{5199$self->check_autoprop($_[0],$_[1],$file,$fbat);5200};5201 SVN::_Core::svn_config_enumerate(5202$conf_t,5203$SVN::_Core::SVN_CONFIG_SECTION_AUTO_PROPS,5204$callback);5205}5206}52075208sub A {5209my($self,$m) =@_;5210my($dir,$file) = split_path($m->{file_b});5211my$pbat=$self->ensure_path($dir);5212my$fbat=$self->add_file($self->repo_path($m->{file_b}),$pbat,5213undef, -1);5214print"\tA\t$m->{file_b}\n"unless$::_q;5215$self->apply_autoprops($file,$fbat);5216$self->chg_file($fbat,$m);5217$self->close_file($fbat,undef,$self->{pool});5218}52195220sub C {5221my($self,$m) =@_;5222my($dir,$file) = split_path($m->{file_b});5223my$pbat=$self->ensure_path($dir);5224my$fbat=$self->add_file($self->repo_path($m->{file_b}),$pbat,5225$self->url_path($m->{file_a}),$self->{r});5226print"\tC\t$m->{file_a} =>$m->{file_b}\n"unless$::_q;5227$self->chg_file($fbat,$m);5228$self->close_file($fbat,undef,$self->{pool});5229}52305231sub delete_entry {5232my($self,$path,$pbat) =@_;5233my$rpath=$self->repo_path($path);5234my($dir,$file) = split_path($rpath);5235$self->{rm}->{$dir} =1;5236$self->SUPER::delete_entry($rpath,$self->{r},$pbat,$self->{pool});5237}52385239sub R {5240my($self,$m) =@_;5241my($dir,$file) = split_path($m->{file_b});5242my$pbat=$self->ensure_path($dir);5243my$fbat=$self->add_file($self->repo_path($m->{file_b}),$pbat,5244$self->url_path($m->{file_a}),$self->{r});5245print"\tR\t$m->{file_a} =>$m->{file_b}\n"unless$::_q;5246$self->apply_autoprops($file,$fbat);5247$self->chg_file($fbat,$m);5248$self->close_file($fbat,undef,$self->{pool});52495250($dir,$file) = split_path($m->{file_a});5251$pbat=$self->ensure_path($dir);5252$self->delete_entry($m->{file_a},$pbat);5253}52545255sub M {5256my($self,$m) =@_;5257my($dir,$file) = split_path($m->{file_b});5258my$pbat=$self->ensure_path($dir);5259my$fbat=$self->open_file($self->repo_path($m->{file_b}),5260$pbat,$self->{r},$self->{pool});5261print"\t$m->{chg}\t$m->{file_b}\n"unless$::_q;5262$self->chg_file($fbat,$m);5263$self->close_file($fbat,undef,$self->{pool});5264}52655266sub T { shift->M(@_) }52675268sub change_file_prop {5269my($self,$fbat,$pname,$pval) =@_;5270$self->SUPER::change_file_prop($fbat,$pname,$pval,$self->{pool});5271}52725273sub change_dir_prop {5274my($self,$pbat,$pname,$pval) =@_;5275$self->SUPER::change_dir_prop($pbat,$pname,$pval,$self->{pool});5276}52775278sub _chg_file_get_blob ($$$$) {5279my($self,$fbat,$m,$which) =@_;5280my$fh= $::_repository->temp_acquire("git_blob_$which");5281if($m->{"mode_$which"} =~/^120/) {5282print$fh'link 'or croak $!;5283$self->change_file_prop($fbat,'svn:special','*');5284}elsif($m->{mode_a} =~/^120/&&$m->{"mode_$which"} !~/^120/) {5285$self->change_file_prop($fbat,'svn:special',undef);5286}5287my$blob=$m->{"sha1_$which"};5288return($fh,)if($blob=~/^0{40}$/);5289my$size= $::_repository->cat_blob($blob,$fh);5290 croak "Failed to read object$blob"if($size<0);5291$fh->flush==0or croak $!;5292seek$fh,0,0or croak $!;52935294my$exp= ::md5sum($fh);5295seek$fh,0,0or croak $!;5296return($fh,$exp);5297}52985299sub chg_file {5300my($self,$fbat,$m) =@_;5301if($m->{mode_b} =~/755$/&&$m->{mode_a} !~/755$/) {5302$self->change_file_prop($fbat,'svn:executable','*');5303}elsif($m->{mode_b} !~/755$/&&$m->{mode_a} =~/755$/) {5304$self->change_file_prop($fbat,'svn:executable',undef);5305}5306my($fh_a,$exp_a) = _chg_file_get_blob $self,$fbat,$m,'a';5307my($fh_b,$exp_b) = _chg_file_get_blob $self,$fbat,$m,'b';5308my$pool= SVN::Pool->new;5309my$atd=$self->apply_textdelta($fbat,$exp_a,$pool);5310if(-s $fh_a) {5311my$txstream= SVN::TxDelta::new ($fh_a,$fh_b,$pool);5312my$res= SVN::TxDelta::send_txstream($txstream,@$atd,$pool);5313if(defined$res) {5314die"Unexpected result from send_txstream:$res\n",5315"(SVN::Core::VERSION:$SVN::Core::VERSION)\n";5316}5317}else{5318my$got= SVN::TxDelta::send_stream($fh_b,@$atd,$pool);5319die"Checksum mismatch\nexpected:$exp_b\ngot:$got\n"5320if($gotne$exp_b);5321}5322 Git::temp_release($fh_b,1);5323 Git::temp_release($fh_a,1);5324$pool->clear;5325}53265327sub D {5328my($self,$m) =@_;5329my($dir,$file) = split_path($m->{file_b});5330my$pbat=$self->ensure_path($dir);5331print"\tD\t$m->{file_b}\n"unless$::_q;5332$self->delete_entry($m->{file_b},$pbat);5333}53345335sub close_edit {5336my($self) =@_;5337my($p,$bat) = ($self->{pool},$self->{bat});5338foreach(sort{$b=~ tr#/#/# <=> $a =~ tr#/#/# } keys %$bat) {5339next if$_eq'';5340$self->close_directory($bat->{$_},$p);5341}5342$self->close_directory($bat->{''},$p);5343$self->SUPER::close_edit($p);5344$p->clear;5345}53465347sub abort_edit {5348my($self) =@_;5349$self->SUPER::abort_edit($self->{pool});5350}53515352sub DESTROY {5353my$self=shift;5354$self->SUPER::DESTROY(@_);5355$self->{pool}->clear;5356}53575358# this drives the editor5359sub apply_diff {5360my($self) =@_;5361my$mods=$self->{mods};5362my%o= ( D =>1, R =>0, C => -1, A =>3, M =>3, T =>3);5363foreachmy$m(sort{$o{$a->{chg}} <=>$o{$b->{chg}} }@$mods) {5364my$f=$m->{chg};5365if(defined$o{$f}) {5366$self->$f($m);5367}else{5368 fatal("Invalid change type:$f");5369}5370}53715372if(defined($self->{mergeinfo})) {5373$self->change_dir_prop($self->{bat}{''},"svn:mergeinfo",5374$self->{mergeinfo});5375}5376$self->rmdirsif$_rmdir;5377if(@$mods==0) {5378$self->abort_edit;5379}else{5380$self->close_edit;5381}5382returnscalar@$mods;5383}53845385package Git::SVN::Ra;5386use vars qw/@ISA $config_dir $_log_window_size/;5387use strict;5388use warnings;5389my($ra_invalid,$can_do_switch,%ignored_err,$RA);53905391BEGIN{5392# enforce temporary pool usage for some simple functions5393no strict 'refs';5394formy$f(qw/rev_proplist get_latest_revnum get_uuid get_repos_root5395 get_file/) {5396my$SUPER="SUPER::$f";5397*$f=sub{5398my$self=shift;5399my$pool= SVN::Pool->new;5400my@ret=$self->$SUPER(@_,$pool);5401$pool->clear;5402wantarray?@ret:$ret[0];5403};5404}5405}54065407sub _auth_providers () {5408[5409 SVN::Client::get_simple_provider(),5410 SVN::Client::get_ssl_server_trust_file_provider(),5411 SVN::Client::get_simple_prompt_provider(5412 \&Git::SVN::Prompt::simple,2),5413 SVN::Client::get_ssl_client_cert_file_provider(),5414 SVN::Client::get_ssl_client_cert_prompt_provider(5415 \&Git::SVN::Prompt::ssl_client_cert,2),5416 SVN::Client::get_ssl_client_cert_pw_file_provider(),5417 SVN::Client::get_ssl_client_cert_pw_prompt_provider(5418 \&Git::SVN::Prompt::ssl_client_cert_pw,2),5419 SVN::Client::get_username_provider(),5420 SVN::Client::get_ssl_server_trust_prompt_provider(5421 \&Git::SVN::Prompt::ssl_server_trust),5422 SVN::Client::get_username_prompt_provider(5423 \&Git::SVN::Prompt::username,2)5424]5425}54265427sub escape_uri_only {5428my($uri) =@_;5429my@tmp;5430foreach(splitm{/},$uri) {5431s/([^~\w.%+-]|%(?![a-fA-F0-9]{2}))/sprintf("%%%02X",ord($1))/eg;5432push@tmp,$_;5433}5434join('/',@tmp);5435}54365437sub escape_url {5438my($url) =@_;5439if($url=~ m#^(https?)://([^/]+)(.*)$#) {5440my($scheme,$domain,$uri) = ($1,$2, escape_uri_only($3));5441$url="$scheme://$domain$uri";5442}5443$url;5444}54455446sub new {5447my($class,$url) =@_;5448$url=~s!/+$!!;5449return$RAif($RA&&$RA->{url}eq$url);54505451::_req_svn();54525453 SVN::_Core::svn_config_ensure($config_dir,undef);5454my($baton,$callbacks) = SVN::Core::auth_open_helper(_auth_providers);5455my$config= SVN::Core::config_get_config($config_dir);5456$RA=undef;5457my$dont_store_passwords=1;5458my$conf_t= ${$config}{'config'};5459{5460no warnings 'once';5461# The usage of $SVN::_Core::SVN_CONFIG_* variables5462# produces warnings that variables are used only once.5463# I had not found the better way to shut them up, so5464# the warnings of type 'once' are disabled in this block.5465if(SVN::_Core::svn_config_get_bool($conf_t,5466$SVN::_Core::SVN_CONFIG_SECTION_AUTH,5467$SVN::_Core::SVN_CONFIG_OPTION_STORE_PASSWORDS,54681) ==0) {5469 SVN::_Core::svn_auth_set_parameter($baton,5470$SVN::_Core::SVN_AUTH_PARAM_DONT_STORE_PASSWORDS,5471bless(\$dont_store_passwords,"_p_void"));5472}5473if(SVN::_Core::svn_config_get_bool($conf_t,5474$SVN::_Core::SVN_CONFIG_SECTION_AUTH,5475$SVN::_Core::SVN_CONFIG_OPTION_STORE_AUTH_CREDS,54761) ==0) {5477$Git::SVN::Prompt::_no_auth_cache =1;5478}5479}# no warnings 'once'5480my$self= SVN::Ra->new(url => escape_url($url), auth =>$baton,5481 config =>$config,5482 pool => SVN::Pool->new,5483 auth_provider_callbacks =>$callbacks);5484$self->{url} =$url;5485$self->{svn_path} =$url;5486$self->{repos_root} =$self->get_repos_root;5487$self->{svn_path} =~ s#^\Q$self->{repos_root}\E(/|$)##;5488$self->{cache} = { check_path => { r =>0, data => {} },5489 get_dir => { r =>0, data => {} } };5490$RA=bless$self,$class;5491}54925493sub check_path {5494my($self,$path,$r) =@_;5495my$cache=$self->{cache}->{check_path};5496if($r==$cache->{r} &&exists$cache->{data}->{$path}) {5497return$cache->{data}->{$path};5498}5499my$pool= SVN::Pool->new;5500my$t=$self->SUPER::check_path($path,$r,$pool);5501$pool->clear;5502if($r!=$cache->{r}) {5503%{$cache->{data}} = ();5504$cache->{r} =$r;5505}5506$cache->{data}->{$path} =$t;5507}55085509sub get_dir {5510my($self,$dir,$r) =@_;5511my$cache=$self->{cache}->{get_dir};5512if($r==$cache->{r}) {5513if(my$x=$cache->{data}->{$dir}) {5514returnwantarray?@$x:$x->[0];5515}5516}5517my$pool= SVN::Pool->new;5518my($d,undef,$props) =$self->SUPER::get_dir($dir,$r,$pool);5519my%dirents=map{$_=> { kind =>$d->{$_}->kind} }keys%$d;5520$pool->clear;5521if($r!=$cache->{r}) {5522%{$cache->{data}} = ();5523$cache->{r} =$r;5524}5525$cache->{data}->{$dir} = [ \%dirents,$r,$props];5526wantarray? (\%dirents,$r,$props) : \%dirents;5527}55285529sub DESTROY {5530# do not call the real DESTROY since we store ourselves in $RA5531}55325533# get_log(paths, start, end, limit,5534# discover_changed_paths, strict_node_history, receiver)5535sub get_log {5536my($self,@args) =@_;5537my$pool= SVN::Pool->new;55385539# svn_log_changed_path_t objects passed to get_log are likely to be5540# overwritten even if only the refs are copied to an external variable,5541# so we should dup the structures in their entirety. Using an5542# externally passed pool (instead of our temporary and quickly cleared5543# pool in Git::SVN::Ra) does not help matters at all...5544my$receiver=pop@args;5545my$prefix="/".$self->{svn_path};5546$prefix=~ s#/+($)##;5547my$prefix_regex= qr#^\Q$prefix\E#;5548push(@args,sub{5549my($paths) =$_[0];5550return&$receiver(@_)unless$paths;5551$_[0] = ();5552foreachmy$p(keys%$paths) {5553my$i=$paths->{$p};5554# Make path relative to our url, not repos_root5555$p=~s/$prefix_regex//;5556my%s=map{$_=>$i->$_; }5557 qw/copyfrom_path copyfrom_rev action/;5558if($s{'copyfrom_path'}) {5559$s{'copyfrom_path'} =~s/$prefix_regex//;5560}5561$_[0]{$p} = \%s;5562}5563&$receiver(@_);5564});556555665567# the limit parameter was not supported in SVN 1.1.x, so we5568# drop it. Therefore, the receiver callback passed to it5569# is made aware of this limitation by being wrapped if5570# the limit passed to is being wrapped.5571if($SVN::Core::VERSION le'1.2.0') {5572my$limit=splice(@args,3,1);5573if($limit>0) {5574my$receiver=pop@args;5575push(@args,sub{ &$receiver(@_)if(--$limit>=0) });5576}5577}5578my$ret=$self->SUPER::get_log(@args,$pool);5579$pool->clear;5580$ret;5581}55825583sub trees_match {5584my($self,$url1,$rev1,$url2,$rev2) =@_;5585my$ctx= SVN::Client->new(auth => _auth_providers);5586my$out= IO::File->new_tmpfile;55875588# older SVN (1.1.x) doesn't take $pool as the last parameter for5589# $ctx->diff(), so we'll create a default one5590my$pool= SVN::Pool->new_default_sub;55915592$ra_invalid=1;# this will open a new SVN::Ra connection to $url15593$ctx->diff([],$url1,$rev1,$url2,$rev2,1,1,0,$out,$out);5594$out->flush;5595my$ret= (($out->stat)[7] ==0);5596close$outor croak $!;55975598$ret;5599}56005601sub get_commit_editor {5602my($self,$log,$cb,$pool) =@_;5603my@lock=$SVN::Core::VERSION ge'1.2.0'? (undef,0) : ();5604$self->SUPER::get_commit_editor($log,$cb,@lock,$pool);5605}56065607sub gs_do_update {5608my($self,$rev_a,$rev_b,$gs,$editor) =@_;5609my$new= ($rev_a==$rev_b);5610my$path=$gs->{path};56115612if($new&& -e $gs->{index}) {5613unlink$gs->{index}or die5614"Couldn't unlink index:$gs->{index}:$!\n";5615}5616my$pool= SVN::Pool->new;5617$editor->set_path_strip($path);5618my(@pc) =split m#/#, $path;5619my$reporter=$self->do_update($rev_b, (@pc?shift@pc:''),56201,$editor,$pool);5621my@lock=$SVN::Core::VERSION ge'1.2.0'? (undef) : ();56225623# Since we can't rely on svn_ra_reparent being available, we'll5624# just have to do some magic with set_path to make it so5625# we only want a partial path.5626my$sp='';5627my$final=join('/',@pc);5628while(@pc) {5629$reporter->set_path($sp,$rev_b,0,@lock,$pool);5630$sp.='/'iflength$sp;5631$sp.=shift@pc;5632}5633die"BUG: '$sp' != '$final'\n"if($spne$final);56345635$reporter->set_path($sp,$rev_a,$new,@lock,$pool);56365637$reporter->finish_report($pool);5638$pool->clear;5639$editor->{git_commit_ok};5640}56415642# this requires SVN 1.4.3 or later (do_switch didn't work before 1.4.3, and5643# svn_ra_reparent didn't work before 1.4)5644sub gs_do_switch {5645my($self,$rev_a,$rev_b,$gs,$url_b,$editor) =@_;5646my$path=$gs->{path};5647my$pool= SVN::Pool->new;56485649my$full_url=$self->{url};5650my$old_url=$full_url;5651$full_url.='/'.$pathiflength$path;5652my($ra,$reparented);56535654if($old_url=~ m#^svn(\+ssh)?://# ||5655($full_url=~ m#^https?://# &&5656 escape_url($full_url)ne$full_url)) {5657$_[0] =undef;5658$self=undef;5659$RA=undef;5660$ra= Git::SVN::Ra->new($full_url);5661$ra_invalid=1;5662}elsif($old_urlne$full_url) {5663 SVN::_Ra::svn_ra_reparent($self->{session},$full_url,$pool);5664$self->{url} =$full_url;5665$reparented=1;5666}56675668$ra||=$self;5669$url_b= escape_url($url_b);5670my$reporter=$ra->do_switch($rev_b,'',1,$url_b,$editor,$pool);5671my@lock=$SVN::Core::VERSION ge'1.2.0'? (undef) : ();5672$reporter->set_path('',$rev_a,0,@lock,$pool);5673$reporter->finish_report($pool);56745675if($reparented) {5676 SVN::_Ra::svn_ra_reparent($self->{session},$old_url,$pool);5677$self->{url} =$old_url;5678}56795680$pool->clear;5681$editor->{git_commit_ok};5682}56835684sub longest_common_path {5685my($gsv,$globs) =@_;5686my%common;5687my$common_max=scalar@$gsv;56885689foreachmy$gs(@$gsv) {5690my@tmp=split m#/#, $gs->{path};5691my$p='';5692foreach(@tmp) {5693$p.=length($p) ?"/$_":$_;5694$common{$p} ||=0;5695$common{$p}++;5696}5697}5698$globs||= [];5699$common_max+=scalar@$globs;5700foreachmy$glob(@$globs) {5701my@tmp=split m#/#, $glob->{path}->{left};5702my$p='';5703foreach(@tmp) {5704$p.=length($p) ?"/$_":$_;5705$common{$p} ||=0;5706$common{$p}++;5707}5708}57095710my$longest_path='';5711foreach(sort{length$b<=>length$a}keys%common) {5712if($common{$_} ==$common_max) {5713$longest_path=$_;5714last;5715}5716}5717$longest_path;5718}57195720sub gs_fetch_loop_common {5721my($self,$base,$head,$gsv,$globs) =@_;5722return if($base>$head);5723my$inc=$_log_window_size;5724my($min,$max) = ($base,$head<$base+$inc?$head:$base+$inc);5725my$longest_path= longest_common_path($gsv,$globs);5726my$ra_url=$self->{url};5727my$find_trailing_edge;5728while(1) {5729my%revs;5730my$err;5731my$err_handler=$SVN::Error::handler;5732$SVN::Error::handler =sub{5733($err) =@_;5734 skip_unknown_revs($err);5735};5736sub _cb {5737my($paths,$r,$author,$date,$log) =@_;5738[$paths,5739{ author =>$author, date =>$date,log=>$log} ];5740}5741$self->get_log([$longest_path],$min,$max,0,1,1,5742sub{$revs{$_[1]} = _cb(@_) });5743if($err) {5744print"Checked through r$max\r";5745}else{5746$find_trailing_edge=1;5747}5748if($errand$find_trailing_edge) {5749print STDERR "Path '$longest_path' ",5750"was probably deleted:\n",5751$err->expanded_message,5752"\nWill attempt to follow ",5753"revisions r$min.. r$max",5754"committed before the deletion\n";5755my$hi=$max;5756while(--$hi>=$min) {5757my$ok;5758$self->get_log([$longest_path],$min,$hi,57590,1,1,sub{5760$ok=$_[1];5761$revs{$_[1]} = _cb(@_) });5762if($ok) {5763print STDERR "r$min.. r$okOK\n";5764last;5765}5766}5767$find_trailing_edge=0;5768}5769$SVN::Error::handler =$err_handler;57705771my%exists=map{$_->{path} =>$_}@$gsv;5772foreachmy$r(sort{$a<=>$b}keys%revs) {5773my($paths,$logged) = @{$revs{$r}};57745775foreachmy$gs($self->match_globs(\%exists,$paths,5776$globs,$r)) {5777if($gs->rev_map_max>=$r) {5778next;5779}5780next unless$gs->match_paths($paths,$r);5781$gs->{logged_rev_props} =$logged;5782if(my$last_commit=$gs->last_commit) {5783$gs->assert_index_clean($last_commit);5784}5785my$log_entry=$gs->do_fetch($paths,$r);5786if($log_entry) {5787$gs->do_git_commit($log_entry);5788}5789$INDEX_FILES{$gs->{index}} =1;5790}5791foreachmy$g(@$globs) {5792my$k="svn-remote.$g->{remote}.".5793"$g->{t}-maxRev";5794 Git::SVN::tmp_config($k,$r);5795}5796if($ra_invalid) {5797$_[0] =undef;5798$self=undef;5799$RA=undef;5800$self= Git::SVN::Ra->new($ra_url);5801$ra_invalid=undef;5802}5803}5804# pre-fill the .rev_db since it'll eventually get filled in5805# with '0' x40 if something new gets committed5806foreachmy$gs(@$gsv) {5807next if$gs->rev_map_max>=$max;5808next ifdefined$gs->rev_map_get($max);5809$gs->rev_map_set($max,0 x40);5810}5811foreachmy$g(@$globs) {5812my$k="svn-remote.$g->{remote}.$g->{t}-maxRev";5813 Git::SVN::tmp_config($k,$max);5814}5815last if$max>=$head;5816$min=$max+1;5817$max+=$inc;5818$max=$headif($max>$head);5819}5820 Git::SVN::gc();5821}58225823sub get_dir_globbed {5824my($self,$left,$depth,$r) =@_;58255826my@x=eval{$self->get_dir($left,$r) };5827return unlessscalar@x==3;5828my$dirents=$x[0];5829my@finalents;5830foreachmy$de(keys%$dirents) {5831next if$dirents->{$de}->{kind} !=$SVN::Node::dir;5832if($depth>1) {5833my@args= ("$left/$de",$depth-1,$r);5834foreachmy$dir($self->get_dir_globbed(@args)) {5835push@finalents,"$de/$dir";5836}5837}else{5838push@finalents,$de;5839}5840}5841@finalents;5842}58435844sub match_globs {5845my($self,$exists,$paths,$globs,$r) =@_;58465847sub get_dir_check {5848my($self,$exists,$g,$r) =@_;58495850my@dirs=$self->get_dir_globbed($g->{path}->{left},5851$g->{path}->{depth},5852$r);58535854foreachmy$de(@dirs) {5855my$p=$g->{path}->full_path($de);5856next if$exists->{$p};5857next if(length$g->{path}->{right} &&5858($self->check_path($p,$r) !=5859$SVN::Node::dir));5860next unless$p=~/$g->{path}->{regex}/;5861$exists->{$p} = Git::SVN->init($self->{url},$p,undef,5862$g->{ref}->full_path($de),1);5863}5864}5865foreachmy$g(@$globs) {5866if(my$path=$paths->{"/$g->{path}->{left}"}) {5867if($path->{action} =~/^[AR]$/) {5868 get_dir_check($self,$exists,$g,$r);5869}5870}5871foreach(keys%$paths) {5872if(/$g->{path}->{left_regex}/&&5873!/$g->{path}->{regex}/) {5874next if$paths->{$_}->{action} !~/^[AR]$/;5875 get_dir_check($self,$exists,$g,$r);5876}5877next unless/$g->{path}->{regex}/;5878my$p=$1;5879my$pathname=$g->{path}->full_path($p);5880next if$exists->{$pathname};5881next if($self->check_path($pathname,$r) !=5882$SVN::Node::dir);5883$exists->{$pathname} = Git::SVN->init(5884$self->{url},$pathname,undef,5885$g->{ref}->full_path($p),1);5886}5887my$c='';5888foreach(split m#/#, $g->{path}->{left}) {5889$c.="/$_";5890next unless($paths->{$c} &&5891($paths->{$c}->{action} =~/^[AR]$/));5892 get_dir_check($self,$exists,$g,$r);5893}5894}5895values%$exists;5896}58975898sub minimize_url {5899my($self) =@_;5900return$self->{url}if($self->{url}eq$self->{repos_root});5901my$url=$self->{repos_root};5902my@components=split(m!/!,$self->{svn_path});5903my$c='';5904do{5905$url.="/$c"iflength$c;5906eval{5907my$ra= (ref$self)->new($url);5908my$latest=$ra->get_latest_revnum;5909$ra->get_log("",$latest,0,1,0,1,sub{});5910};5911}while($@&& ($c=shift@components));5912$url;5913}59145915sub can_do_switch {5916my$self=shift;5917unless(defined$can_do_switch) {5918my$pool= SVN::Pool->new;5919my$rep=eval{5920$self->do_switch(1,'',0,$self->{url},5921 SVN::Delta::Editor->new,$pool);5922};5923if($@) {5924$can_do_switch=0;5925}else{5926$rep->abort_report($pool);5927$can_do_switch=1;5928}5929$pool->clear;5930}5931$can_do_switch;5932}59335934sub skip_unknown_revs {5935my($err) =@_;5936my$errno=$err->apr_err();5937# Maybe the branch we're tracking didn't5938# exist when the repo started, so it's5939# not an error if it doesn't, just continue5940#5941# Wonderfully consistent library, eh?5942# 160013 - svn:// and file://5943# 175002 - http(s)://5944# 175007 - http(s):// (this repo required authorization, too...)5945# More codes may be discovered later...5946if($errno==175007||$errno==175002||$errno==160013) {5947my$err_key=$err->expanded_message;5948# revision numbers change every time, filter them out5949$err_key=~s/\d+/\0/g;5950$err_key="$errno\0$err_key";5951unless($ignored_err{$err_key}) {5952warn"W: Ignoring error from SVN, path probably ",5953"does not exist: ($errno): ",5954$err->expanded_message,"\n";5955warn"W: Do not be alarmed at the above message ",5956"git-svn is just searching aggressively for ",5957"old history.\n",5958"This may take a while on large repositories\n";5959$ignored_err{$err_key} =1;5960}5961return;5962}5963die"Error from SVN, ($errno): ",$err->expanded_message,"\n";5964}59655966package Git::SVN::Log;5967use strict;5968use warnings;5969use POSIX qw/strftime/;5970use Time::Local;5971useconstant commit_log_separator => ('-' x 72) ."\n";5972use vars qw/$TZ $limit $color $pager $non_recursive $verbose $oneline5973%rusers $show_commit $incremental/;5974my$l_fmt;59755976sub cmt_showable {5977my($c) =@_;5978return1ifdefined$c->{r};59795980# big commit message got truncated by the 16k pretty buffer in rev-list5981if($c->{l} &&$c->{l}->[-1]eq"...\n"&&5982$c->{a_raw} =~/\@([a-f\d\-]+)>$/) {5983@{$c->{l}} = ();5984my@log= command(qw/cat-file commit/,$c->{c});59855986# shift off the headers5987shift@logwhile($log[0]ne'');5988shift@log;59895990# TODO: make $c->{l} not have a trailing newline in the future5991@{$c->{l}} =map{"$_\n"}grep!/^git-svn-id: /,@log;59925993(undef,$c->{r},undef) = ::extract_metadata(5994(grep(/^git-svn-id: /,@log))[-1]);5995}5996returndefined$c->{r};5997}59985999sub log_use_color {6000return$color|| Git->repository->get_colorbool('color.diff');6001}60026003sub git_svn_log_cmd {6004my($r_min,$r_max,@args) =@_;6005my$head='HEAD';6006my(@files,@log_opts);6007foreachmy$x(@args) {6008if($xeq'--'||@files) {6009push@files,$x;6010}else{6011if(::verify_ref("$x^0")) {6012$head=$x;6013}else{6014push@log_opts,$x;6015}6016}6017}60186019my($url,$rev,$uuid,$gs) = ::working_head_info($head);6020$gs||= Git::SVN->_new;6021my@cmd= (qw/log --abbrev-commit --pretty=raw --default/,6022$gs->refname);6023push@cmd,'-r'unless$non_recursive;6024push@cmd, qw/--raw --name-status/if$verbose;6025push@cmd,'--color'if log_use_color();6026push@cmd,@log_opts;6027if(defined$r_max&&$r_max==$r_min) {6028push@cmd,'--max-count=1';6029if(my$c=$gs->rev_map_get($r_max)) {6030push@cmd,$c;6031}6032}elsif(defined$r_max) {6033if($r_max<$r_min) {6034($r_min,$r_max) = ($r_max,$r_min);6035}6036my(undef,$c_max) =$gs->find_rev_before($r_max,1,$r_min);6037my(undef,$c_min) =$gs->find_rev_after($r_min,1,$r_max);6038# If there are no commits in the range, both $c_max and $c_min6039# will be undefined. If there is at least 1 commit in the6040# range, both will be defined.6041return()if!defined$c_min|| !defined$c_max;6042if($c_mineq$c_max) {6043push@cmd,'--max-count=1',$c_min;6044}else{6045push@cmd,'--boundary',"$c_min..$c_max";6046}6047}6048return(@cmd,@files);6049}60506051# adapted from pager.c6052sub config_pager {6053if(! -t *STDOUT) {6054$ENV{GIT_PAGER_IN_USE} ='false';6055$pager=undef;6056return;6057}6058chomp($pager= command_oneline(qw(var GIT_PAGER)));6059if($pagereq'cat') {6060$pager=undef;6061}6062$ENV{GIT_PAGER_IN_USE} =defined($pager);6063}60646065sub run_pager {6066return unlessdefined$pager;6067pipe my($rfd,$wfd)orreturn;6068defined(my$pid=fork)or::fatal "Can't fork:$!";6069if(!$pid) {6070open STDOUT,'>&',$wfdor6071::fatal "Can't redirect to stdout:$!";6072return;6073}6074open STDIN,'<&',$rfdor::fatal "Can't redirect stdin:$!";6075$ENV{LESS} ||='FRSX';6076exec$pageror::fatal "Can't run pager:$!($pager)";6077}60786079sub format_svn_date {6080# some systmes don't handle or mishandle %z, so be creative.6081my$t=shift||time;6082my$gm= timelocal(gmtime($t));6083my$sign=qw( + + - )[$t<=>$gm];6084my$gmoff=sprintf("%s%02d%02d",$sign, (gmtime(abs($t-$gm)))[2,1]);6085return strftime("%Y-%m-%d%H:%M:%S$gmoff(%a,%d%b%Y)",localtime($t));6086}60876088sub parse_git_date {6089my($t,$tz) =@_;6090# Date::Parse isn't in the standard Perl distro :(6091if($tz=~s/^\+//) {6092$t+= tz_to_s_offset($tz);6093}elsif($tz=~s/^\-//) {6094$t-= tz_to_s_offset($tz);6095}6096return$t;6097}60986099sub set_local_timezone {6100if(defined$TZ) {6101$ENV{TZ} =$TZ;6102}else{6103delete$ENV{TZ};6104}6105}61066107sub tz_to_s_offset {6108my($tz) =@_;6109$tz=~s/(\d\d)$//;6110return($1*60) + ($tz*3600);6111}61126113sub get_author_info {6114my($dest,$author,$t,$tz) =@_;6115$author=~s/(?:^\s*|\s*$)//g;6116$dest->{a_raw} =$author;6117my$au;6118if($::_authors) {6119$au=$rusers{$author} ||undef;6120}6121if(!$au) {6122($au) = ($author=~/<([^>]+)\@[^>]+>$/);6123}6124$dest->{t} =$t;6125$dest->{tz} =$tz;6126$dest->{a} =$au;6127$dest->{t_utc} = parse_git_date($t,$tz);6128}61296130sub process_commit {6131my($c,$r_min,$r_max,$defer) =@_;6132if(defined$r_min&&defined$r_max) {6133if($r_min==$c->{r} &&$r_min==$r_max) {6134 show_commit($c);6135return0;6136}6137return1if$r_min==$r_max;6138if($r_min<$r_max) {6139# we need to reverse the print order6140return0if(defined$limit&& --$limit<0);6141push@$defer,$c;6142return1;6143}6144if($r_min!=$r_max) {6145return1if($r_min<$c->{r});6146return1if($r_max>$c->{r});6147}6148}6149return0if(defined$limit&& --$limit<0);6150 show_commit($c);6151return1;6152}61536154sub show_commit {6155my$c=shift;6156if($oneline) {6157my$x="\n";6158if(my$l=$c->{l}) {6159while($l->[0] =~/^\s*$/) {shift@$l}6160$x=$l->[0];6161}6162$l_fmt||='A'.length($c->{r});6163print'r',pack($l_fmt,$c->{r}),' | ';6164print"$c->{c} | "if$show_commit;6165print$x;6166}else{6167 show_commit_normal($c);6168}6169}61706171sub show_commit_changed_paths {6172my($c) =@_;6173return unless$c->{changed};6174print"Changed paths:\n", @{$c->{changed}};6175}61766177sub show_commit_normal {6178my($c) =@_;6179print commit_log_separator,"r$c->{r} | ";6180print"$c->{c} | "if$show_commit;6181print"$c->{a} | ", format_svn_date($c->{t_utc}),' | ';6182my$nr_line=0;61836184if(my$l=$c->{l}) {6185while($l->[$#$l]eq"\n"&&$#$l>06186&&$l->[($#$l-1)]eq"\n") {6187pop@$l;6188}6189$nr_line=scalar@$l;6190if(!$nr_line) {6191print"1 line\n\n\n";6192}else{6193if($nr_line==1) {6194$nr_line='1 line';6195}else{6196$nr_line.=' lines';6197}6198print$nr_line,"\n";6199 show_commit_changed_paths($c);6200print"\n";6201print$_foreach@$l;6202}6203}else{6204print"1 line\n";6205 show_commit_changed_paths($c);6206print"\n";62076208}6209foreachmy$x(qw/raw stat diff/) {6210if($c->{$x}) {6211print"\n";6212print$_foreach@{$c->{$x}}6213}6214}6215}62166217sub cmd_show_log {6218my(@args) =@_;6219my($r_min,$r_max);6220my$r_last= -1;# prevent dupes6221 set_local_timezone();6222if(defined$::_revision) {6223if($::_revision =~/^(\d+):(\d+)$/) {6224($r_min,$r_max) = ($1,$2);6225}elsif($::_revision =~/^\d+$/) {6226$r_min=$r_max= $::_revision;6227}else{6228::fatal "-r$::_revision is not supported, use ",6229"standard 'git log' arguments instead";6230}6231}62326233 config_pager();6234@args= git_svn_log_cmd($r_min,$r_max,@args);6235if(!@args) {6236print commit_log_separator unless$incremental||$oneline;6237return;6238}6239my$log= command_output_pipe(@args);6240 run_pager();6241my(@k,$c,$d,$stat);6242my$esc_color=qr/(?:\033\[(?:(?:\d+;)*\d*)?m)*/;6243while(<$log>) {6244if(/^${esc_color}commit (?:- )?($::sha1_short)/o) {6245my$cmt=$1;6246if($c&& cmt_showable($c) &&$c->{r} !=$r_last) {6247$r_last=$c->{r};6248 process_commit($c,$r_min,$r_max, \@k)or6249goto out;6250}6251$d=undef;6252$c= { c =>$cmt};6253}elsif(/^${esc_color}author (.+) (\d+) ([\-\+]?\d+)$/o) {6254 get_author_info($c,$1,$2,$3);6255}elsif(/^${esc_color}(?:tree|parent|committer) /o) {6256# ignore6257}elsif(/^${esc_color}:\d{6} \d{6} $::sha1_short/o) {6258push@{$c->{raw}},$_;6259}elsif(/^${esc_color}[ACRMDT]\t/) {6260# we could add $SVN->{svn_path} here, but that requires6261# remote access at the moment (repo_path_split)...6262 s#^(${esc_color})([ACRMDT])\t#$1 $2 #o;6263push@{$c->{changed}},$_;6264}elsif(/^${esc_color}diff /o) {6265$d=1;6266push@{$c->{diff}},$_;6267}elsif($d) {6268push@{$c->{diff}},$_;6269}elsif(/^\ .+\ \|\s*\d+\ $esc_color[\+\-]*6270$esc_color*[\+\-]*$esc_color$/x) {6271$stat=1;6272push@{$c->{stat}},$_;6273}elsif($stat&&/^ \d+ files changed, \d+ insertions/) {6274push@{$c->{stat}},$_;6275$stat=undef;6276}elsif(/^${esc_color} (git-svn-id:.+)$/o) {6277($c->{url},$c->{r},undef) = ::extract_metadata($1);6278}elsif(s/^${esc_color} //o) {6279push@{$c->{l}},$_;6280}6281}6282if($c&&defined$c->{r} &&$c->{r} !=$r_last) {6283$r_last=$c->{r};6284 process_commit($c,$r_min,$r_max, \@k);6285}6286if(@k) {6287($r_min,$r_max) = ($r_max,$r_min);6288 process_commit($_,$r_min,$r_max)foreachreverse@k;6289}6290out:6291close$log;6292print commit_log_separator unless$incremental||$oneline;6293}62946295sub cmd_blame {6296my$path=pop;62976298 config_pager();6299 run_pager();63006301my($fh,$ctx,$rev);63026303if($_git_format) {6304($fh,$ctx) = command_output_pipe('blame',@_,$path);6305while(my$line= <$fh>) {6306if($line=~/^\^?([[:xdigit:]]+)\s/) {6307# Uncommitted edits show up as a rev ID of6308# all zeros, which we can't look up with6309# cmt_metadata6310if($1!~/^0+$/) {6311(undef,$rev,undef) =6312::cmt_metadata($1);6313$rev='0'if(!$rev);6314}else{6315$rev='0';6316}6317$rev=sprintf('%-10s',$rev);6318$line=~s/^\^?[[:xdigit:]]+(\s)/$rev$1/;6319}6320print$line;6321}6322}else{6323($fh,$ctx) = command_output_pipe('blame','-p',@_,'HEAD',6324'--',$path);6325my($sha1);6326my%authors;6327my@buffer;6328my%dsha;#distinct sha keys63296330while(my$line= <$fh>) {6331push@buffer,$line;6332if($line=~/^([[:xdigit:]]{40})\s\d+\s\d+/) {6333$dsha{$1} =1;6334}6335}63366337my$s2r= ::cmt_sha2rev_batch([keys%dsha]);63386339foreachmy$line(@buffer) {6340if($line=~/^([[:xdigit:]]{40})\s\d+\s\d+/) {6341$rev=$s2r->{$1};6342$rev='0'if(!$rev)6343}6344elsif($line=~/^author (.*)/) {6345$authors{$rev} =$1;6346$authors{$rev} =~s/\s/_/g;6347}6348elsif($line=~/^\t(.*)$/) {6349printf("%6s%10s%s\n",$rev,$authors{$rev},$1);6350}6351}6352}6353 command_close_pipe($fh,$ctx);6354}63556356package Git::SVN::Migration;6357# these version numbers do NOT correspond to actual version numbers6358# of git nor git-svn. They are just relative.6359#6360# v0 layout: .git/$id/info/url, refs/heads/$id-HEAD6361#6362# v1 layout: .git/$id/info/url, refs/remotes/$id6363#6364# v2 layout: .git/svn/$id/info/url, refs/remotes/$id6365#6366# v3 layout: .git/svn/$id, refs/remotes/$id6367# - info/url may remain for backwards compatibility6368# - this is what we migrate up to this layout automatically,6369# - this will be used by git svn init on single branches6370# v3.1 layout (auto migrated):6371# - .rev_db => .rev_db.$UUID, .rev_db will remain as a symlink6372# for backwards compatibility6373#6374# v4 layout: .git/svn/$repo_id/$id, refs/remotes/$repo_id/$id6375# - this is only created for newly multi-init-ed6376# repositories. Similar in spirit to the6377# --use-separate-remotes option in git-clone (now default)6378# - we do not automatically migrate to this (following6379# the example set by core git)6380#6381# v5 layout: .rev_db.$UUID => .rev_map.$UUID6382# - newer, more-efficient format that uses 24-bytes per record6383# with no filler space.6384# - use xxd -c24 < .rev_map.$UUID to view and debug6385# - This is a one-way migration, repositories updated to the6386# new format will not be able to use old git-svn without6387# rebuilding the .rev_db. Rebuilding the rev_db is not6388# possible if noMetadata or useSvmProps are set; but should6389# be no problem for users that use the (sensible) defaults.6390use strict;6391use warnings;6392use Carp qw/croak/;6393use File::Path qw/mkpath/;6394use File::Basename qw/dirname basename/;6395use vars qw/$_minimize/;63966397sub migrate_from_v0 {6398my$git_dir=$ENV{GIT_DIR};6399returnundefunless-d $git_dir;6400my($fh,$ctx) = command_output_pipe(qw/rev-parse --symbolic --all/);6401my$migrated=0;6402while(<$fh>) {6403chomp;6404my($id,$orig_ref) = ($_,$_);6405next unless$id=~ s#^refs/heads/(.+)-HEAD$#$1#;6406next unless-f "$git_dir/$id/info/url";6407my$new_ref="refs/remotes/$id";6408if(::verify_ref("$new_ref^0")) {6409print STDERR "W:$orig_refis probably an old ",6410"branch used by an ancient version of ",6411"git-svn.\n",6412"However,$new_refalso exists.\n",6413"We will not be able ",6414"to use this branch until this ",6415"ambiguity is resolved.\n";6416next;6417}6418print STDERR "Migrating from v0 layout...\n"if!$migrated;6419print STDERR "Renaming ref:$orig_ref=>$new_ref\n";6420 command_noisy('update-ref',$new_ref,$orig_ref);6421 command_noisy('update-ref','-d',$orig_ref,$orig_ref);6422$migrated++;6423}6424 command_close_pipe($fh,$ctx);6425print STDERR "Done migrating from v0 layout...\n"if$migrated;6426$migrated;6427}64286429sub migrate_from_v1 {6430my$git_dir=$ENV{GIT_DIR};6431my$migrated=0;6432return$migratedunless-d $git_dir;6433my$svn_dir="$git_dir/svn";64346435# just in case somebody used 'svn' as their $id at some point...6436return$migratedif-d $svn_dir&& ! -f "$svn_dir/info/url";64376438print STDERR "Migrating from a git-svn v1 layout...\n";6439 mkpath([$svn_dir]);6440print STDERR "Data from a previous version of git-svn exists, but\n\t",6441"$svn_dir\n\t(required for this version ",6442"($::VERSION) of git-svn) does not exist.\n";6443my($fh,$ctx) = command_output_pipe(qw/rev-parse --symbolic --all/);6444while(<$fh>) {6445my$x=$_;6446next unless$x=~ s#^refs/remotes/##;6447chomp$x;6448next unless-f "$git_dir/$x/info/url";6449my$u=eval{ ::file_to_s("$git_dir/$x/info/url") };6450next unless$u;6451my$dn= dirname("$git_dir/svn/$x");6452 mkpath([$dn])unless-d $dn;6453if($xeq'svn') {# they used 'svn' as GIT_SVN_ID:6454 mkpath(["$git_dir/svn/svn"]);6455print STDERR " -$git_dir/$x/info=> ",6456"$git_dir/svn/$x/info\n";6457rename"$git_dir/$x/info","$git_dir/svn/$x/info"or6458 croak "$!:$x";6459# don't worry too much about these, they probably6460# don't exist with repos this old (save for index,6461# and we can easily regenerate that)6462foreachmy$f(qw/unhandled.log index .rev_db/) {6463rename"$git_dir/$x/$f","$git_dir/svn/$x/$f";6464}6465}else{6466print STDERR " -$git_dir/$x=>$git_dir/svn/$x\n";6467rename"$git_dir/$x","$git_dir/svn/$x"or6468 croak "$!:$x";6469}6470$migrated++;6471}6472 command_close_pipe($fh,$ctx);6473print STDERR "Done migrating from a git-svn v1 layout\n";6474$migrated;6475}64766477sub read_old_urls {6478my($l_map,$pfx,$path) =@_;6479my@dir;6480foreach(<$path/*>) {6481if(-r "$_/info/url") {6482$pfx.='/'if$pfx&&$pfx!~ m!/$!;6483my$ref_id=$pfx. basename $_;6484my$url= ::file_to_s("$_/info/url");6485$l_map->{$ref_id} =$url;6486}elsif(-d $_) {6487push@dir,$_;6488}6489}6490foreach(@dir) {6491my$x=$_;6492$x=~s!^\Q$ENV{GIT_DIR}\E/svn/!!o;6493 read_old_urls($l_map,$x,$_);6494}6495}64966497sub migrate_from_v2 {6498my@cfg= command(qw/config -l/);6499return ifgrep/^svn-remote\..+\.url=/,@cfg;6500my%l_map;6501 read_old_urls(\%l_map,'',"$ENV{GIT_DIR}/svn");6502my$migrated=0;65036504foreachmy$ref_id(sort keys%l_map) {6505eval{ Git::SVN->init($l_map{$ref_id},'',undef,$ref_id) };6506if($@) {6507 Git::SVN->init($l_map{$ref_id},'',$ref_id,$ref_id);6508}6509$migrated++;6510}6511$migrated;6512}65136514sub minimize_connections {6515my$r= Git::SVN::read_all_remotes();6516my$new_urls= {};6517my$root_repos= {};6518foreachmy$repo_id(keys%$r) {6519my$url=$r->{$repo_id}->{url}ornext;6520my$fetch=$r->{$repo_id}->{fetch}ornext;6521my$ra= Git::SVN::Ra->new($url);65226523# skip existing cases where we already connect to the root6524if(($ra->{url}eq$ra->{repos_root}) ||6525($ra->{repos_root}eq$repo_id)) {6526$root_repos->{$ra->{url}} =$repo_id;6527next;6528}65296530my$root_ra= Git::SVN::Ra->new($ra->{repos_root});6531my$root_path=$ra->{url};6532$root_path=~ s#^\Q$ra->{repos_root}\E(/|$)##;6533foreachmy$path(keys%$fetch) {6534my$ref_id=$fetch->{$path};6535my$gs= Git::SVN->new($ref_id,$repo_id,$path);65366537# make sure we can read when connecting to6538# a higher level of a repository6539my($last_rev,undef) =$gs->last_rev_commit;6540if(!defined$last_rev) {6541$last_rev=eval{6542$root_ra->get_latest_revnum;6543};6544next if$@;6545}6546my$new=$root_path;6547$new.=length$path?"/$path":'';6548eval{6549$root_ra->get_log([$new],$last_rev,$last_rev,65500,0,1,sub{ });6551};6552next if$@;6553$new_urls->{$ra->{repos_root}}->{$new} =6554{ ref_id =>$ref_id,6555 old_repo_id =>$repo_id,6556 old_path =>$path};6557}6558}65596560my@emptied;6561foreachmy$url(keys%$new_urls) {6562# see if we can re-use an existing [svn-remote "repo_id"]6563# instead of creating a(n ugly) new section:6564my$repo_id=$root_repos->{$url} ||$url;65656566my$fetch=$new_urls->{$url};6567foreachmy$path(keys%$fetch) {6568my$x=$fetch->{$path};6569 Git::SVN->init($url,$path,$repo_id,$x->{ref_id});6570my$pfx="svn-remote.$x->{old_repo_id}";65716572my$old_fetch=quotemeta("$x->{old_path}:".6573"$x->{ref_id}");6574 command_noisy(qw/config --unset/,6575"$pfx.fetch",'^'.$old_fetch.'$');6576delete$r->{$x->{old_repo_id}}->6577{fetch}->{$x->{old_path}};6578if(!keys%{$r->{$x->{old_repo_id}}->{fetch}}) {6579 command_noisy(qw/config --unset/,6580"$pfx.url");6581push@emptied,$x->{old_repo_id}6582}6583}6584}6585if(@emptied) {6586my$file=$ENV{GIT_CONFIG} ||"$ENV{GIT_DIR}/config";6587print STDERR <<EOF;6588The following [svn-remote] sections in your config file ($file) are empty6589and can be safely removed:6590EOF6591print STDERR "[svn-remote\"$_\"]\n"foreach@emptied;6592}6593}65946595sub migration_check {6596 migrate_from_v0();6597 migrate_from_v1();6598 migrate_from_v2();6599 minimize_connections()if$_minimize;6600}66016602package Git::IndexInfo;6603use strict;6604use warnings;6605use Git qw/command_input_pipe command_close_pipe/;66066607sub new {6608my($class) =@_;6609my($gui,$ctx) = command_input_pipe(qw/update-index -z --index-info/);6610bless{ gui =>$gui, ctx =>$ctx, nr =>0},$class;6611}66126613sub remove {6614my($self,$path) =@_;6615if(print{$self->{gui} }'0 ',0 x 40,"\t",$path,"\0") {6616return++$self->{nr};6617}6618undef;6619}66206621sub update {6622my($self,$mode,$hash,$path) =@_;6623if(print{$self->{gui} }$mode,' ',$hash,"\t",$path,"\0") {6624return++$self->{nr};6625}6626undef;6627}66286629sub DESTROY {6630my($self) =@_;6631 command_close_pipe($self->{gui},$self->{ctx});6632}66336634package Git::SVN::GlobSpec;6635use strict;6636use warnings;66376638sub new {6639my($class,$glob,$pattern_ok) =@_;6640my$re=$glob;6641$re=~s!/+$!!g;# no need for trailing slashes6642my(@left,@right,@patterns);6643my$state="left";6644my$die_msg="Only one set of wildcard directories ".6645"(e.g. '*' or '*/*/*') is supported: '$glob'\n";6646formy$part(split(m|/|,$glob)) {6647if($part=~/\*/&&$partne"*") {6648die"Invalid pattern in '$glob':$part\n";6649}elsif($pattern_ok&&$part=~/[{}]/&&6650$part!~/^\{[^{}]+\}/) {6651die"Invalid pattern in '$glob':$part\n";6652}6653if($parteq"*") {6654die$die_msgif$stateeq"right";6655$state="pattern";6656push(@patterns,"[^/]*");6657}elsif($pattern_ok&&$part=~/^\{(.*)\}$/) {6658die$die_msgif$stateeq"right";6659$state="pattern";6660my$p=quotemeta($1);6661$p=~s/\\,/|/g;6662push(@patterns,"(?:$p)");6663}else{6664if($stateeq"left") {6665push(@left,$part);6666}else{6667push(@right,$part);6668$state="right";6669}6670}6671}6672my$depth=@patterns;6673if($depth==0) {6674die"One '*' is needed in glob: '$glob'\n";6675}6676my$left=join('/',@left);6677my$right=join('/',@right);6678$re=join('/',@patterns);6679$re=join('\/',6680grep(length,quotemeta($left),"($re)",quotemeta($right)));6681my$left_re=qr/^\/\Q$left\E(\/|$)/;6682bless{ left =>$left, right =>$right, left_regex =>$left_re,6683 regex =>qr/$re/,glob=>$glob, depth =>$depth},$class;6684}66856686sub full_path {6687my($self,$path) =@_;6688return(length$self->{left} ?"$self->{left}/":'') .6689$path. (length$self->{right} ?"/$self->{right}":'');6690}66916692__END__66936694Data structures:669566966697$remotes= {# returned by read_all_remotes()6698'svn'=> {6699# svn-remote.svn.url=https://svn.musicpd.org6700 url =>'https://svn.musicpd.org',6701# svn-remote.svn.fetch=mpd/trunk:trunk6702 fetch => {6703'mpd/trunk'=>'trunk',6704},6705# svn-remote.svn.tags=mpd/tags/*:tags/*6706 tags => {6707 path => {6708 left =>'mpd/tags',6709 right =>'',6710 regex =>qr!mpd/tags/([^/]+)$!,6711glob=>'tags/*',6712},6713ref=> {6714 left =>'tags',6715 right =>'',6716 regex =>qr!tags/([^/]+)$!,6717glob=>'tags/*',6718},6719}6720}6721};67226723$log_entry hashref as returned by libsvn_log_entry()6724{6725log=>'whitespace-formatted log entry6726',# trailing newline is preserved6727 revision =>'8',# integer6728 date =>'2004-02-24T17:01:44.108345Z',# commit date6729 author =>'committer name'6730};673167326733# this is generated by generate_diff();6734@mods= array of diff-index line hashes,each element represents one line6735 of diff-index output67366737diff-index line ($m hash)6738{6739 mode_a => first column of diff-index output,no leading ':',6740 mode_b => second column of diff-index output,6741 sha1_b => sha1sum of the final blob,6742 chg => change type [MCRADT],6743 file_a => original file name of a file (iff chg is'C'or'R')6744 file_b => new/current file name of a file (any chg)6745}6746;67476748# retval of read_url_paths{,_all}();6749$l_map= {6750# repository root url6751'https://svn.musicpd.org'=> {6752# repository path # GIT_SVN_ID6753'mpd/trunk'=>'trunk',6754'mpd/tags/0.11.5'=>'tags/0.11.5',6755},6756}67576758Notes:6759 I don't trust the each() function on unless I created%hashmyself6760 because the internal iterator may not have started at base.