1. ホーム
  2. git

[解決済み] この blob があるのはどのコミットですか?

2022-04-22 21:51:15

質問

ある blob のハッシュが与えられたとき、その blob をツリー内に持つコミットのリストを取得する方法はありますか?

どのように解決するのですか?

次の両方のスクリプトは、blobのSHA1を最初の引数として受け取り、その後に、オプションで、以下のような引数を受け取ります。 git log が理解できるようになります。例 --all を使えば、現在のブランチだけでなく、すべてのブランチを検索することができます。 -g を使えば、reflogの中を検索することもできますし、その他何でも構いません。

これはシェルスクリプトで、短く、甘いですが、遅いです。

#!/bin/sh
obj_name="$1"
shift
git log "$@" --pretty=tformat:'%T %h %s' \
| while read tree commit subject ; do
    if git ls-tree -r $tree | grep -q "$obj_name" ; then
        echo $commit "$subject"
    fi
done

そして、Perlによる最適化版。まだかなり短いが、ずっと速い。

#!/usr/bin/perl
use 5.008;
use strict;
use Memoize;

my $obj_name;

sub check_tree {
    my ( $tree ) = @_;
    my @subtree;

    {
        open my $ls_tree, '-|', git => 'ls-tree' => $tree
            or die "Couldn't open pipe to git-ls-tree: $!\n";

        while ( <$ls_tree> ) {
            /\A[0-7]{6} (\S+) (\S+)/
                or die "unexpected git-ls-tree output";
            return 1 if $2 eq $obj_name;
            push @subtree, $2 if $1 eq 'tree';
        }
    }

    check_tree( $_ ) && return 1 for @subtree;

    return;
}

memoize 'check_tree';

die "usage: git-find-blob <blob> [<git-log arguments ...>]\n"
    if not @ARGV;

my $obj_short = shift @ARGV;
$obj_name = do {
    local $ENV{'OBJ_NAME'} = $obj_short;
     `git rev-parse --verify \$OBJ_NAME`;
} or die "Couldn't parse $obj_short: $!\n";
chomp $obj_name;

open my $log, '-|', git => log => @ARGV, '--pretty=format:%T %h %s'
    or die "Couldn't open pipe to git-log: $!\n";

while ( <$log> ) {
    chomp;
    my ( $tree, $commit, $subject ) = split " ", $_, 3;
    print "$commit $subject\n" if check_tree( $tree );
}