Могу ли я изменить свое имя и фамилию во всех предыдущих коммитах?

122

Я бы хотел изменить свое имя, фамилию и адрес электронной почты во всех моих коммитах, возможно ли это?

Джошуа
источник
2
Это репозиторий только для вас, для нескольких человек или для большого проекта?
thejh
3
возможный дубликат Как мне изменить автора коммита в git?
Джош Ли
Возможный дубликат изменения автора и имени
коммиттера,

Ответы:

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. В противном случае я понятия не имею.
Джош Ли
@Joshua, если вы используете что-то, где у вас нет bash, вы, вероятно, могли бы использовать пакетные сценарии Windows, хотя я не пробовал.
MatrixFrog
а что насчет тегов? это решение не изменит автора тегов
piotrek 05
@Joshua проверьте репозиторий git на Linux-сервере и исправьте там исправление
Уилл Шеппард,
Есть ли варианты, которые приводят к созданию новой ветки и оставляют исходные коммиты нетронутыми?
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 "$@"'
user11153
источник
1
Но как применить изменения к удаленному серверу?
vikyd
5
@Viky Trygit push --all origin --force
user11153
2
Меня устраивает ! Я использую GitLab, мне нужно снять защиту с ветки перед командой push.
vikyd
37

Если других авторов нет, можно:

git filter-branch --commit-filter 'export GIT_AUTHOR_NAME="authorname"; \
export GIT_AUTHOR_EMAIL=mail@example.com; git commit-tree "$@"'
denis.peplin
источник
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'

Raw (скачать)

Zaz
источник
В короткой заметке ~/.bin/должна быть внутри пользователей $PATHи потребности файл исполняемым, так работать: chmod +x ~/.bin/git-replace-author.
Майкл Гехт
А что с аргументами?
Eugen Konkov
2

Только если вы еще не опубликовали свои коммиты в мире. В противном случае у всех остальных будет ваше старое имя в своем репо, и вряд ли вы сможете изменить имя каждого.

EnabrenTane
источник
Верно, но в некоторых случаях у вас нет выбора. В моем случае у меня был неправильный адрес электронной почты, настроенный в моей конфигурации git (как я мог видеть с помощью «git config --global -l»). В результате в моем собственном репозитории Github не отображалось никаких действий по фиксации (потому что адрес электронной почты не совпадал с адресом электронной почты, настроенным в Github)! Чтобы решить эту проблему, я исправил свои локальные коммиты, используя рецепт из stackoverflow.com/a/23564785/2474068 (работал отлично), а затем я отправил измененные коммиты в Github, используя "git push -u -f origin master" (с силой флаг "-f"). Это противоречит общепринятой практике, но у меня не было выбора!
leo
1
Да, я хотел сказать, что у любых форков этого репо не будет этого изменения, если они не примут ваш принудительный толчок. Было бы сложно обновить каждую вилку :)
EnabrenTane
1

С Git 2.24 (4 квартал 2019 г.) git filter-branch(и BFG) устарели .

Эквивалентом будет, using 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Раздел об авторе сопоставления для получения точного синтаксиса

VonC
источник