it-swarm.com.ru

Как я могу получить список веток Git, упорядоченный по последнему коммиту?

Я хочу получить список всех ветвей в Git-репозитории с «самыми свежими» ветвями вверху, где «самая свежая» ветвь - это та, которая была зафиксирована в последнее время (и, следовательно, с большей вероятностью будет одной Хочу обратить внимание).

Есть ли способ, которым я могу использовать Git, чтобы (а) отсортировать список ветвей по последнему коммиту или (б) получить список веток вместе с датой последнего коммита, в каком-то машиночитаемом формате?

В худшем случае я всегда мог запустить git branch, чтобы получить список всех ветвей, проанализировать его выходные данные, а затем git log -n 1 branchname --format=format:%ci для каждой, чтобы получить дату фиксации каждой ветви. Но это будет работать на Windows-боксе, где ускорение нового процесса является относительно дорогим, поэтому запуск исполняемого файла Git один раз на ветку может замедлиться, если имеется много ветвей. Есть ли способ сделать все это с помощью одной команды?

984
Joe White

Используйте параметр --sort=-committerdate для git for-each-ref ;

Также доступно начиная с Git 2.7.0 for git branch :

Основное использование:

git for-each-ref --sort=-committerdate refs/heads/

# Or using git branch (since version 2.7.0)
git branch --sort=-committerdate  # DESC
git branch --sort=committerdate  # ASC

Результат:

Result

Расширенное использование:

git for-each-ref --sort=committerdate refs/heads/ --format='%(HEAD) %(color:yellow)%(refname:short)%(color:reset) - %(color:red)%(objectname:short)%(color:reset) - %(contents:subject) - %(authorname) (%(color:green)%(committerdate:relative)%(color:reset))'

Результат:

Result

1407
Jakub Narębski

Список имен веток git, упорядоченный по последним коммитам…

Расширяя ответ Якуба и совет Джо, следующее исключит "refs /heads", поэтому в выводе будут отображаться только имена ветвей:


Команда:

git for-each-ref --count=30 --sort=-committerdate refs/heads/ --format='%(refname:short)'

Результат:

 recent git branches

113
Beau Smith

Вот оптимальный код, который объединяет два других ответа:

git for-each-ref --sort=-committerdate refs/heads/ --format='%(committerdate:short) %(authorname) %(refname:short)'
78
nikolay

Вот простая команда, которая перечисляет все ветви с последними коммитами:

git branch -v

Чтобы упорядочить по последнему коммиту, используйте

git branch -v --sort=committerdate

Источник: http://git-scm.com/book/en/Git-Branching-Branch-Management

68
user1682406

Я использую следующий псевдоним:

recent = "!r(){git for-each-ref --sort=-committerdate refs/heads --format='%(HEAD)%(color:yellow)%(refname:short)|%(color:bold green)%(committerdate:relative)|%(color:blue)%(subject)|%(color:Magenta)%(authorname)%(color:reset)'|column -ts'|'}; r"

который производит: result

Правка: используйте '|' отделить, благодаря @ Björn Lindqvist

Обновление: добавлено * перед текущей веткой, спасибо @elhadi
Правка: исправлен случай, когда текущая ветвь была подстрокой другой ветки.

Правка: использовать более простой синтаксис для текущей ветви, благодаря @Joshua Skrzypek

39
dimid

Мне также были нужны цвета, теги и удаленные ссылки без дубликатов:

for ref in $(git for-each-ref --sort=-committerdate --format="%(refname)" refs/heads/ refs/remotes ); do git log -n1 $ref --pretty=format:"%Cgreen%cr%Creset %C(yellow)%d%Creset %C(bold blue)<%an>%Creset%n" | cat ; done | awk '! a[$0]++'

Поскольку цитирование может быть трудным, здесь псевдоним для bash:

alias glist='for ref in $(git for-each-ref --sort=-committerdate --format="%(refname)" refs/heads/ refs/remotes ); do git log -n1 $ref --pretty=format:"%Cgreen%cr%Creset %C(yellow)%d%Creset %C(bold blue)<%an>%Creset%n" | cat ; done | awk '"'! a["'$0'"]++'"
36
estani

Я смог сослаться на приведенные выше примеры, чтобы создать что-то, что работает лучше для меня.

git for-each-ref --sort = -committerdate refs /heads / --format = '% (authordate: short)% (цвет: красный)% (имя объекта: короткий)% (цвет: желтый)% (refname: short)% (цвет: сброс) (% (Цвет: зеленый)% (committerdate: относительная)% (цвет: сброс))»

 Screenshot of Output

24
Andy

Другие ответы не позволяют передавать -vv для получения подробного вывода.

Итак, вот одна строка, которая сортирует git branch -vv по дате фиксации, сохраняя цвет и т. Д .:

git branch -vv --color=always | while read; do echo -e $(git log -1 --format=%ct $(echo "_$REPLY" | awk '{print $2}' | Perl -pe 's/\e\[?.*?[\@-~]//g') 2> /dev/null || git log -1 --format=%ct)"\t$REPLY"; done | sort -r | cut -f 2

Если вы дополнительно хотите распечатать дату фиксации, вы можете использовать эту версию вместо:

git branch -vv --color=always | while read; do echo -e $(git log -1 --format=%ci $(echo "_$REPLY" | awk '{print $2}' | Perl -pe 's/\e\[?.*?[\@-~]//g') 2> /dev/null || git log -1 --format=%ci)" $REPLY"; done | sort -r | cut -d ' ' -f -1,4-

Образец вывода:

2013-09-15   master                  da39a3e [Origin/master: behind 7] Some patch
2013-09-11 * (detached from 3eba4b8) 3eba4b8 Some other patch
2013-09-09   my-feature              e5e6b4b [master: ahead 2, behind 25] WIP

Это, вероятно, более читаемый разделить на несколько строк:

git branch -vv --color=always | while read; do
    # The underscore is because the active branch is preceded by a '*', and
    # for awk I need the columns to line up. The Perl call is to strip out
    # ansi colors; if you don't pass --color=always above you can skip this
    local branch=$(echo "_$REPLY" | awk '{print $2}' | Perl -pe 's/\e\[?.*?[\@-~]//g')
    # git log fails when you pass a detached head as a branch name.
    # Hide the error and get the date of the current head.
    local branch_modified=$(git log -1 --format=%ci "$branch" 2> /dev/null || git log -1 --format=%ci)
    echo -e "$branch_modified $REPLY"
# cut strips the time and timezone columns, leaving only the date
done | sort -r | cut -d ' ' -f -1,4-

Это также должно работать с другими аргументами git branch, например, -vvr для вывода списка удаленных отслеживаний или -vva для отображения как удаленных отслеживаний, так и локальных филиалов.

23
John Mellor

git 2.7 (четвертый квартал 2015 года) представит сортировку веток с использованием прямого git branch:
См. Commit aa3bc55 , commit aedcb7d , commit 1511b22 , commit f65f139 , ... (23 сентября 2015 г.), commit aedcb7d , commit 1511b22 , commit ca41799 (24 сентября 2015 г.) и commit f65f139 , ... (23 сентября 2015 г.) Картик Наяк (KarthikNayak) .
(Объединено Junio ​​C Hamano - gitster - в commit 7f11b48 , 15 октября 2015 г.)

В частности, commit aedcb7d :

branch.c: использовать API ref-filter

Заставьте 'branch.c' использовать API-интерфейсы 'ref-filter' для перебора сортировки ссылок. Это удаляет большую часть кода, используемого в 'branch.c', заменяя его Вызовами в библиотеку 'ref-filter'.

Это добавляет параметр --sort=<key> :

Сортировка по заданному ключу.
Префикс - для сортировки в порядке убывания значения. 

Вы можете использовать опцию --sort=<key> несколько раз, и в этом случае последний ключ становится первичным ключом. 

Поддерживаются ключи такие же, как в git for-each-ref .
Порядок сортировки по умолчанию сортируется на основе полного имени (включая префикс refs/...). В этом списке сначала отображается отсоединенный HEAD (если он есть), затем локальные ветви и, наконец, ветви удаленного отслеживания.

Вот:

git branch --sort=-committerdate 

Или (см. Ниже с Git 2.19)

# if you are sure to /always/ want to see branches ordered by commits:
git config --global branch.sort -committerdate
git branch

Смотрите также commit 9e46833 (30 октября 2015 г.) от Karthik Nayak (KarthikNayak) .
При поддержке: Junio ​​C Hamano (gitster) .
(Объединено Junio ​​C Hamano - gitster - в commit 415095f , 03 ноября 2015 г.)

При сортировке по числовым значениям (например, --sort=objectsize) нет запасного сравнения, когда обе ссылки содержат одно и то же значение. Это может привести к неожиданным результатам (т. Е. Порядок перечисления ссылок с одинаковыми значениями не может быть предварительно определен), как указал Йоханнес Сикст ( $ gmane/280117 ).

Следовательно, возврат к алфавитному сравнению на основе refname всякий раз, когда другой критерий равен.

$ git branch --sort=objectsize

*  (HEAD detached from fromtag)
      branch-two
      branch-one
      master

С Git 2.19 порядок сортировки может быть установлен по умолчанию.
git branch поддерживает конфигурацию branch.sort, такую ​​как git tag, которая уже имела конфигурацию tag.sort.
См. Commit 560ae1c (16 августа 2018 г.) от Самуэль Мафтул (``) .
(Объединено с Junio ​​C Hamano - gitster - в commit d89db6f , 27 августа 2018 г.)

branch.sort:

Эта переменная управляет порядком сортировки веток, когда отображается git-branch.
Без предоставленной опции «--sort=<value>» значение этой переменной будет использоваться по умолчанию.


Для просмотра удаленных веток используйте git branch -r --sort=objectsize. Флаг -r заставляет его перечислять удаленные ветви вместо локальных.

21
VonC

Мне нравится использовать относительную дату и сокращать название филиала следующим образом:

git for-each-ref --sort='-authordate:iso8601' --format=' %(authordate:relative)%09%(refname:short)' refs/heads

Что дает вам вывод:

21 minutes ago  nathan/a_recent_branch
6 hours ago     master
27 hours ago    nathan/some_other_branch
29 hours ago    branch_c
6 days ago      branch_d

Я рекомендую создать файл bash для добавления всех ваших любимых псевдонимов, а затем поделиться сценарием с вашей командой. Вот пример, чтобы добавить только этот: 

#!/bin/sh

git config --global alias.branches "!echo ' ------------------------------------------------------------' && git for-each-ref --sort='-authordate:iso8601' --format=' %(authordate:relative)%09%(refname:short)' refs/heads && echo ' ------------------------------------------------------------'"

Затем вы можете просто сделать это, чтобы получить хорошо отформатированный и отсортированный список локальных веток:

git branches

Обновление: Сделайте это, если хотите раскрасить:

#!/bin/sh
#
(echo ' ------------------------------------------------------------‌​' && git for-each-ref --sort='-authordate:iso8601' --format=' %(authordate:relative)%09%(refname:short)' refs/heads && echo ' ------------------------------------------------------------‌​') | grep --color -E "$(git rev-parse --abbrev-ref HEAD)$|$"
19
n8tr

Добавляет немного цвета (так как pretty-format недоступен)

[alias]
    branchdate = for-each-ref --sort=-committerdate refs/heads/ --format="%(authordate:short)%09%(objectname:short)%09%1B[0;33m%(refname:short)%1B[m%09"
11
epylinkn

Начиная с git 2.19 вы можете просто:

git branch --sort=-committerdate

Вы также можете:

git config branch.sort -committerdate

Таким образом, всякий раз, когда вы перечисляете ветки в текущем репозитории, они будут перечислены отсортированными по committerdate.

Если всякий раз, когда вы перечисляете ветви, вы хотите, чтобы они сортировались по comitterdate:

git config --global branch.sort -committerdate

Отказ от ответственности: я являюсь автором этой функции в git, я реализовал ее, когда увидел этот вопрос.

9
user801247

У меня была такая же проблема, поэтому я написал рубиновый гем под названием Twig . В нем перечислены ветви в хронологическом порядке (сначала самые новые), а также можно указать максимальный возраст, чтобы вы не перечисляли все ветви (если их много). Например:

$ twig

                              issue  status       todo            branch
                              -----  ------       ----            ------
2013-01-26 18:00:21 (7m ago)  486    In progress  Rebase          optimize-all-the-things
2013-01-26 16:49:21 (2h ago)  268    In progress  -               whitespace-all-the-things
2013-01-23 18:35:21 (3d ago)  159    Shipped      Test in prod  * refactor-all-the-things
2013-01-22 17:12:09 (4d ago)  -      -            -               development
2013-01-20 19:45:42 (6d ago)  -      -            -               master

Он также позволяет хранить пользовательские свойства для каждой ветви, например, идентификатор заявки, статус, задачи и фильтровать список ветвей в соответствии с этими свойствами. Больше информации: http://rondevera.github.io/twig/

8
Ron DeVera

Я придумал следующую команду (для Git 2.13 и позже):

git branch -r --sort=creatordate \
    --format "%(creatordate:relative);%(committername);%(refname:lstrip=-1)" \
    | grep -v ";HEAD$" \
    | column -s ";" -t

Если у вас нет column, вы можете заменить последнюю строку на

    | sed -e "s/;/\t/g"

Выход выглядит как

6 years ago             Tom Preston-Werner  book
4 years, 4 months ago   Parker Moore        0.12.1-release
4 years ago             Matt Rogers         1.0-branch
3 years, 11 months ago  Matt Rogers         1.2_branch
3 years, 1 month ago    Parker Moore        v1-stable
12 months ago           Ben Balter          pages-as-documents
10 months ago           Jordon Bedwell      make-jekyll-parallel
6 months ago            Pat Hawks           to_integer
5 months ago            Parker Moore        3.4-stable-backport-5920
4 months ago            Parker Moore        yajl-Ruby-2-4-patch
4 weeks ago             Parker Moore        3.4-stable
3 weeks ago             Parker Moore        rouge-1-and-2
19 hours ago            jekyllbot           master

Я написал сообщение в блоге о том, как работают различные части.

8
bdesham

К вашему сведению, если вы хотите получить список недавних проверенных веток (в отличие от недавно совершенных), вы можете использовать git reflog:

$ git reflog | egrep -io "moving from ([^[:space:]]+)" | awk '{ print $3 }' | head -n5
master
stable
master
some-cool-feature
feature/improve-everything

Смотрите также: Как я могу получить список веток Git, которые я недавно проверил?

7
Jordan Brough

Вот еще один скрипт, который делает то же, что и все остальные скрипты. На самом деле, это обеспечивает функцию для вашей оболочки.

Его вклад заключается в том, что он вытягивает некоторые цвета из вашего git config (или использует значения по умолчанию).

# Git Branch by Date
# Usage: gbd [ -r ]
gbd() {
    local reset_color=`tput sgr0`
    local subject_color=`tput setaf 4 ; tput bold`
    local author_color=`tput setaf 6`

    local target=refs/heads
    local branch_color=`git config --get-color color.branch.local white`

    if [ "$1" = -r ]
    then
        target=refs/remotes/Origin
        branch_color=`git config --get-color color.branch.remote red`
    fi

    git for-each-ref --sort=committerdate $target --format="${branch_color}%(refname:short)${reset_color} ${subject_color}%(subject)${reset_color} ${author_color}- %(authorname) (%(committerdate:relative))${reset_color}"
}
4
joeytwiddle

Вот небольшой скрипт, который я использую для переключения между последними ветвями:

#!/bin/bash
# Sudo bash

re='^[0-9]+$'

if [[ "$1" =~ $re ]]; then
    lines="$1"
else
    lines=10
fi
branchs="$(git recent | tail -n $lines | nl)"
branchs_nf="$(git recent-nf | tail -n $lines | nl)"
echo "$branchs"

# Prompt which server to connect to
max="$(echo "$branchs" | wc -l)"
index=
while [[ ! ( "$index" =~ ^[0-9]+$ && "$index" -gt 0 && "$index" -le "$max" ) ]]; do
    echo -n "Checkout to: " 
    read index
done

branch="$( echo "$branchs_nf" | sed -n "${index}p" | awk '{ print $NF }' )"
git co $branch
clear

Используя эти два псевдонима

recent = for-each-ref --sort=committerdate refs/heads/ --format=' %(color:blue) %(authorname) %(color:yellow)%(refname:short)%(color:reset)'
recent-nf = for-each-ref --sort=committerdate refs/heads/ --format=' %(authorname) %(refname:short)'

Просто позвоните в git-репо, и он покажет вам последние N ветвей (по умолчанию 10) и число в каждой. Введите номер филиала, и он проверит:

 enter image description here

3
Agus Arias

Основано на версии Иилиуса, но с текущей веткой, отмеченной звездочкой и цветом, и показывающей только то, что не описано как «месяцы» или «годы» назад:

current_branch="$(git symbolic-ref --short -q HEAD)"
git for-each-ref --sort=committerdate refs/heads \
  --format='%(refname:short)|%(committerdate:relative)' \
  | grep -v '\(year\|month\)s\? ago' \
  | while IFS='|' read branch date
    do
      start='  '
      end=''
      if [[ $branch = $current_branch ]]; then
        start='* \e[32m'
        end='\e[0m'
      fi
      printf "$start%-30s %s$end\\n" "$branch" "$date"
    done
3
Mark Lodato

Мой лучший результат в виде сценария:

git for-each-ref --sort=-committerdate refs/heads/ --format='%(refname:short)|%(committerdate:iso)|%(authorname)' |
    sed 's/refs\/heads\///g' |
    grep -v BACKUP  | 
    while IFS='|' read branch date author
    do 
        printf '%-15s %-30s %s\n' "$branch" "$date" "$author"
    done
2
saeedgnu

Обычно мы рассматриваем удаленные филиалы в последнее время. Так попробуй это

git fetch
git for-each-ref --sort=-committerdate refs/remotes/Origin
1
Victor Choy

Опоздал на вечеринку здесь. Принятый ответ CML качается, но если вы хотите что-то более красивое, например, GUI, и ваш Origin === "github". 

Вы можете нажать «Филиалы» в репо. или перейдите по прямой ссылке: https://github.com/ORGANIZATION_NAME/REPO_NAME/branches

1
jahrichie

git branch --sort=-committerdate | head -5

Для любого, кто заинтересован в том, чтобы отсортировать только топ-5 названий веток на основе даты коммиттера. 

1
vardin

Вот вариант, который я искал:

git for-each-ref --sort=-committerdate --format='%(committerdate)%09%(refname:short)' refs/heads/ | tail -r

Этот tail -r переворачивает список, поэтому последняя commiterdate является последней.

1
Ben

Я знаю, что ответов уже много, но вот мои два цента за простой псевдоним (мне нравится, когда моя последняя ветка внизу):

[alias]
        br = !git branch --sort=committerdate --color=always | tail -n15
[color "branch"]
        current = yellow
        local = cyan
        remote = red

Это даст вам хороший обзор последних 15 цветных веток с выделением текущей ветви (и звездочкой).

1
Tom

Я перенаправляю вывод принятого ответа в dialog, чтобы получить интерактивный список:

#!/bin/bash

TMP_FILE=/tmp/selected-git-branch

eval `resize`
dialog --title "Recent Git Branches" --menu "Choose a branch" $LINES $COLUMNS $(( $LINES - 8 )) $(git for-each-ref --sort=-committerdate refs/heads/ --format='%(refname:short) %(committerdate:short)') 2> $TMP_FILE

if [ $? -eq 0 ]
then
    git checkout $(< $TMP_FILE)
fi

rm -f $TMP_FILE

clear

Сохранить как (например) ~/bin/git_recent_branches.sh и chmod +x. Затем git config --global alias.rb '!git_recent_branches.sh', чтобы дать мне новую команду git rb.

1
John Delaney

Git v2.19 вводит параметр конфигурации branch.sort (см. branch.sort ).

Таким образом, git branch будет сортировать по дате коммиттера (desc) по умолчанию с

# gitconfig
[branch]
    sort = -committerdate     # desc

сценарий:

$ git config --global branch.sort -committerdate

Обновление:

Так,

$ git branch
* dev
  master
  _

а также

$ git branch -v
* dev    0afecf5 Merge branch 'oc' into dev
  master 652428a Merge branch 'dev'
  _      7159cf9 Merge branch 'bashrc' into dev
0
hIpPy

git for-each-ref --sort=committerdate refs/heads/ --format='%(HEAD) %(color:yellow)%(refname:short)%(color:reset) - %(color:red)%(objectname:short)%(color:reset) - %(contents:subject) - %(authorname) (%(color:green)%(committerdate:relative)%(color:reset))' это то, что вам нужно

0
ChunkCoder