stdlib: Fix detection of already build package when rpmautospec
[goals.git] / stdlib / fedora.gl
1 # Goals stdlib Fedora module.
2 # Copyright (C) 2020 Richard W.M. Jones
3 # Copyright (C) 2020 Red Hat Inc.
4 #
5 # This program is free software; you can redistribute it and/or modify
6 # it under the terms of the GNU General Public License as published by
7 # the Free Software Foundation; either version 2 of the License, or
8 # (at your option) any later version.
9 #
10 # This program is distributed in the hope that it will be useful,
11 # but WITHOUT ANY WARRANTY; without even the implied warranty of
12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 # GNU General Public License for more details.
14 #
15 # You should have received a copy of the GNU General Public License along
16 # with this program; if not, write to the Free Software Foundation, Inc.,
17 # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18
19 # Goals for dealing with Fedora builds.
20
21 # For these to work you must set up your environment in
22 # a particular way as described below:
23 #
24 # %fedora-dir:
25 # Points to a directory containing Fedora packages checked out
26 # of dist git.  You must use the fedpkg clone -B command to
27 # create separate branches.
28 #
29 # %fedora-branch:   Working branch, eg. "rawhide".
30 # %fedora-tag:      Build tag, eg. "f32-build".
31 #
32 # %fedora-rebuild-name:
33 # Magic string that identifies this rebuild.  This is used to
34 # detect if a rebuild has been attempted before in this run.
35 # eg. "Perl 5.99 rebuild"
36 #
37 # %fedora-source-packages:
38 # List of source packages that will be rebuilt.
39 #
40 # %fedora-ignored:
41 # List of source packages which will be ignored.  These are treated
42 # as if they have been rebuilt.
43 #
44 # %fedora-blocked:
45 # List of packages which are blocked.  Any dependent packages are
46 # also blocked. XXX NOT IMPLEMENTED
47
48 # Check if the source package has been built in Koji.
49
50 predicate is-koji-built (pkg) = {
51     cd %fedora-dir/%pkg/%fedora-branch
52     specfile=%pkg.spec
53
54     # Packages which are ignored are treated as if they were rebuilt already.
55     for p in %fedora-ignored; do
56         if [ %pkg = "$p" ]; then exit 0; fi
57     done
58
59     if ! grep -sq "^%%autochangelog" $specfile; then
60         # If the specfile doesn't have the magic string then the
61         # package definitely needs to be rebuilt.
62         grep -sq %fedora-rebuild-name $specfile || exit 99
63     else
64         # Using autochangelog.
65         git log | grep -sq %fedora-rebuild-name || exit 99
66     fi
67
68     # Else we must check Koji itself.
69     # Koji sends some messages to stderr.
70     nvr=$(fedpkg verrel)
71     buildinfo=$(koji buildinfo $nvr 2>&1 ||:)
72
73     # No build at all, needs rebuild.
74     echo "$buildinfo" | grep -sq "No such build" && exit 99
75
76     # Existing build, find out what state it is in.
77     state=$(echo "$buildinfo" | grep ^State: | awk '{print $2}')
78     taskid=$(echo "$buildinfo" | grep ^Task: | awk '{print $2}')
79
80     case "$state" in
81     COMPLETE)
82         # Complete so we don't need to rebuild.
83         exit 0 ;;
84     FAILED)
85         # Failed builds must be examined and fixed manually.
86         exit 1 ;;
87     BUILDING)
88         # Cancel the build, we will resubmit it.
89         koji cancel $taskid ||:
90         exit 99 ;;
91     CANCELED|DELETED)
92         # Do a rebuild.
93         exit 99 ;;
94     esac
95     # Don't know what happened there, so fail.
96     exit 1
97 }
98
99 # Rebuild a Fedora package.  This rebuilds any dependencies first.
100
101 goal fedora-rebuild (pkg) =
102 is-koji-built ("%pkg") :
103         wrap ("is-koji-built", fedora-source-dependencies (pkg)) {
104     cd %fedora-dir/%pkg/%fedora-branch
105     specfile=%pkg.spec
106
107     # We have to wait for the dependencies to become available
108     # before we can start the new build.
109     for p in %<; do
110         nvr=$(koji --quiet latest-build %fedora-tag $p | awk '{print $1}')
111         while ! koji wait-repo --timeout=10000 %fedora-tag --build=$nvr; do sleep 1m; done
112     done
113
114     # Make sure the local directory is up to date.
115     # This should also fail if there are local changes, which
116     # would need to be corrected/integrated by hand.
117     git pull
118
119     # Add the new commit.
120     if ! grep -sq "^%%autochangelog" $specfile; then
121         # If we're not building for Rawhide then we must use the
122         # rpmdev-bumpspec rightmost (-r) flag so there's an upgrade
123         # path to Rawhide.
124         rightmost=
125         if test %fedora-branch != "rawhide"; then rightmost=-r; fi
126
127         # If the specfile doesn't have the magic string then add
128         # that now.
129         if ! grep -sq %fedora-rebuild-name $specfile; then
130             rpmdev-bumpspec -c "- "%fedora-rebuild-name $rightmost *.spec
131         else
132             rpmdev-bumpspec -c "- Bump release and rebuild" $rightmost *.spec
133         fi
134
135         fedpkg commit -c
136     else
137         # Using autochangelog.
138         if ! git log | grep -sq %fedora-rebuild-name; then
139             git commit --allow-empty -m %fedora-rebuild-name
140         else
141             git commit --allow-empty -m "Bump release and rebuild"
142         fi
143     fi
144
145     # Push the new commit.
146     fedpkg push
147
148     # Do the Koji build in the side tag.
149     fedpkg build --target %fedora-tag
150 }
151
152 # Get the source package names for a particular package.
153 # Note this is not merely the BuildRequires, since those are
154 # the binary packages.  Also this will only find packages
155 # which are in the list of fedora-source-packages.
156 pure function fedora-source-dependencies (pkg) returning strings = @{
157     echo Calculating dependencies of %pkg >&2
158
159     specfile=%fedora-dir/%pkg/%fedora-branch/%pkg.spec
160
161     # We will require the mapping of all source packages to the
162     # list of binary packages they build, so work this out in advance.
163     declare -A bin2src
164     for p in %fedora-source-packages; do
165         p_specfile=%fedora-dir/$p/%fedora-branch/$p.spec
166         p_depcache=%fedora-dir/$p/%fedora-branch/.depcache
167         if ! test -f "$p_depcache" || test "$p_specfile" -nt "$p_depcache";
168         then
169             rm -f $p_depcache
170             for b in $(rpmspec -q --provides $p_specfile 2>/dev/null | awk '{print $1}'); do
171                 echo "bin2src['$b']='$p'" >> $p_depcache
172             done
173         fi
174         source $p_depcache
175     done
176
177     for b in $(rpmspec -q --buildrequires $specfile 2>/dev/null |
178                awk '{print $1}'); do
179         # Find the source package that produces these binary requirements.
180         echo ${bin2src[$b]}
181     done | grep -v '^$' | sort -u
182 }