gitweb: group remote heads by remote
In remote and summary view, display a block for each remote, with the fetch and push URL(s) as well as the list of the remote heads. In summary view, if the number of remotes is higher than a prescribed limit, only display the first <limit> remotes and their fetch and push urls, without any heads information and without grouping. Signed-off-by: Giuseppe Bilotta <giuseppe.bilotta@gmail.com> Acked-by: Jakub Narebski <jnareb@gmail.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
		
				
					committed by
					
						
						Junio C Hamano
					
				
			
			
				
	
			
			
			
						parent
						
							b891d52a64
						
					
				
				
					commit
					9d0d42f345
				
			@ -2772,6 +2772,44 @@ sub git_get_last_activity {
 | 
				
			|||||||
	return (undef, undef);
 | 
						return (undef, undef);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Implementation note: when a single remote is wanted, we cannot use 'git
 | 
				
			||||||
 | 
					# remote show -n' because that command always work (assuming it's a remote URL
 | 
				
			||||||
 | 
					# if it's not defined), and we cannot use 'git remote show' because that would
 | 
				
			||||||
 | 
					# try to make a network roundtrip. So the only way to find if that particular
 | 
				
			||||||
 | 
					# remote is defined is to walk the list provided by 'git remote -v' and stop if
 | 
				
			||||||
 | 
					# and when we find what we want.
 | 
				
			||||||
 | 
					sub git_get_remotes_list {
 | 
				
			||||||
 | 
						my $wanted = shift;
 | 
				
			||||||
 | 
						my %remotes = ();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						open my $fd, '-|' , git_cmd(), 'remote', '-v';
 | 
				
			||||||
 | 
						return unless $fd;
 | 
				
			||||||
 | 
						while (my $remote = <$fd>) {
 | 
				
			||||||
 | 
							chomp $remote;
 | 
				
			||||||
 | 
							$remote =~ s!\t(.*?)\s+\((\w+)\)$!!;
 | 
				
			||||||
 | 
							next if $wanted and not $remote eq $wanted;
 | 
				
			||||||
 | 
							my ($url, $key) = ($1, $2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							$remotes{$remote} ||= { 'heads' => () };
 | 
				
			||||||
 | 
							$remotes{$remote}{$key} = $url;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						close $fd or return;
 | 
				
			||||||
 | 
						return wantarray ? %remotes : \%remotes;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Takes a hash of remotes as first parameter and fills it by adding the
 | 
				
			||||||
 | 
					# available remote heads for each of the indicated remotes.
 | 
				
			||||||
 | 
					sub fill_remote_heads {
 | 
				
			||||||
 | 
						my $remotes = shift;
 | 
				
			||||||
 | 
						my @heads = map { "remotes/$_" } keys %$remotes;
 | 
				
			||||||
 | 
						my @remoteheads = git_get_heads_list(undef, @heads);
 | 
				
			||||||
 | 
						foreach my $remote (keys %$remotes) {
 | 
				
			||||||
 | 
							$remotes->{$remote}{'heads'} = [ grep {
 | 
				
			||||||
 | 
								$_->{'name'} =~ s!^$remote/!!
 | 
				
			||||||
 | 
								} @remoteheads ];
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
sub git_get_references {
 | 
					sub git_get_references {
 | 
				
			||||||
	my $type = shift || "";
 | 
						my $type = shift || "";
 | 
				
			||||||
	my %refs;
 | 
						my %refs;
 | 
				
			||||||
@ -5051,6 +5089,101 @@ sub git_heads_body {
 | 
				
			|||||||
	print "</table>\n";
 | 
						print "</table>\n";
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Display a single remote block
 | 
				
			||||||
 | 
					sub git_remote_block {
 | 
				
			||||||
 | 
						my ($remote, $rdata, $limit, $head) = @_;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						my $heads = $rdata->{'heads'};
 | 
				
			||||||
 | 
						my $fetch = $rdata->{'fetch'};
 | 
				
			||||||
 | 
						my $push = $rdata->{'push'};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						my $urls_table = "<table class=\"projects_list\">\n" ;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (defined $fetch) {
 | 
				
			||||||
 | 
							if ($fetch eq $push) {
 | 
				
			||||||
 | 
								$urls_table .= format_repo_url("URL", $fetch);
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								$urls_table .= format_repo_url("Fetch URL", $fetch);
 | 
				
			||||||
 | 
								$urls_table .= format_repo_url("Push URL", $push) if defined $push;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						} elsif (defined $push) {
 | 
				
			||||||
 | 
							$urls_table .= format_repo_url("Push URL", $push);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							$urls_table .= format_repo_url("", "No remote URL");
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						$urls_table .= "</table>\n";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						my $dots;
 | 
				
			||||||
 | 
						if (defined $limit && $limit < @$heads) {
 | 
				
			||||||
 | 
							$dots = $cgi->a({-href => href(action=>"remotes", hash=>$remote)}, "...");
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						print $urls_table;
 | 
				
			||||||
 | 
						git_heads_body($heads, $head, 0, $limit, $dots);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Display a list of remote names with the respective fetch and push URLs
 | 
				
			||||||
 | 
					sub git_remotes_list {
 | 
				
			||||||
 | 
						my ($remotedata, $limit) = @_;
 | 
				
			||||||
 | 
						print "<table class=\"heads\">\n";
 | 
				
			||||||
 | 
						my $alternate = 1;
 | 
				
			||||||
 | 
						my @remotes = sort keys %$remotedata;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						my $limited = $limit && $limit < @remotes;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						$#remotes = $limit - 1 if $limited;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						while (my $remote = shift @remotes) {
 | 
				
			||||||
 | 
							my $rdata = $remotedata->{$remote};
 | 
				
			||||||
 | 
							my $fetch = $rdata->{'fetch'};
 | 
				
			||||||
 | 
							my $push = $rdata->{'push'};
 | 
				
			||||||
 | 
							if ($alternate) {
 | 
				
			||||||
 | 
								print "<tr class=\"dark\">\n";
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								print "<tr class=\"light\">\n";
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							$alternate ^= 1;
 | 
				
			||||||
 | 
							print "<td>" .
 | 
				
			||||||
 | 
							      $cgi->a({-href=> href(action=>'remotes', hash=>$remote),
 | 
				
			||||||
 | 
								       -class=> "list name"},esc_html($remote)) .
 | 
				
			||||||
 | 
							      "</td>";
 | 
				
			||||||
 | 
							print "<td class=\"link\">" .
 | 
				
			||||||
 | 
							      (defined $fetch ? $cgi->a({-href=> $fetch}, "fetch") : "fetch") .
 | 
				
			||||||
 | 
							      " | " .
 | 
				
			||||||
 | 
							      (defined $push ? $cgi->a({-href=> $push}, "push") : "push") .
 | 
				
			||||||
 | 
							      "</td>";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							print "</tr>\n";
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if ($limited) {
 | 
				
			||||||
 | 
							print "<tr>\n" .
 | 
				
			||||||
 | 
							      "<td colspan=\"3\">" .
 | 
				
			||||||
 | 
							      $cgi->a({-href => href(action=>"remotes")}, "...") .
 | 
				
			||||||
 | 
							      "</td>\n" . "</tr>\n";
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						print "</table>";
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Display remote heads grouped by remote, unless there are too many
 | 
				
			||||||
 | 
					# remotes, in which case we only display the remote names
 | 
				
			||||||
 | 
					sub git_remotes_body {
 | 
				
			||||||
 | 
						my ($remotedata, $limit, $head) = @_;
 | 
				
			||||||
 | 
						if ($limit and $limit < keys %$remotedata) {
 | 
				
			||||||
 | 
							git_remotes_list($remotedata, $limit);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							fill_remote_heads($remotedata);
 | 
				
			||||||
 | 
							while (my ($remote, $rdata) = each %$remotedata) {
 | 
				
			||||||
 | 
								git_print_section({-class=>"remote", -id=>$remote},
 | 
				
			||||||
 | 
									["remotes", $remote, $remote], sub {
 | 
				
			||||||
 | 
										git_remote_block($remote, $rdata, $limit, $head);
 | 
				
			||||||
 | 
									});
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
sub git_search_grep_body {
 | 
					sub git_search_grep_body {
 | 
				
			||||||
	my ($commitlist, $from, $to, $extra) = @_;
 | 
						my ($commitlist, $from, $to, $extra) = @_;
 | 
				
			||||||
	$from = 0 unless defined $from;
 | 
						$from = 0 unless defined $from;
 | 
				
			||||||
@ -5197,7 +5330,7 @@ sub git_summary {
 | 
				
			|||||||
	# there are more ...
 | 
						# there are more ...
 | 
				
			||||||
	my @taglist  = git_get_tags_list(16);
 | 
						my @taglist  = git_get_tags_list(16);
 | 
				
			||||||
	my @headlist = git_get_heads_list(16);
 | 
						my @headlist = git_get_heads_list(16);
 | 
				
			||||||
	my @remotelist = $remote_heads ? git_get_heads_list(16, 'remotes') : ();
 | 
						my %remotedata = $remote_heads ? git_get_remotes_list() : ();
 | 
				
			||||||
	my @forklist;
 | 
						my @forklist;
 | 
				
			||||||
	my $check_forks = gitweb_check_feature('forks');
 | 
						my $check_forks = gitweb_check_feature('forks');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -5275,11 +5408,9 @@ sub git_summary {
 | 
				
			|||||||
		               $cgi->a({-href => href(action=>"heads")}, "..."));
 | 
							               $cgi->a({-href => href(action=>"heads")}, "..."));
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (@remotelist) {
 | 
						if (%remotedata) {
 | 
				
			||||||
		git_print_header_div('remotes');
 | 
							git_print_header_div('remotes');
 | 
				
			||||||
		git_heads_body(\@remotelist, $head, 0, 15,
 | 
							git_remotes_body(\%remotedata, 15, $head);
 | 
				
			||||||
		               $#remotelist <= 15 ? undef :
 | 
					 | 
				
			||||||
		               $cgi->a({-href => href(action=>"remotes")}, "..."));
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (@forklist) {
 | 
						if (@forklist) {
 | 
				
			||||||
@ -5596,6 +5727,7 @@ sub git_heads {
 | 
				
			|||||||
	git_footer_html();
 | 
						git_footer_html();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# used both for single remote view and for list of all the remotes
 | 
				
			||||||
sub git_remotes {
 | 
					sub git_remotes {
 | 
				
			||||||
	gitweb_check_feature('remote_heads')
 | 
						gitweb_check_feature('remote_heads')
 | 
				
			||||||
		or die_error(403, "Remote heads view is disabled");
 | 
							or die_error(403, "Remote heads view is disabled");
 | 
				
			||||||
@ -5603,32 +5735,26 @@ sub git_remotes {
 | 
				
			|||||||
	my $head = git_get_head_hash($project);
 | 
						my $head = git_get_head_hash($project);
 | 
				
			||||||
	my $remote = $input_params{'hash'};
 | 
						my $remote = $input_params{'hash'};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	my @remotelist;
 | 
						my $remotedata = git_get_remotes_list($remote);
 | 
				
			||||||
 | 
						die_error(500, "Unable to get remote information") unless defined $remotedata;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (defined $remote) {
 | 
						unless (%$remotedata) {
 | 
				
			||||||
		# only display the heads in a given remote, stripping the
 | 
							die_error(404, defined $remote ?
 | 
				
			||||||
		# remote name which is already visible elsewhere
 | 
								"Remote $remote not found" :
 | 
				
			||||||
		@remotelist = map {
 | 
								"No remotes found");
 | 
				
			||||||
			my $ref = $_ ;
 | 
					 | 
				
			||||||
			$ref->{'name'} =~ s!^$remote/!!;
 | 
					 | 
				
			||||||
			$ref
 | 
					 | 
				
			||||||
		} git_get_heads_list(undef, "remotes/$remote");
 | 
					 | 
				
			||||||
	} else {
 | 
					 | 
				
			||||||
		@remotelist = git_get_heads_list(undef, 'remotes');
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	git_header_html(undef, undef, -action_extra => $remote);
 | 
						git_header_html(undef, undef, -action_extra => $remote);
 | 
				
			||||||
	git_print_page_nav('', '',  $head, undef, $head,
 | 
						git_print_page_nav('', '',  $head, undef, $head,
 | 
				
			||||||
		format_ref_views($remote ? '' : 'remotes'));
 | 
							format_ref_views($remote ? '' : 'remotes'));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fill_remote_heads($remotedata);
 | 
				
			||||||
	if (defined $remote) {
 | 
						if (defined $remote) {
 | 
				
			||||||
		git_print_header_div('remotes', "$remote remote for $project");
 | 
							git_print_header_div('remotes', "$remote remote for $project");
 | 
				
			||||||
 | 
							git_remote_block($remote, $remotedata->{$remote}, undef, $head);
 | 
				
			||||||
	} else {
 | 
						} else {
 | 
				
			||||||
		git_print_header_div('summary', "$project remotes");
 | 
							git_print_header_div('summary', "$project remotes");
 | 
				
			||||||
	}
 | 
							git_remotes_body($remotedata, undef, $head);
 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (@remotelist) {
 | 
					 | 
				
			||||||
		git_heads_body(\@remotelist, $head);
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	git_footer_html();
 | 
						git_footer_html();
 | 
				
			||||||
 | 
				
			|||||||
@ -573,6 +573,12 @@ div.binary {
 | 
				
			|||||||
	font-style: italic;
 | 
						font-style: italic;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					div.remote {
 | 
				
			||||||
 | 
						margin: .5em;
 | 
				
			||||||
 | 
						border: 1px solid #d9d8d1;
 | 
				
			||||||
 | 
						display: inline-block;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Style definition generated by highlight 2.4.5, http://www.andre-simon.de/ */
 | 
					/* Style definition generated by highlight 2.4.5, http://www.andre-simon.de/ */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Highlighting theme definition: */
 | 
					/* Highlighting theme definition: */
 | 
				
			||||||
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user