You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
258 lines
7.8 KiB
258 lines
7.8 KiB
package Git::SVN::Migration; |
|
# these version numbers do NOT correspond to actual version numbers |
|
# of git or git-svn. They are just relative. |
|
# |
|
# v0 layout: .git/$id/info/url, refs/heads/$id-HEAD |
|
# |
|
# v1 layout: .git/$id/info/url, refs/remotes/$id |
|
# |
|
# v2 layout: .git/svn/$id/info/url, refs/remotes/$id |
|
# |
|
# v3 layout: .git/svn/$id, refs/remotes/$id |
|
# - info/url may remain for backwards compatibility |
|
# - this is what we migrate up to this layout automatically, |
|
# - this will be used by git svn init on single branches |
|
# v3.1 layout (auto migrated): |
|
# - .rev_db => .rev_db.$UUID, .rev_db will remain as a symlink |
|
# for backwards compatibility |
|
# |
|
# v4 layout: .git/svn/$repo_id/$id, refs/remotes/$repo_id/$id |
|
# - this is only created for newly multi-init-ed |
|
# repositories. Similar in spirit to the |
|
# --use-separate-remotes option in git-clone (now default) |
|
# - we do not automatically migrate to this (following |
|
# the example set by core git) |
|
# |
|
# v5 layout: .rev_db.$UUID => .rev_map.$UUID |
|
# - newer, more-efficient format that uses 24-bytes per record |
|
# with no filler space. |
|
# - use xxd -c24 < .rev_map.$UUID to view and debug |
|
# - This is a one-way migration, repositories updated to the |
|
# new format will not be able to use old git-svn without |
|
# rebuilding the .rev_db. Rebuilding the rev_db is not |
|
# possible if noMetadata or useSvmProps are set; but should |
|
# be no problem for users that use the (sensible) defaults. |
|
use strict; |
|
use warnings; |
|
use Carp qw/croak/; |
|
use File::Path qw/mkpath/; |
|
use File::Basename qw/dirname basename/; |
|
|
|
our $_minimize; |
|
use Git qw( |
|
command |
|
command_noisy |
|
command_output_pipe |
|
command_close_pipe |
|
); |
|
|
|
sub migrate_from_v0 { |
|
my $git_dir = $ENV{GIT_DIR}; |
|
return undef unless -d $git_dir; |
|
my ($fh, $ctx) = command_output_pipe(qw/rev-parse --symbolic --all/); |
|
my $migrated = 0; |
|
while (<$fh>) { |
|
chomp; |
|
my ($id, $orig_ref) = ($_, $_); |
|
next unless $id =~ s#^refs/heads/(.+)-HEAD$#$1#; |
|
next unless -f "$git_dir/$id/info/url"; |
|
my $new_ref = "refs/remotes/$id"; |
|
if (::verify_ref("$new_ref^0")) { |
|
print STDERR "W: $orig_ref is probably an old ", |
|
"branch used by an ancient version of ", |
|
"git-svn.\n", |
|
"However, $new_ref also exists.\n", |
|
"We will not be able ", |
|
"to use this branch until this ", |
|
"ambiguity is resolved.\n"; |
|
next; |
|
} |
|
print STDERR "Migrating from v0 layout...\n" if !$migrated; |
|
print STDERR "Renaming ref: $orig_ref => $new_ref\n"; |
|
command_noisy('update-ref', $new_ref, $orig_ref); |
|
command_noisy('update-ref', '-d', $orig_ref, $orig_ref); |
|
$migrated++; |
|
} |
|
command_close_pipe($fh, $ctx); |
|
print STDERR "Done migrating from v0 layout...\n" if $migrated; |
|
$migrated; |
|
} |
|
|
|
sub migrate_from_v1 { |
|
my $git_dir = $ENV{GIT_DIR}; |
|
my $migrated = 0; |
|
return $migrated unless -d $git_dir; |
|
my $svn_dir = "$git_dir/svn"; |
|
|
|
# just in case somebody used 'svn' as their $id at some point... |
|
return $migrated if -d $svn_dir && ! -f "$svn_dir/info/url"; |
|
|
|
print STDERR "Migrating from a git-svn v1 layout...\n"; |
|
mkpath([$svn_dir]); |
|
print STDERR "Data from a previous version of git-svn exists, but\n\t", |
|
"$svn_dir\n\t(required for this version ", |
|
"($::VERSION) of git-svn) does not exist.\n"; |
|
my ($fh, $ctx) = command_output_pipe(qw/rev-parse --symbolic --all/); |
|
while (<$fh>) { |
|
my $x = $_; |
|
next unless $x =~ s#^refs/remotes/##; |
|
chomp $x; |
|
next unless -f "$git_dir/$x/info/url"; |
|
my $u = eval { ::file_to_s("$git_dir/$x/info/url") }; |
|
next unless $u; |
|
my $dn = dirname("$git_dir/svn/$x"); |
|
mkpath([$dn]) unless -d $dn; |
|
if ($x eq 'svn') { # they used 'svn' as GIT_SVN_ID: |
|
mkpath(["$git_dir/svn/svn"]); |
|
print STDERR " - $git_dir/$x/info => ", |
|
"$git_dir/svn/$x/info\n"; |
|
rename "$git_dir/$x/info", "$git_dir/svn/$x/info" or |
|
croak "$!: $x"; |
|
# don't worry too much about these, they probably |
|
# don't exist with repos this old (save for index, |
|
# and we can easily regenerate that) |
|
foreach my $f (qw/unhandled.log index .rev_db/) { |
|
rename "$git_dir/$x/$f", "$git_dir/svn/$x/$f"; |
|
} |
|
} else { |
|
print STDERR " - $git_dir/$x => $git_dir/svn/$x\n"; |
|
rename "$git_dir/$x", "$git_dir/svn/$x" or |
|
croak "$!: $x"; |
|
} |
|
$migrated++; |
|
} |
|
command_close_pipe($fh, $ctx); |
|
print STDERR "Done migrating from a git-svn v1 layout\n"; |
|
$migrated; |
|
} |
|
|
|
sub read_old_urls { |
|
my ($l_map, $pfx, $path) = @_; |
|
my @dir; |
|
foreach (<$path/*>) { |
|
if (-r "$_/info/url") { |
|
$pfx .= '/' if $pfx && $pfx !~ m!/$!; |
|
my $ref_id = $pfx . basename $_; |
|
my $url = ::file_to_s("$_/info/url"); |
|
$l_map->{$ref_id} = $url; |
|
} elsif (-d $_) { |
|
push @dir, $_; |
|
} |
|
} |
|
foreach (@dir) { |
|
my $x = $_; |
|
$x =~ s!^\Q$ENV{GIT_DIR}\E/svn/!!o; |
|
read_old_urls($l_map, $x, $_); |
|
} |
|
} |
|
|
|
sub migrate_from_v2 { |
|
my @cfg = command(qw/config -l/); |
|
return if grep /^svn-remote\..+\.url=/, @cfg; |
|
my %l_map; |
|
read_old_urls(\%l_map, '', "$ENV{GIT_DIR}/svn"); |
|
my $migrated = 0; |
|
|
|
require Git::SVN; |
|
foreach my $ref_id (sort keys %l_map) { |
|
eval { Git::SVN->init($l_map{$ref_id}, '', undef, $ref_id) }; |
|
if ($@) { |
|
Git::SVN->init($l_map{$ref_id}, '', $ref_id, $ref_id); |
|
} |
|
$migrated++; |
|
} |
|
$migrated; |
|
} |
|
|
|
sub minimize_connections { |
|
require Git::SVN; |
|
require Git::SVN::Ra; |
|
|
|
my $r = Git::SVN::read_all_remotes(); |
|
my $new_urls = {}; |
|
my $root_repos = {}; |
|
foreach my $repo_id (keys %$r) { |
|
my $url = $r->{$repo_id}->{url} or next; |
|
my $fetch = $r->{$repo_id}->{fetch} or next; |
|
my $ra = Git::SVN::Ra->new($url); |
|
|
|
# skip existing cases where we already connect to the root |
|
if (($ra->url eq $ra->{repos_root}) || |
|
($ra->{repos_root} eq $repo_id)) { |
|
$root_repos->{$ra->url} = $repo_id; |
|
next; |
|
} |
|
|
|
my $root_ra = Git::SVN::Ra->new($ra->{repos_root}); |
|
my $root_path = $ra->url; |
|
$root_path =~ s#^\Q$ra->{repos_root}\E(/|$)##; |
|
foreach my $path (keys %$fetch) { |
|
my $ref_id = $fetch->{$path}; |
|
my $gs = Git::SVN->new($ref_id, $repo_id, $path); |
|
|
|
# make sure we can read when connecting to |
|
# a higher level of a repository |
|
my ($last_rev, undef) = $gs->last_rev_commit; |
|
if (!defined $last_rev) { |
|
$last_rev = eval { |
|
$root_ra->get_latest_revnum; |
|
}; |
|
next if $@; |
|
} |
|
my $new = $root_path; |
|
$new .= length $path ? "/$path" : ''; |
|
eval { |
|
$root_ra->get_log([$new], $last_rev, $last_rev, |
|
0, 0, 1, sub { }); |
|
}; |
|
next if $@; |
|
$new_urls->{$ra->{repos_root}}->{$new} = |
|
{ ref_id => $ref_id, |
|
old_repo_id => $repo_id, |
|
old_path => $path }; |
|
} |
|
} |
|
|
|
my @emptied; |
|
foreach my $url (keys %$new_urls) { |
|
# see if we can re-use an existing [svn-remote "repo_id"] |
|
# instead of creating a(n ugly) new section: |
|
my $repo_id = $root_repos->{$url} || $url; |
|
|
|
my $fetch = $new_urls->{$url}; |
|
foreach my $path (keys %$fetch) { |
|
my $x = $fetch->{$path}; |
|
Git::SVN->init($url, $path, $repo_id, $x->{ref_id}); |
|
my $pfx = "svn-remote.$x->{old_repo_id}"; |
|
|
|
my $old_fetch = quotemeta("$x->{old_path}:". |
|
"$x->{ref_id}"); |
|
command_noisy(qw/config --unset/, |
|
"$pfx.fetch", '^'. $old_fetch . '$'); |
|
delete $r->{$x->{old_repo_id}}-> |
|
{fetch}->{$x->{old_path}}; |
|
if (!keys %{$r->{$x->{old_repo_id}}->{fetch}}) { |
|
command_noisy(qw/config --unset/, |
|
"$pfx.url"); |
|
push @emptied, $x->{old_repo_id} |
|
} |
|
} |
|
} |
|
if (@emptied) { |
|
my $file = $ENV{GIT_CONFIG} || "$ENV{GIT_DIR}/config"; |
|
print STDERR <<EOF; |
|
The following [svn-remote] sections in your config file ($file) are empty |
|
and can be safely removed: |
|
EOF |
|
print STDERR "[svn-remote \"$_\"]\n" foreach @emptied; |
|
} |
|
} |
|
|
|
sub migration_check { |
|
migrate_from_v0(); |
|
migrate_from_v1(); |
|
migrate_from_v2(); |
|
minimize_connections() if $_minimize; |
|
} |
|
|
|
1;
|
|
|