gpt4 book ai didi

git - 如何设置 git 驱动程序以在 merge 时忽略文件夹

转载 作者:IT王子 更新时间:2023-10-29 00:40:42 24 4
gpt4 key购买 nike

我已经进行了广泛的搜索,阅读了很多 SO 问题和解决方案并尝试了不同的方法,但我一直无法做我想做的事,这很简单。

我有 master 分支,所有主要代码都驻留在其中,还有 design 分支,rails 应用程序的布局由设计团队构建。他们在公共(public)文件夹中添加了一个名为“photoshop”的文件夹,以将图像源也置于版本控制之下。但是我不希望这个文件夹在 merge 到 master 分支时被复制,因为,好吧,它不是必需的。

显然,实现此目的的方法是通过 merge 驱动程序。所以,我创建了“忽略”驱动程序:

[merge "ignore"]
name = always ignore during merge
driver = ignore.sh %0 %A %B

并在我的 $PATH 中创建了 ignore.sh 文件:

exit 0

我在 public/中创建了 .gitattributes 文件,因为 photoshop 文件夹应该被整体忽略,它会出现在 public/下:

photoshop  merge=ignore
photoshop/ merge=ignore
photoshop/* merge=ignore
photoshop/**/* merge=ignore

如您所见,我尝试了几种不同的模式来忽略整个文件夹,但它不起作用。我相信这是因为 master 分支上没有文件夹,所以没有冲突,所以 git 不使用忽略驱动程序。有没有一种方法可以实现这一点而不必在 master 上创建 public/photoshop 文件夹?

谢谢!

最佳答案

正如我的其他回答所建议的,这里是该解决方案的扩展、通用、工业强度版本。

(是的,我在家很无聊,也没有更好的事可做 :P)

此脚本将根据您的本地 设计分支添加一个新的、分离的 提交,因此它不会影响设计存储库或您的设计分支。提交将删除所有需要的文件。然后它执行 merge 。

对于那些懒得阅读完整代码的人,这些步骤的“核心”可以简化为:

original=$(gitbranch HEAD)    # current branch name, or sha1 if not in a branch
branchsha=$(gitsha "$branch") # sha1 of a ref, to force detached commit

git checkout "$branchsha" &&
git rm -rf "${files[@]}" &&
git commit -m "$msgcommit" &&
newsha=$(gitsha HEAD) &&
git checkout "$original" &&
git merge -m "$msgmerge" "${mergeopts[@]}" "$newsha"

这是完整的脚本:

(稍微修改以应对 SO 语法着色的弱和有限,因此最好从下面的链接获取原始源代码)

git-strip-merge

#!/bin/bash
#
# git-strip-merge - a git-merge that delete files on branch before merging
#
# Copyright (C) 2012 Rodrigo Silva (MestreLion) <linux@rodrigosilva.com>
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not see <http://www.gnu.org/licenses/gpl.html>
#
# Answer for "How to setup a git driver to ignore a folder on merge?"
# See http://stackoverflow.com/questions/3111515

#Defaults:
msgcommit="remove files from '<branch>' before merge"
msgmerge="Merge stripped branch '<branch>'"
verbose=0
quiet=(--quiet)

usage() {
cat <<- USAGE
Usage: $myname [git-merge options] [-M <commitmsg>] <branch> FILE...
USAGE
if [[ "$1" ]] ; then
cat >&2 <<- USAGE
Try '$myname --help' for more information.
USAGE
exit 1
fi
cat <<-USAGE

"git-merge that delete files on "foreign" <branch> before merging

Useful for ignoring a folder in <branch> before merging it with
current branch. Works by deleting FILE(S) in a detached commit based
on <branch>, and then performing the merge of this new commit in the
current branch. Note that <branch> is not changed by this procedure.
Also note that <branch> may actually be any reference, like a tag,
or a remote branch, or even a commit SHA.

For more information, see <http://stackoverflow.com/questions/3111515>

Options:
-h, --help
show this page.

-v, --verbose
do not use -q to supress normal output of internal steps from git
checkout, rm, commit. By default, only git merge output is shown.
Errors, however, are never supressed

-M <message>, --msgcommit=<message>
message for the removal commit in <branch>. Not to be confused
with the message of the merge commit, which is set by -m. Default
message is: "$msgcommit"

-m <message>, --message=<message>
message for the merge commit. Since we are not merging <branch>
directly, but rather a detached commit based on it, we forge a
message similar to git's default for a branch merge. Otherwise
git would use in message the full and ugly SHA1 of our commit.
Default message is: "$msgmerge"

For both commit messages, the token "<branch>" is replaced for the
actual <branch> name.

Additional options are passed unchecked to git merge.

All options must precede <branch> and FILE(s), except -h and --help
that may appear anywhere on the command line.

Example:
$myname design "photoshop/*"

Copyright (C) 2012 Rodrigo Silva (MestreLion) <linux@rodrigosilva.com>
License: GPLv3 or later. See <http://www.gnu.org/licenses/gpl.html>"
USAGE
exit 0
}

# Helper functions
myname="${0##*/}"
argerr() { printf "%s: %s\n" "${0##*/}" "${1:-error}" >&2 ; usage 1 ; }
invalid() { argerr "invalid option: $1" ; }
missing() { argerr "missing ${2:+$2 }operand${1:+ from $1}." ; }

# Option handling
files=()
mergeopts=()
for arg in "$@"; do case "$arg" in -h|--help) usage ;; esac; done
while (( $# )); do
case "$1" in
-v|--verbose ) verbose=1 ;;
-M ) shift ; msgcommit=$1 ;;
-m ) shift ; msgmerge=$1 ;;
--msgcommit=* ) msgcommit=${1#*=} ;;
--message=* ) msgmerge=${1#*=} ;;
-* ) mergeopts+=( "$1" ) ;;
* ) branch="$1"
shift ; break ;;
esac
shift
done
files+=( "$@" )

# Argument handling

msgcommit=${msgcommit//<branch>/$branch}
msgmerge=${msgmerge//<branch>/$branch}

[[ "$msgcommit" ]] || missing "msgcommit" "MSG"
[[ "$branch" ]] || missing "" "<branch>"
(( ${#files[@]} )) || missing "" "FILE"

((verbose)) && quiet=()

# Here the fun begins...
gitsha() { git rev-parse "$1" ; }
gitbranch() {
git symbolic-ref "$1" 2> /dev/null | sed 's/refs\/heads\///' ||
gitsha "$1"
}

original=$(gitbranch HEAD)
branchsha=$(gitsha "$branch")

trap 'git checkout --quiet "$original"' EXIT

git checkout "$branchsha" "${quiet[@]}" &&
git rm -rf "${files[@]}" "${quiet[@]}" &&
git commit -m "$msgcommit" "${quiet[@]}" &&
newsha=$(gitsha HEAD) &&
git checkout "$original" "${quiet[@]}" &&
git merge -m "$msgmerge" "${mergeopts[@]}" "$newsha"

尽情享受吧!

一张图片胜过千言万语...

merge 前:

enter image description here

merge 后:

enter image description here

请注意,由于分离提交技巧,“设计”分支提示根本没有受到影响,即使是本地分支也是如此。除此之外,两个提交(删除和 merge )都是常规提交,具有合适的提交消息和父级。并且“master”分支没有任何不需要的文件。

关于git - 如何设置 git 驱动程序以在 merge 时忽略文件夹,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3111515/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com