33

gitリポジトリ内の特定のファイルの以前のバージョンをすべて取得したい。

checkoutコマンドを使用して特定のバージョンを1つ取得することは可能ですが、すべてが必要です。また、depthオプションを指定したgit cloneコマンドでは、サブフォルダー(「無効なリポジトリ名」)のクローンを作成できないようです。

それが可能かどうか、そしてどのように知っていますか?

ありがとうございました

4

6 に答える 6

40

OPはすべてのバージョンを取得したかったのですが、答えは得られませんでした。特にファイルに何百ものリビジョンがある場合(すべての提案は手動すぎます)。コメントで@Tobiasによって提案された唯一の半ば機能するソリューションですが、提案されたbashループはランダムな順序でファイルを構築し、リポジトリに対して使用すると何百もの空のファイルを生成します。その理由の1つは、「rev-list --all --objects」がさまざまなオブジェクトを一覧表示することでした(ツリーは含まれていますが、私たちの目的には役に立ちません)。

私はTobiasのソリューションから始め、カウンターを追加し、少しクリーンアップして、以下にリストされているbashスクリプトの形式でホイールを再発明することになりました。

スクリプトは次のようになります。

  • すべてのファイルバージョンを/tmp/all_versions_exportedに抽出します
  • 1つの引数を取る-gitリポジトリ内のファイルへの相対パス
  • 結果のファイル名に数値プレフィックスを付ける(ソート可能)
  • 結果ファイルで検査されたファイル名に言及します(リンゴをオレンジと区別するために:)
  • 結果のファイル名にコミット日を記載します(以下の出力例を参照)
  • 空の結果ファイルを作成しない

cat / usr / local / bin / git_export_all_file_versions

#!/bin/bash

# we'll write all git versions of the file to this folder:
EXPORT_TO=/tmp/all_versions_exported

# take relative path to the file to inspect
GIT_PATH_TO_FILE=$1

# ---------------- don't edit below this line --------------

USAGE="Please cd to the root of your git proj and specify path to file you with to inspect (example: $0 some/path/to/file)"

# check if got argument
if [ "${GIT_PATH_TO_FILE}" == "" ]; then
    echo "error: no arguments given. ${USAGE}" >&2
    exit 1
fi

# check if file exist
if [ ! -f ${GIT_PATH_TO_FILE} ]; then
    echo "error: File '${GIT_PATH_TO_FILE}' does not exist. ${USAGE}" >&2
    exit 1
fi

# extract just a filename from given relative path (will be used in result file names)
GIT_SHORT_FILENAME=$(basename $GIT_PATH_TO_FILE)

# create folder to store all revisions of the file
if [ ! -d ${EXPORT_TO} ]; then
    echo "creating folder: ${EXPORT_TO}"
    mkdir ${EXPORT_TO}
fi

## uncomment next line to clear export folder each time you run script
#rm ${EXPORT_TO}/*

# reset coutner
COUNT=0

# iterate all revisions
git rev-list --all --objects -- ${GIT_PATH_TO_FILE} | \
    cut -d ' ' -f1 | \
while read h; do \
     COUNT=$((COUNT + 1)); \
     COUNT_PRETTY=$(printf "%04d" $COUNT); \
     COMMIT_DATE=`git show $h | head -3 | grep 'Date:' | awk '{print $4"-"$3"-"$6}'`; \
     if [ "${COMMIT_DATE}" != "" ]; then \
         git cat-file -p ${h}:${GIT_PATH_TO_FILE} > ${EXPORT_TO}/${COUNT_PRETTY}.${COMMIT_DATE}.${h}.${GIT_SHORT_FILENAME};\
     fi;\
done    

# return success code
echo "result stored to ${EXPORT_TO}"
exit 0

使用例:
cd /home/myname/my-git-repo

git_export_all_file_versions docs/howto/readme.txt
    result stored to /tmp/all_versions_exported

ls /tmp/all_versions_exported
    0001.17-Oct-2016.ee0a1880ab815fd8f67bc4299780fc0b34f27b30.readme.txt
    0002.3-Oct-2016.d305158b94bedabb758ff1bb5e1ad74ed7ccd2c3.readme.txt
    0003.29-Sep-2016.7414a3de62529bfdd3cb1dd20ebc1a977793102f.readme.txt
    0004.28-Sep-2016.604cc0a34ec689606f7d3b2b5bbced1eece7483d.readme.txt
    0005.28-Sep-2016.198043c219c81d776c6d8a20e4f36bd6d8a57825.readme.txt
    0006.9-Sep-2016.5aea5191d4b86aec416b031cb84c2b78603a8b0f.readme.txt
    <and so on and on . . .>

編集:次のようなエラーが表示された場合:

致命的:有効なオブジェクト名ではありません3e93eba38b31b8b81905ceaa95eb47bbaed46494:readme.txt

これは、gitプロジェクトのルートフォルダーからではなく、スクリプトを開始したことを意味します。

于 2017-01-24T17:23:20.003 に答える
32

Dmitryが提供するスクリプトは実際には問題を解決しますが、いくつかの問題があり、自分のニーズにより適したものに適合させることができました。具体的には:

  1. git showデフォルトの日付形式設定が原因で、使用が中断されました。
  2. 結果を日付の逆順ではなく、日付順に並べ替えたいと思いました。
  3. リポジトリから削除されたファイルに対して実行できるようにしたかったのです。
  4. すべてのブランチですべてのリビジョンが必要ではありませんでした。HEADから到達可能なリビジョンが欲しかっただけです。
  5. gitリポジトリにない場合は、エラーが発生するようにしたかったのです。
  6. 特定のオプションを調整するためにスクリプトを編集する必要はありませんでした。
  7. それが機能する方法は非効率的でした。
  8. 出力ファイル名に番号を付ける必要はありませんでした。(適切にフォーマットされた日付は同じ目的を果たします。)
  9. より安全な「スペースのあるパス」の処理が必要でした

私の変更の最新バージョンは、私のgithubリポジトリで確認できます。または、この記事の執筆時点でのバージョンは次のとおりです。

#!/bin/sh

# based on script provided by Dmitry Shevkoplyas at http://stackoverflow.com/questions/12850030/git-getting-all-previous-version-of-a-specific-file-folder

set -e

if ! git rev-parse --show-toplevel >/dev/null 2>&1 ; then
    echo "Error: you must run this from within a git working directory" >&2
    exit 1
fi

if [ "$#" -lt 1 ] || [ "$#" -gt 2 ]; then
    echo "Usage: $0 <relative path to file> [<output directory>]" >&2
    exit 2
fi

FILE_PATH="$1"

EXPORT_TO=/tmp/all_versions_exported
if [ -n "$2" ]; then
    EXPORT_TO="$2"
fi

FILE_NAME="$(basename "$FILE_PATH")"

if [ ! -d "$EXPORT_TO" ]; then
    echo "Creating directory '$EXPORT_TO'"
    mkdir -p "$EXPORT_TO"
fi

echo "Writing files to '$EXPORT_TO'"
git log --diff-filter=d --date-order --reverse --format="%ad %H" --date=iso-strict "$FILE_PATH" | grep -v '^commit' | \
    while read LINE; do \
        COMMIT_DATE=`echo $LINE | cut -d ' ' -f 1`; \
        COMMIT_SHA=`echo $LINE | cut -d ' ' -f 2`; \
        printf '.' ; \
        git cat-file -p "$COMMIT_SHA:$FILE_PATH" > "$EXPORT_TO/$COMMIT_DATE.$COMMIT_SHA.$FILE_NAME" ; \
    done
echo

exit 0

出力の例:

$ git_export_all_file_versions bin/git_export_all_file_versions /tmp/stackoverflow/demo
Creating directory '/tmp/stackoverflow/demo'
Writing files to '/tmp/stackoverflow/demo'
...

$ ls -1 /tmp/stackoverflow/demo/
2017-05-02T15:52:52-04:00.c72640ed968885c3cc86812a2e1aabfbc2bc3b2a.git_export_all_file_versions
2017-05-02T16:58:56-04:00.bbbcff388d6f75572089964e3dc8d65a3bdf7817.git_export_all_file_versions
2017-05-02T17:05:50-04:00.67cbdeab97cd62813cec58d8e16d7c386c7dae86.git_export_all_file_versions
于 2017-05-02T21:13:33.057 に答える
8
git rev-list --all --objects -- path/to/file.txt

リポジトリパスに関連付けられているすべてのBLOBを一覧表示します

ファイルの特定のバージョンを取得するには

git cat-file -p commitid:path/to/file.txt

(commitidは何でもかまいません

  • シンボリック参照(ブランチ、タグ名、リモートも)
  • コミットハッシュ
  • HEAD〜3、branch1@{4}などのリビジョン仕様。
于 2012-10-11T23:25:22.423 に答える
0

古いバージョンのファイルは、を介してのみ利用できる場合がありますgit reflog。最近、インタラクティブなリベース中に誤って上書きしたためにログの一部ではなくなったコミットも含めて、すべてのコミットを掘り下げる必要がある状況になりました。

このRubyスクリプトを作成して、以前のすべてのバージョンのファイルを出力し、孤立したコミットを見つけました。不足しているファイルを追跡するために、これの出力をgrepするのは簡単でした。それが誰かを助けることを願っています。

#!/usr/bin/env ruby
path_to_file = ""
`git reflog`.split("\n").each do |log|
   puts commit = log.split(" ").first
   puts `git show #{commit}:#{path_to_file}`
   puts
 end

同じことが。でも実行できますgit log

于 2015-02-07T04:33:58.527 に答える
0

を使用できますgit blame <file>

こちらもご覧ください:https ://docs.github.com/en/github/managing-files-in-a-repository/managing-files-on-github/tracking-changes-in-a-file

于 2021-07-02T19:52:31.227 に答える
-2

ファイルのクローンを作成すると、ファイルのすべてのバージョンがすでにgitリポジトリにあります。特定のコミットのチェックアウトに関連付けられたブランチを作成できます。

git checkout -b branchname {commit#}

これは、変更を手動ですばやく汚く比較するのに十分な場合があります。

  • 支店へのチェックアウト
  • エディターバッファーにコピーする

gitの組み込みコマンドではありますが、関係するバージョンがいくつかあり、手動で少し気にしない場合は、これで問題ない可能性があります。

スクリプト化されたソリューションの場合、他の回答で提供された他のソリューションがすでにいくつかあります。

于 2012-10-11T23:23:56.710 に答える