8
8

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

.gitignoreを後から設定するときのベストプラクティス

Last updated at Posted at 2021-07-13

すでにGitの管理対象になっているファイルを後から除外したい場合には、.gitignoreを追加、追記するだけでは除外されず、git rm --cached <file-name>などで、管理対象から除外してあげないといけません。
しかし、除外ファイル数が多くなったり、うっかりgit rm --cachedを忘れて管理対象から外れていないファイルがあったりした場合には少々手間です。

よく紹介されているgit rm --cached <file-name>だと、除外したいファイルを一つ一つ指定していないといけませんし、git rm --cached .だと、管理対象のファイルが一旦全て除外されてしまうため、その後にgit addしてまたステージングエリアに上げないといけないので二度手間です。

後から.gitignoreする方法に関してはすでに色々な情報がありますが、なかなか万能な方法が見つからなかったので僕なりのベストプラクティスを考えてみました。

動作環境

この記事は、以下のバージョンのGitで動作確認しています。

git version 2.32.0

まずは結論から

.gitignoreを追加、または追記した後に以下のコマンドを実行することで、管理対象から新しく除外したいファイルのみを除外することができます。

git rm --cached `git ls-files -ci --exclude-standard`

なお、ローカルのファイル自体が削除されるわけではありません。

上記の方法だと.gitignore以外にも、.git/info/excludeやグローバルな設定などの一般的なGitの除外設定も含まれるので、リポジトリの.gitignoreだけを対象に除外したい場合は以下のようにします。

git rm --cached `git ls-files -ci --exclude-from=.gitignore`

これで、ワンコマンドで管理対象になっているファイルの中から、.gitignoreで指定されているファイルだけを除外することができます。

特に理由がなければ--exclude-standardの方でいいかなと思います。

.gitignoreに差分がある場合は、そちらも忘れずにステージングエリアに上げてからコミットしましょう。

とりあえず目的が達成できればいいよって方は、エイリアスにしておくと便利まで飛ばしてもらって、紹介しているエイリアスを設定して使ってもらうと便利です。

理解してないコマンドは叩きたくない!という方のために、以降から詳しく解説していきます。

背景

対象ファイルが把握できる数ならgit rm --cached <file-name>でも問題ないですが、例えば後から新規に.gitignoreのテンプレを追加したりして、除外対象のファイルが多くなってくると面倒です。

また、過去の遺物や引き継ぎなどで.gitignoreに記載はあるものの、管理対象から除外されていないファイルがあるリポジトリを整備する際などには一気に除外したいものですが、前述の通りgit rm --cached .では管理対象のファイルが一旦全て除外されてしまうので、git addしてステージングエリアに上げなおす手間がかかります。

今回の方法だと、ワンコマンドで目的のファイルだけピンポイントに除外できるのでオススメです。

引き継ぎリポジトリとかだとメンテが行き届いていなくて、除外対象のファイルが大量に出てくる場合があるので、エイリアスにしておくと便利セクションではそんなときに便利な、除外する前に除外されるファイルを確認する方法も紹介しています。

できるだけGitの公式リファレンスのリンクをつけているので、より詳細な説明はそちらも参考にしてみてください。(リンクは全てGitバージョン2.32.0のリファレンスへのリンクになっています。)

コマンドの説明

使用している各コマンドの詳細を説明します。
ここからは便宜上、--exclude-standard--exclude-from=.gitignoreのexclude系のオプション群を、まとめて--exclude*と表記します。

git rm --cached **<**file1 file2 ...>

指定されたファイルを管理対象から除外します。スペース区切りで複数ファイルの指定が可能です。
ローカルのファイルは削除されずに残ります。
--cachedオプションを付けないとファイルごと削除されてしまうので注意です。

今回の方法ではgit ls-files -ci --exclude*で出力されたファイルを、管理対象から除外しているというわけです。

git ls-files

オプションなしで実行することで、Gitの管理対象になっているファイルを表示します。
暗黙的に-cオプションがデフォルトで指定されています。
-oオプションを指定することで、逆にGitの管理対象になっていないファイルを表示することができます。

オプション説明

git ls-files -ci --exclude*で、除外対象ファイルの特定に使用しているgit ls-filesコマンドのオプションについて説明します。

-c, --cached

出力の中でキャッシュされているファイルのみを表示します。
キャッシュされているとは、Gitのインデックスに登録されている、つまり管理対象であるということで、要は管理対象のファイルが表示されるということです。
これはls-filesのデフォルトの挙動となっていますが、-iオプションを使用する際には明示的に指定してあげないといけません。

-i, --ignored

無視されたファイルのみを出力に表示します。
少なくとも1つの--exclude*オプションを使って無視ルールを指定する必要があります。

このオプションは--exclude*オプションとセットで使う必要があるため、否定の否定みたいで少しわかりにくいですが、--exclude*で無視ルールとして指定されたパターンに一致するファイル(無視されたファイル)が(逆に)出力されるというイメージです。

また、-oまたは-cオプションも一緒に使用する必要があります。
指定しないと以下のエラーが出ます。

fatal: ls-files -i must be used with either -o or -c

--exclude*系のオプションは-oまたは-iオプションが指定されている場合にのみ利用できるようです。

git ls-files can use a list of "exclude patterns" when traversing the directory tree and finding files to show when the flags --others or --ignored are specified. gitignore[5] specifies the format of exclude patterns.

git ls-files -c --exclude-standardでも実行自体はできましたが、出力からexcludeパターンに指定されているファイルが除外されませんでした。
git ls-files -o --exclude-standardだと除外ルールが適用され、excludeパターンに指定されているファイルが出力から除外されました。

--exclude-from=**<**file>

fileから除外パターンを読み込んで、そのルールにしたがって出力から除外します。
今回はこのオプションに.gitignoreを渡しています。
出力から.gitignoreに指定のあるファイルのみを除外するイメージです。
前述の通り、--exclude*系のオプションは-oまたは-iオプションが指定されている場合にのみ利用できるようです。

--exclude-standard

標準的なGitの除外設定で指定されているファイルを出力から除外します。
.git/info/exclude、各ディレクトリのgitignore、ユーザーのグローバルな除外ファイルが対象のようです。

Add the standard Git exclusions: .git/info/exclude, .gitignore in each directory, and the user’s global exclusion file.


少しわかりにくいのが、-iオプションと--exclude*オプションを組み合わせたときに、**「excludeパターンにマッチするものだけが表示される」**というところです。冒頭のコマンドの一部を以下のように、オプション指定の順番を入れ替えてみるとわかりやすいかもしれません。

git ls-files -c --exclude-standard -i

つまり、git ls-files -cの出力の中で、--exclude*オプションによって、無視(除外)されたファイルを、-iオプションで表示させている、と考えると少しわかりやすいかもしれません。

エイリアスにしておくと便利

Gitのグローバル設定でaliasにしておくと便利です。

~/.gitconfig
# 以下を追記
[alias]
	# 除外設定されているファイルのうち、まだ管理対象から除外されていないファイルを表示します。
	unignored = ls-files -ci --exclude-standard
	# または
	# unignored = ls-files -ci --exclude-from=.gitignore

	# 除外設定されているファイルのうち、まだ管理対象から除外されていないファイルを除外します。
	cache-clean = !git rm --cached `git ls-files -ci --exclude-standard`
	# または
	# cache-clean = !git rm --cached `git ls-files -ci --exclude-from=.gitignore`

以下のようにして使います。

  1. .gitignoreを追加するなり追記するなりして、Git管理から除外したいファイルを設定する。

  2. .gitignoreに記載されているものの、まだ管理対象から除外されていないファイルを表示して確認する。

    git unignored
    
  3. 前の手順で確認したファイルを管理対象から除外する。

    git cache-clean
    
  4. コミットする時は、忘れずに.gitignoreの変更をステージングエリアに上げてからコミットする。

補足

  • unignoredは英単語として正しくないかもしれませんが、利便性を考慮してこのように表現してみました。
  • cache-cleanは文法的にはclean-cacheの方が正しいかもしれませんが、万が一git cleanが実行されてしまうと怖いので、このように表現しています。
  • cache-cleanにて、Gitのエイリアスではバッククォートがコマンド置換されないので、エイリアスの先頭に!をつけてシェルコマンドとして扱うようにしています。ただし、!のコマンドの実行ディレクトリはカレントディレクトリではなく、リポジトリの最上位ディレクトリになるというところには注意が必要です。

If the alias expansion is prefixed with an exclamation point, it will be treated as a shell command.

Note that shell commands will be executed from the top-level directory of a repository, which may not necessarily be the current directory.

何か気づいた点があればコメントいただけると嬉しいです🙌

参考

8
8
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
8
8

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?