diff options
| -rw-r--r-- | Makefile | 2 | ||||
| -rwxr-xr-x | git-remote.perl | 277 | 
2 files changed, 278 insertions, 1 deletions
| @@ -179,7 +179,7 @@ SCRIPT_SH = \  SCRIPT_PERL = \  	git-add--interactive.perl \  	git-archimport.perl git-cvsimport.perl git-relink.perl \ -	git-cvsserver.perl \ +	git-cvsserver.perl git-remote.perl \  	git-svnimport.perl git-cvsexportcommit.perl \  	git-send-email.perl git-svn.perl diff --git a/git-remote.perl b/git-remote.perl new file mode 100755 index 0000000000..059c141b68 --- /dev/null +++ b/git-remote.perl @@ -0,0 +1,277 @@ +#!/usr/bin/perl -w + +use Git; +my $git = Git->repository(); + +sub add_remote_config { +	my ($hash, $name, $what, $value) = @_; +	if ($what eq 'url') { +		if (exists $hash->{$name}{'URL'}) { +			print STDERR "Warning: more than one remote.$name.url\n"; +		} +		$hash->{$name}{'URL'} = $value; +	} +	elsif ($what eq 'fetch') { +		$hash->{$name}{'FETCH'} ||= []; +		push @{$hash->{$name}{'FETCH'}}, $value; +	} +	if (!exists $hash->{$name}{'SOURCE'}) { +		$hash->{$name}{'SOURCE'} = 'config'; +	} +} + +sub add_remote_remotes { +	my ($hash, $file, $name) = @_; + +	if (exists $hash->{$name}) { +		$hash->{$name}{'WARNING'} = 'ignored due to config'; +		return; +	} + +	my $fh; +	if (!open($fh, '<', $file)) { +		print STDERR "Warning: cannot open $file\n"; +		return; +	} +	my $it = { 'SOURCE' => 'remotes' }; +	$hash->{$name} = $it; +	while (<$fh>) { +		chomp; +		if (/^URL:\s*(.*)$/) { +			# Having more than one is Ok -- it is used for push. +			if (! exists $it->{'URL'}) { +				$it->{'URL'} = $1; +			} +		} +		elsif (/^Push:\s*(.*)$/) { +			; # later +		} +		elsif (/^Pull:\s*(.*)$/) { +			$it->{'FETCH'} ||= []; +			push @{$it->{'FETCH'}}, $1; +		} +		elsif (/^\#/) { +			; # ignore +		} +		else { +			print STDERR "Warning: funny line in $file: $_\n"; +		} +	} +	close($fh); +} + +sub list_remote { +	my ($git) = @_; +	my %seen = (); +	my @remotes = eval { +		$git->command(qw(repo-config --get-regexp), '^remote\.'); +	}; +	for (@remotes) { +		if (/^remote\.([^.]*)\.(\S*)\s+(.*)$/) { +			add_remote_config(\%seen, $1, $2, $3); +		} +	} + +	my $dir = $git->repo_path() . "/remotes"; +	if (opendir(my $dh, $dir)) { +		local $_; +		while ($_ = readdir($dh)) { +			chomp; +			next if (! -f "$dir/$_" || ! -r _); +			add_remote_remotes(\%seen, "$dir/$_", $_); +		} +	} + +	return \%seen; +} + +sub add_branch_config { +	my ($hash, $name, $what, $value) = @_; +	if ($what eq 'remote') { +		if (exists $hash->{$name}{'REMOTE'}) { +			print STDERR "Warning: more than one branch.$name.remote\n"; +		} +		$hash->{$name}{'REMOTE'} = $value; +	} +	elsif ($what eq 'merge') { +		$hash->{$name}{'MERGE'} ||= []; +		push @{$hash->{$name}{'MERGE'}}, $value; +	} +} + +sub list_branch { +	my ($git) = @_; +	my %seen = (); +	my @branches = eval { +		$git->command(qw(repo-config --get-regexp), '^branch\.'); +	}; +	for (@branches) { +		if (/^branch\.([^.]*)\.(\S*)\s+(.*)$/) { +			add_branch_config(\%seen, $1, $2, $3); +		} +	} + +	return \%seen; +} + +my $remote = list_remote($git); +my $branch = list_branch($git); + +sub update_ls_remote { +	my ($harder, $info) = @_; + +	return if (($harder == 0) || +		   (($harder == 1) && exists $info->{'LS_REMOTE'})); + +	my @ref = map { +		s|^[0-9a-f]{40}\s+refs/heads/||; +		$_; +	} $git->command(qw(ls-remote --heads), $info->{'URL'}); +	$info->{'LS_REMOTE'} = \@ref; +} + +sub show_wildcard_mapping { +	my ($forced, $ours, $ls) = @_; +	my %refs; +	for (@$ls) { +		$refs{$_} = 01; # bit #0 to say "they have" +	} +	for ($git->command('for-each-ref', "refs/remotes/$ours")) { +		chomp; +		next unless (s|^[0-9a-f]{40}\s[a-z]+\srefs/remotes/$ours/||); +		next if ($_ eq 'HEAD'); +		$refs{$_} ||= 0; +		$refs{$_} |= 02; # bit #1 to say "we have" +	} +	my (@new, @stale, @tracked); +	for (sort keys %refs) { +		my $have = $refs{$_}; +		if ($have == 1) { +			push @new, $_; +		} +		elsif ($have == 2) { +			push @stale, $_; +		} +		elsif ($have == 3) { +			push @tracked, $_; +		} +	} +	if (@new) { +		print "  New remote branches (next fetch will store in remotes/$ours)\n"; +		print "    @new\n"; +	} +	if (@stale) { +		print "  Stale tracking branches in remotes/$ours (you'd better remove them)\n"; +		print "    @stale\n"; +	} +	if (@tracked) { +		print "  Tracked remote branches\n"; +		print "    @tracked\n"; +	} +} + +sub show_mapping { +	my ($name, $info) = @_; +	my $fetch = $info->{'FETCH'}; +	my $ls = $info->{'LS_REMOTE'}; +	my (@stale, @tracked); + +	for (@$fetch) { +		next unless (/(\+)?([^:]+):(.*)/); +		my ($forced, $theirs, $ours) = ($1, $2, $3); +		if ($theirs eq 'refs/heads/*' && +		    $ours =~ /^refs\/remotes\/(.*)\/\*$/) { +			# wildcard mapping +			show_wildcard_mapping($forced, $1, $ls); +		} +		elsif ($theirs =~ /\*/ || $ours =~ /\*/) { +			print STDERR "Warning: unrecognized mapping in remotes.$name.fetch: $_\n"; +		} +		elsif ($theirs =~ s|^refs/heads/||) { +			if (!grep { $_ eq $theirs } @$ls) { +				push @stale, $theirs; +			} +			elsif ($ours ne '') { +				push @tracked, $theirs; +			} +		} +	} +	if (@stale) { +		print "  Stale tracking branches in remotes/$name (you'd better remove them)\n"; +		print "    @stale\n"; +	} +	if (@tracked) { +		print "  Tracked remote branches\n"; +		print "    @tracked\n"; +	} +} + +sub show_remote { +	my ($name, $ls_remote) = @_; +	if (!exists $remote->{$name}) { +		print STDERR "No such remote $name\n"; +		return; +	} +	my $info = $remote->{$name}; +	update_ls_remote($ls_remote, $info); + +	print "* remote $name\n"; +	print "  URL: $info->{'URL'}\n"; +	for my $branchname (sort keys %$branch) { +		next if ($branch->{$branchname}{'REMOTE'} ne $name); +		my @merged = map { +			s|^refs/heads/||; +			$_; +		} split(' ',"@{$branch->{$branchname}{'MERGE'}}"); +		next unless (@merged); +		print "  Remote branch(es) merged with 'git pull' while on branch $branchname\n"; +		print "    @merged\n"; +	} +	if ($info->{'LS_REMOTE'}) { +		show_mapping($name, $info); +	} +} + +sub add_remote { +	my ($name, $url) = @_; +	if (exists $remote->{$name}) { +		print STDERR "remote $name already exists.\n"; +		exit(1); +	} +	$git->command('repo-config', "remote.$name.url", $url); +	$git->command('repo-config', "remote.$name.fetch", +		      "+refs/heads/*:refs/remotes/$name/*"); +} + +if (!@ARGV) { +	for (sort keys %$remote) { +		print "$_\n"; +	} +} +elsif ($ARGV[0] eq 'show') { +	my $ls_remote = 1; +	my $i; +	for ($i = 1; $i < @ARGV; $i++) { +		if ($ARGV[$i] eq '-n') { +			$ls_remote = 0; +		} +		else { +			last; +		} +	} +	if ($i >= @ARGV) { +		print STDERR "Usage: git remote show <remote>\n"; +		exit(1); +	} +	for (; $i < @ARGV; $i++) { +		show_remote($ARGV[$i], $ls_remote); +	} +} +elsif ($ARGV[0] eq 'add') { +	if (@ARGV != 3) { +		print STDERR "Usage: git remote add <name> <url>\n"; +		exit(1); +	} +	add_remote($ARGV[1], $ARGV[2]); +} + | 
