-
Notifications
You must be signed in to change notification settings - Fork 205
/
Copy pathversions.sh
executable file
·195 lines (181 loc) · 6.03 KB
/
versions.sh
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
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
#!/bin/bash
set -euo pipefail
cd "$(dirname "$(readlink -f "$BASH_SOURCE")")"
yq='./.yq'
# https://github.com/mikefarah/yq/releases
# TODO detect host architecture
yqUrl='https://github.com/mikefarah/yq/releases/download/v4.40.5/yq_linux_amd64'
yqSha256='0d6aaf1cf44a8d18fbc7ed0ef14f735a8df8d2e314c4cc0f0242d35c0a440c95'
if command -v xq-python &> /dev/null; then
# if we have the Python-based "yq" installed, it also comes with "xq-python" (at least, from Debian), which does meet our needs here
yq='xq-python'
else
if command -v yq &> /dev/null && yq --help |& grep -F -- ' --input-format' | grep -qF xml; then
# if we have a "yq" on the host, make sure it's the Go-based "yq" (not the Python-based one, handled above)
yq='yq'
elif [ ! -x "$yq" ] || ! sha256sum <<<"$yqSha256 *$yq" --quiet --strict --check; then
wget -qO "$yq.new" "$yqUrl"
sha256sum <<<"$yqSha256 *$yq.new" --quiet --strict --check
chmod +x "$yq.new"
"$yq.new" --version
mv "$yq.new" "$yq"
fi
yq+=' --input-format xml'
fi
releases="$(
wget -qO- 'https://updates.drupal.org/release-history/drupal/current' \
| $yq -r '@json' \
| jq -c '
# https://stackoverflow.com/a/75770668/433558
def semver:
sub("[+].*$"; "")
| capture("^(?<v>[^-]+)(?:-(?<p>.*))?$") | [.v, .p // empty]
| map(split(".") | map(tonumber? // .))
| .[1] |= (. // {})
;
.project
| [
.supported_branches? // empty,
.supported_majors? // empty
| split(",")
| .[]
| rtrimstr(".")
] as $versions
| reduce (
.releases.release[]
# skip "dev" releases entirely (download artifacts are too unstable / change too often)
| select(
.status == "published"
and (
.version
| endswith("-dev")
| not
)
)
# add a key for the appropriate "X.Y" or "X.Y-rc" value
| .folder = (.version | ([ split("[.-]"; "") | if .[0] == "7" then .[0] else .[0,1] end ] | join(".")) + if index("-") then "-rc" else "" end)
# filter to *just* versions that the upstream file claims are actually supported ("supported_branches")
| select((.folder | rtrimstr("-rc")) | IN($versions[]) | not|not)
) as $rel ({}; .[$rel.version] = $rel)
| to_entries
# put all releases in sorted order
| sort_by(.value.version | semver)
| reverse
# ... so we can remove all but the most recent (pre)?release for each branch / folder
| unique_by(
.value.folder
| rtrimstr("-rc")
| split("[.-]"; "")
| map(tonumber? // .)
)
| reverse
| map(.key = .value.folder)
| from_entries
'
)"
versions=( "$@" )
if [ ${#versions[@]} -eq 0 ]; then
# if no versions are specified, assume the "canonical" list of supported versions
versions="$(jq <<<"$releases" -r '[ .[].folder | @sh ] | join(" ")')"
eval "versions=( $versions )"
json='{}'
else
json="$(< versions.json)"
fi
versions=( "${versions[@]%/}" )
for version in "${versions[@]}"; do
export version rcVersion="${version%-rc}"
doc="$(jq <<<"$releases" -c '.[env.version] // empty')"
if [ -z "$doc" ]; then
echo >&2 "warning: skipping/removing '$version' (does not exist or is not supported upstream)"
json="$(jq <<<"$json" -c '.[env.version] = null')"
continue
fi
doc="$(jq <<<"$doc" -c '
first(.files.file[] | select(.archive_type == "tar.gz")) as $file
| {
version: .version,
url: $file.url,
md5: $file.md5,
date: (.date | tonumber),
notes: .release_link,
# TODO adjust this in a way that is easier to manage over time (semi-automatic variant combinations, for example, based on availability/supported status of upstream PHP images)
phpVersions: (
# https://www.drupal.org/docs/system-requirements/php-requirements
[
# https://www.drupal.org/project/drupal/releases/10.2.0-rc1#php-deps
# Drupal now supports PHP 8.3 and recommends at least PHP 8.2.
if env.version | IN("10.0") then empty else
"8.3"
end,
# https://www.drupal.org/node/3413288 ("Drupal 11 will require PHP 8.3")
if env.version | IN("10.0", "10.3") then
"8.2"
else empty end,
if env.version | IN("10.0") then
"8.1"
else empty end,
# https://www.drupal.org/docs/system-requirements/php-requirements
empty
]
),
variants: [
"bookworm",
"bullseye",
"alpine3.21",
"alpine3.20",
empty
| if startswith("alpine") then empty else
"apache-" + .
end,
"fpm-" + .
],
}
')"
fullVersion="$(jq <<<"$doc" -r '.version')"
[ -n "$fullVersion" ] # sanity check
if [ "$rcVersion" != "$version" ] && gaFullVersion="$(jq <<<"$json" -er '.[env.rcVersion] | if . then .version else empty end')"; then
# Drupal pre-releases appear to be only for .0, so if our pre-release now has a relevant GA, it should go away 👀
# just in case, we'll also do a version comparison to make sure we don't have a pre-release that's newer than the relevant GA
latestVersion="$({ echo "$fullVersion"; echo "$gaFullVersion"; } | sort -V | tail -1)"
if [[ "$fullVersion" == "$gaFullVersion"* ]] || [ "$latestVersion" = "$gaFullVersion" ]; then
# "x.y.z-rc1" == x.y.z*
json="$(jq <<<"$json" -c 'del(.[env.version])')"
continue
fi
fi
composerVersion="$(
wget -qO- "https://github.com/drupal/drupal/raw/$fullVersion/composer.lock" \
| jq -r '
(.packages, ."packages-dev")[]
| select(.name == "composer/composer")
| .version
| split(".")[0:1] | join(".")
' \
|| :
)"
if [ -z "$composerVersion" ]; then
echo >&2 "error: cannot find composer version for '$version' ('$fullVersion')"
exit 1
fi
if [ -n "$composerVersion" ]; then
export composerVersion
doc="$(jq <<<"$doc" -c '.composer = { version: env.composerVersion }')"
fi
echo "$version: $fullVersion${composerVersion:+ (composer $composerVersion)}"
json="$(
jq <<<"$json" -c --argjson doc "$doc" '
.[env.version] = $doc
'
)"
# make sure pre-release versions have a placeholder for GA
if [ "$version" != "$rcVersion" ]; then
json="$(jq <<<"$json" -c '.[env.rcVersion] //= null')"
fi
done
jq <<<"$json" '
to_entries
| sort_by(.key | split("[.-]"; "") | map(tonumber? // .))
| reverse
| from_entries
' > versions.json