aboutsummaryrefslogtreecommitdiff
blob: b5d98688c4ac9f8cdd8550ce9ff49118b362c1a3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
#!/bin/bash
# gentoo-infra: infra/githooks.git:postrecv-bugs
# Copyright 2017-2024 Gentoo Authors
# Distributed under the terms of the GNU General Public License v2 or later

# Author: Michał Górny <mgorny@gentoo.org>

export LANG=en_US.UTF-8
export LC_MESSAGES=C
export TZ=UTC

shopt -o -s noglob

ALLOWED_BRANCHES=$(git config --get gentoo.bugs.allowed-branches)

declare -A COMMENT_BUGS=()
declare -A CLOSE_BUGS=()
declare -A CLOSE_RESO=()

while read -r oldrev newrev refname; do
    # operate only on branches in gentoo.bugs.allowed-branches
    # (or 'master' if unset)
    allowed=0
    for allowed_branch in ${ALLOWED_BRANCHES:-master}; do
        if [[ ${refname#refs/heads/} == ${allowed_branch} ]]; then
            allowed=1
            break
        fi
    done
    [[ ${allowed} == 0 ]] && continue

    while read -r commithash; do
        while read -r l; do
            is_fixes=0
            case ${l} in
                # kinda-like github/gitlab/bitbucket but:
                # 1. we accept only -s forms for simplicity,
                # 2. we accept only footer-style to avoid false positives,
                # 3. we have to scan the whole commit message because
                # developers still fail to have just one footer.
                Closes:*|Resolves:*)
                    close=1;;
                # normally used to reference commit ids
                Fixes:*)
                    is_fixes=1
                    close=1;;
                # alternate form to ref without closing
                Bug:*)
                    close=0;;
                *)
                    continue;;
            esac

            # strip whitespace, split words
            bugref=( ${l#*:} )
            if [[ ${is_fixes} == 1 ]]; then
                case ${bugref} in
                    # commit ref
                    [0-9a-fA-F][0-9a-fA-F][0-9a-fA-F][0-9a-fA-F][0-9a-fA-F][0-9a-fA-F][0-9a-fA-F][0-9a-fA-F]*)
                        continue;;
                    *)
                        echo "WARNING: 'Fixes' tag should reference commit id, not: ${bugref[*]}"
                        ;;
                esac
            fi
            for bug in "${bugref[@]}"; do
                bug="${bug,,}"
                unset resolution
                case ${bug} in
                    # long bugzilla URL
                    http://bugs.gentoo.org/show_bug.cgi\?*|https://bugs.gentoo.org/show_bug.cgi\?*)
                        bugno=${bug#*[?&]id=}
                        bugno=${bugno%%[&#]*}
                        ;;
                    # short bugzilla URL
                    http://bugs.gentoo.org/[0-9]*|https://bugs.gentoo.org/[0-9]*)
                        bugno=${bug##*/}
                        bugno=${bugno%%[?#]*}
                        ;;
                    # silently ignore github, mirror hook will handle it
                    http://github.com/*|https://github.com/*)
                        continue;;
                    \(fixed\)|\(obsolete\)|\(pkgremoved\))
                        resolution="${bug#(}"
                        resolution="${resolution%)}"
                        resolution="${resolution^^}"
                        ;;
                    \(*\))
                        echo "WARNING: invalid/unsupported resolution: ${bug}"
                        continue
                        ;;
                    *)
                        echo "WARNING: invalid/unsupported bug ref: ${bug}"
                        continue;;
                esac

                if [[ -n ${bugno//[0-9]} ]]; then
                    echo "WARNING: invalid Gentoo Bugzilla URL: ${bug}"
                    continue
                fi

                newmsg="

https://gitweb.gentoo.org/${GL_REPO}.git/commit/?id=${commithash}

$(git show --pretty=fuller --date=iso-local --stat "${commithash}")"
# TODO: --show-signature with some nice short output

                if [[ ${close} == 1 ]]; then
                    if [[ -z ${resolution} ]]; then # we catch the resolution on second pass over the same line
                        CLOSE_BUGS[${bugno}]+=${newmsg}
                    fi
                    CLOSE_RESO[${bugno}]=${resolution:-FIXED}
                else
                    COMMENT_BUGS[${bugno}]+=${newmsg}
                fi
                unset resolution
            done
        done < <(git show -q --pretty=format:'%B' "${commithash}")
    done < <(git rev-list "${oldrev}..${newrev}")
done

for bug in "${!CLOSE_BUGS[@]}"; do
    msg="The bug has been closed via the following commit(s):${CLOSE_BUGS[${bug}]}"

    if [[ -n ${COMMENT_BUGS[${bug}]} ]]; then
        msg+="

Additionally, it has been referenced in the following commit(s):${COMMENT_BUGS[${bug}]}"
    fi

    bugz modify -s RESOLVED -r "${CLOSE_RESO[${bug}]}" -c "${msg}" "${bug}"
done

for bug in "${!COMMENT_BUGS[@]}"; do
    [[ -n ${CLOSE_BUGS[${bug}]} ]] && continue

    msg="The bug has been referenced in the following commit(s):${COMMENT_BUGS[${bug}]}"

    bugz modify -c "${msg}" "${bug}"
done

exit 0