ฉันสามารถเปลี่ยนชื่อและนามสกุลของฉันในการกระทำก่อนหน้านี้ทั้งหมดได้หรือไม่?


122

ฉันต้องการเปลี่ยนชื่อนามสกุลและอีเมลในการกระทำทั้งหมดของฉันเป็นไปได้หรือไม่?


2
เป็นที่เก็บข้อมูลสำหรับคุณเพียงไม่กี่คนหรือสำหรับโครงการขนาดใหญ่?
thejh



คำตอบ:


212

ใช้git-filter-branch.

git filter-branch --commit-filter 'if [ "$GIT_AUTHOR_NAME" = "Josh Lee" ];
  then export GIT_AUTHOR_NAME="Hobo Bob"; export GIT_AUTHOR_EMAIL=hobo@example.com;
  fi; git commit-tree "$@"'

สิ่งนี้มีผลกับผู้เขียนเท่านั้นไม่ใช่ผู้คอมมิต (ซึ่งสำหรับคอมมิตส่วนใหญ่จะเหมือนกับผู้เขียน) หากคุณต้องการเขียนสิ่งเหล่านี้ใหม่ด้วยให้ตั้งค่าตัวแปรGIT_COMMITTER_NAMEและGIT_COMMITTER_EMAIL

เตือนมาตรฐานเกี่ยวกับการเขียนประวัติศาสตร์ใช้; ทำเฉพาะกับประวัติที่ยังไม่ได้แชร์เท่านั้น

อัปเดตประจำเดือนมิถุนายน 2018

ขณะนี้คู่มือมีวิธีแก้ปัญหาโดยใช้--env-filterในตัวอย่าง: https://git-scm.com/docs/git-filter-branch#_examples :

git filter-branch --env-filter '
    if test "$GIT_AUTHOR_EMAIL" = "root@localhost"
    then
        GIT_AUTHOR_EMAIL=john@example.com
    fi
    if test "$GIT_COMMITTER_EMAIL" = "root@localhost"
    then
        GIT_COMMITTER_EMAIL=john@example.com
    fi
' -- --all

4
หากคุณกำลังใช้msysgitงานคุณยังสามารถเข้าถึงbashไฟล์. ไม่งั้นฉันไม่มีเงื่อนงำ
Josh Lee

@ Joshua หากคุณกำลังใช้บางอย่างที่คุณไม่มี bash คุณอาจใช้การเขียนสคริปต์ Windows batch แม้ว่าฉันจะไม่ได้ลอง
MatrixFrog

แล้วแท็กล่ะ โซลูชันนี้จะไม่เปลี่ยนผู้เขียนแท็ก
piotrek

@ Joshua ตรวจสอบ git repo บนกล่อง linux และทำการแก้ไขที่นั่น
Will Sheppard

มีตัวเลือกที่ส่งผลให้เกิดสาขาใหม่และออกจากแหล่งที่มาหรือไม่
Eugen Konkov

56

ในการเขียนทั้งผู้เขียนและผู้คอมมิทเตอร์ในคอมมิตที่เลือกทั้งหมด

git filter-branch --commit-filter \
'if [ "$GIT_AUTHOR_NAME" = "OldAuthor Name" ]; then \
export GIT_AUTHOR_NAME="Author Name";\
export GIT_AUTHOR_EMAIL=authorEmail@example.com;\
export GIT_COMMITTER_NAME="Commmiter Name";\
export GIT_COMMITTER_EMAIL=commiterEmail@example.com;\
fi;\
git commit-tree "$@"'

1
แต่จะใช้การเปลี่ยนแปลงกับเซิร์ฟเวอร์ระยะไกลได้อย่างไร?
vikyd

5
@Viky Trygit push --all origin --force
user11153

2
ได้ผลสำหรับฉัน! ฉันใช้ GitLab ฉันต้องยกเลิกการป้องกันสาขาก่อนคำสั่งพุช
vikyd

37

หากไม่มีผู้เขียนคนอื่นคุณสามารถทำได้:

git filter-branch --commit-filter 'export GIT_AUTHOR_NAME="authorname"; \
export GIT_AUTHOR_EMAIL=mail@example.com; git commit-tree "$@"'

1
ไม่ใช่การเขียนข้อมูล "Committer:" ใหม่
user11153

1
ไม่ได้มีวัตถุประสงค์เพื่อเขียนข้อมูลคอมมิทเตอร์ใหม่ หากคุณต้องการทำเช่นนั้นให้ส่งออก GIT_COMMITTER_NAME และ GIT_COMMITTER_EMAIL ด้วย (ดูคำตอบที่ยอมรับ)
chronospoon

12

บันทึกสคริปต์ด้านล่างเป็นเช่น~/.bin/git-replace-authorและเรียกใช้โดยใช้เช่น:

git replace-author "John Ssmith" "John Smith" "johnsmith@example.com"

โดยไม่มีข้อโต้แย้งจะอัปเดตคำมั่นสัญญาทั้งหมดด้วยชื่อของคุณเพื่อใช้ที่อยู่อีเมลปัจจุบันของคุณตามการกำหนดค่า Git

DEFAULT_NAME="$(git config user.name)"
DEFAULT_EMAIL="$(git config user.email)"
export OLD_NAME="${1:-$DEFAULT_NAME}"
export NEW_NAME="${2:-$DEFAULT_NAME}"
export NEW_EMAIL="${3:-$DEFAULT_EMAIL}"

echo "Old:" $OLD_NAME "<*>"
echo "New:" "$NEW_NAME <$NEW_EMAIL>"
echo "To undo, use: git reset $(git rev-parse HEAD)"

git filter-branch --env-filter \
'if [ "$GIT_AUTHOR_NAME" = "${OLD_NAME}" ]; then
    export GIT_AUTHOR_NAME="${NEW_NAME}"
    export GIT_AUTHOR_EMAIL="${NEW_EMAIL}"
    export GIT_COMMITTER_NAME="${NEW_NAME}"
    export GIT_COMMITTER_EMAIL="${NEW_EMAIL}"
fi'

ดิบ (เพื่อดาวน์โหลด)


ในฐานะที่เป็นโน้ตสั้น: ~/.bin/จะต้องมีผู้ใช้ภายในและความต้องการของไฟล์ที่จะปฏิบัติการเรียกใช้เพื่อ:$PATH chmod +x ~/.bin/git-replace-author
Michael Gecht

และจะทำอย่างไรกับข้อโต้แย้ง?
Eugen Konkov

2

เฉพาะในกรณีที่คุณไม่ได้ผลักดันความมุ่งมั่นของคุณไปทั่วโลก คนอื่น ๆ ที่ฉลาดคนอื่น ๆ มีชื่อเก่าของคุณใน repo ซึ่งไม่น่าเป็นไปได้ที่คุณจะเปลี่ยนทุกคนได้


จริง แต่ในบางกรณีคุณไม่มีทางเลือก กรณีของฉันฉันกำหนดที่อยู่อีเมลผิดในการกำหนดค่าคอมไพล์ของฉัน (อย่างที่เห็นใน "git config --global -l") ด้วยเหตุนี้ฉันจึงไม่มีกิจกรรมคอมมิตปรากฏในที่เก็บ Github ของฉันเอง (เนื่องจากที่อยู่อีเมลไม่ตรงกับอีเมลที่กำหนดค่าใน Github)! ในการแก้ปัญหานี้ฉันได้แก้ไขคอมมิตในเครื่องของฉันโดยใช้สูตรจากstackoverflow.com/a/23564785/2474068 (ทำงานได้อย่างสมบูรณ์แบบ) จากนั้นฉันก็พุชคอมมิตที่เปลี่ยนแปลงไปยัง Github โดยใช้ "git push -u -f origin master" (ด้วยการบังคับ ธง "-f") นั่นขัดกับการปฏิบัติที่ยอมรับ แต่ฉันไม่มีทางเลือก!
สิงห์

1
ใช่ประเด็นของฉันคือส้อมใด ๆ ของ repo นั้นจะไม่มีการเปลี่ยนแปลงนั้นเว้นแต่พวกเขาจะยอมรับแรงผลักดันของคุณ มันเป็นเรื่องยากที่จะได้รับการอัปเดตทุกส้อม :)
EnabrenTane

1

ด้วย Git 2.24 (ไตรมาสที่ 4 ปี 2019) git filter-branch(และ BFG) จะเลิก

สิ่งที่เทียบเท่าจะใช้newren/git-filter-repoและส่วนตัวอย่าง :

cd repo
git filter-repo --mailmap my-mailmap

กับmy-mailmap:

Correct Name <correct@email.com> <old@email.com>

ซึ่งจะแทนที่ชื่อผู้เขียนและอีเมลของการกระทำใด ๆ ที่ทำโดยทุกคนด้วย <old@email.com>

ดูgit shortlogส่วนผู้เขียนการแมปสำหรับไวยากรณ์ที่แน่นอนของ

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.