395 lines
12 KiB
Bash
Executable file
395 lines
12 KiB
Bash
Executable file
#!/usr/bin/env nix-shell
|
|
#!nix-shell -I nixpkgs=./. -i bash -p curl jq nix gnused
|
|
# shellcheck shell=bash
|
|
|
|
set -Eeuo pipefail
|
|
|
|
rids=({linux-{,musl-}{arm,arm64,x64},osx-{arm64,x64},win-{arm64,x64,x86}})
|
|
|
|
release () {
|
|
local content="$1"
|
|
local version="$2"
|
|
|
|
jq -r '.releases[] | select(."release-version" == "'"$version"'")' <<< "$content"
|
|
}
|
|
|
|
release_files () {
|
|
local release="$1"
|
|
local expr="$2"
|
|
|
|
jq -r '[('"$expr"').files[] | select(.name | test("^.*.tar.gz$"))]' <<< "$release"
|
|
}
|
|
|
|
release_platform_attr () {
|
|
local release_files="$1"
|
|
local platform="$2"
|
|
local attr="$3"
|
|
|
|
jq -r '.[] | select((.rid == "'"$platform"'") and (.name | contains("composite") | not)) | ."'"$attr"'"' <<< "$release_files"
|
|
}
|
|
|
|
platform_sources () {
|
|
local release_files="$1"
|
|
|
|
echo "srcs = {"
|
|
for rid in "${rids[@]}"; do
|
|
local url hash
|
|
|
|
url=$(release_platform_attr "$release_files" "$rid" url)
|
|
hash=$(release_platform_attr "$release_files" "$rid" hash)
|
|
|
|
[[ -z "$url" || -z "$hash" ]] && continue
|
|
|
|
hash=$(nix-hash --to-sri --type sha512 "$hash")
|
|
|
|
echo " $rid = {
|
|
url = \"$url\";
|
|
hash = \"$hash\";
|
|
};"
|
|
done
|
|
echo " };"
|
|
}
|
|
|
|
nuget_url="$(curl -fsSL "https://api.nuget.org/v3/index.json" | jq --raw-output '.resources[] | select(."@type" == "PackageBaseAddress/3.0.0")."@id"')"
|
|
|
|
generate_package_list() {
|
|
local version="$1" indent="$2"
|
|
shift 2
|
|
local pkgs=( "$@" ) pkg url hash
|
|
|
|
for pkg in "${pkgs[@]}"; do
|
|
url=${nuget_url}${pkg,,}/${version,,}/${pkg,,}.${version,,}.nupkg
|
|
if ! hash=$(nix-prefetch-url "$url"); then
|
|
echo "Failed to fetch hash for $url" >&2
|
|
exit 1
|
|
fi
|
|
hash=$(nix-hash --to-sri --type sha256 "$hash")
|
|
|
|
echo "$indent(fetchNupkg { pname = \"${pkg}\"; version = \"${version}\"; hash = \"${hash}\"; })"
|
|
done
|
|
}
|
|
|
|
versionAtLeast () {
|
|
local cur_version=$1 min_version=$2
|
|
printf "%s\0%s" "$min_version" "$cur_version" | sort -zVC
|
|
}
|
|
|
|
# These packages are implicitly references by the build process,
|
|
# based on the specific project configurations (RIDs, used features, etc.)
|
|
# They are always referenced with the same version as the SDK used for building.
|
|
# Since we lock nuget dependencies, when these packages are included in the generated
|
|
# lock files (deps.nix), every update of SDK required those lock files to be
|
|
# updated to reflect the new versions of these packages - otherwise, the build
|
|
# would fail due to missing dependencies.
|
|
#
|
|
# Moving them to a separate list stored alongside the SDK package definitions,
|
|
# and implicitly including them along in buildDotnetModule allows us
|
|
# to make updating .NET SDK packages a lot easier - we now just update
|
|
# the versions of these packages in one place, and all packages that
|
|
# use buildDotnetModule continue building with the new .NET version without changes.
|
|
#
|
|
# Keep in mind that there is no canonical list of these implicitly
|
|
# referenced packages - this list was created based on looking into
|
|
# the deps.nix files of existing packages, and which dependencies required
|
|
# updating after a SDK version bump.
|
|
#
|
|
# Due to this, make sure to check if new SDK versions introduce any new packages.
|
|
# This should not happend in minor or bugfix updates, but probably happens
|
|
# with every new major .NET release.
|
|
aspnetcore_packages () {
|
|
local version=$1
|
|
local pkgs=(
|
|
Microsoft.AspNetCore.App.Ref
|
|
)
|
|
|
|
generate_package_list "$version" ' ' "${pkgs[@]}"
|
|
}
|
|
|
|
aspnetcore_target_packages () {
|
|
local version=$1
|
|
local rid=$2
|
|
local pkgs=(
|
|
"Microsoft.AspNetCore.App.Runtime.$rid"
|
|
)
|
|
|
|
generate_package_list "$version" ' ' "${pkgs[@]}"
|
|
}
|
|
|
|
netcore_packages () {
|
|
local version=$1
|
|
local pkgs=(
|
|
Microsoft.NETCore.DotNetAppHost
|
|
Microsoft.NETCore.App.Ref
|
|
)
|
|
|
|
if ! versionAtLeast "$version" 9; then
|
|
pkgs+=(
|
|
Microsoft.NETCore.DotNetHost
|
|
Microsoft.NETCore.DotNetHostPolicy
|
|
Microsoft.NETCore.DotNetHostResolver
|
|
)
|
|
fi
|
|
|
|
if versionAtLeast "$version" 7; then
|
|
pkgs+=(
|
|
Microsoft.DotNet.ILCompiler
|
|
)
|
|
fi
|
|
|
|
if versionAtLeast "$version" 8; then
|
|
pkgs+=(
|
|
Microsoft.NET.ILLink.Tasks
|
|
)
|
|
fi
|
|
|
|
generate_package_list "$version" ' ' "${pkgs[@]}"
|
|
}
|
|
|
|
netcore_host_packages () {
|
|
local version=$1
|
|
local rid=$2
|
|
local pkgs=(
|
|
"Microsoft.NETCore.App.Crossgen2.$rid"
|
|
)
|
|
|
|
local min_ilcompiler=
|
|
case "$rid" in
|
|
linux-musl-arm) ;;
|
|
linux-arm) ;;
|
|
win-x86) ;;
|
|
osx-arm64) min_ilcompiler=8 ;;
|
|
*) min_ilcompiler=7 ;;
|
|
esac
|
|
|
|
if [[ -n "$min_ilcompiler" ]] && versionAtLeast "$version" "$min_ilcompiler"; then
|
|
pkgs+=(
|
|
"runtime.$rid.Microsoft.DotNet.ILCompiler"
|
|
)
|
|
fi
|
|
|
|
generate_package_list "$version" ' ' "${pkgs[@]}"
|
|
}
|
|
|
|
netcore_target_packages () {
|
|
local version=$1
|
|
local rid=$2
|
|
local pkgs=(
|
|
"Microsoft.NETCore.App.Host.$rid"
|
|
"Microsoft.NETCore.App.Runtime.$rid"
|
|
"runtime.$rid.Microsoft.NETCore.DotNetAppHost"
|
|
)
|
|
|
|
if ! versionAtLeast "$version" 9; then
|
|
pkgs+=(
|
|
"runtime.$rid.Microsoft.NETCore.DotNetHost"
|
|
"runtime.$rid.Microsoft.NETCore.DotNetHostPolicy"
|
|
"runtime.$rid.Microsoft.NETCore.DotNetHostResolver"
|
|
)
|
|
case "$rid" in
|
|
linux-musl-arm*) ;;
|
|
win-arm64) ;;
|
|
*) pkgs+=(
|
|
"Microsoft.NETCore.App.Runtime.Mono.$rid"
|
|
) ;;
|
|
esac
|
|
fi
|
|
|
|
generate_package_list "$version" ' ' "${pkgs[@]}"
|
|
}
|
|
|
|
usage () {
|
|
echo "Usage: $pname [[--sdk] [-o output] sem-version] ...
|
|
Get updated dotnet src (platform - url & sha512) expressions for specified versions
|
|
|
|
Examples:
|
|
$pname 6.0.14 7.0.201 - specific x.y.z versions
|
|
$pname 6.0 7.0 - latest x.y versions
|
|
" >&2
|
|
}
|
|
|
|
update() {
|
|
local -r sem_version=$1 sdk=$2
|
|
local output=$3
|
|
|
|
local patch_specified=false
|
|
# Check if a patch was specified as an argument.
|
|
# If so, generate file for the specific version.
|
|
# If only x.y version was provided, get the latest patch
|
|
# version of the given x.y version.
|
|
if [[ "$sem_version" =~ ^[0-9]{1,}\.[0-9]{1,}\.[0-9]{1,} ]]; then
|
|
patch_specified=true
|
|
elif [[ ! "$sem_version" =~ ^[0-9]{1,}\.[0-9]{1,}$ ]]; then
|
|
usage
|
|
return 1
|
|
fi
|
|
|
|
: ${output:="$(dirname "${BASH_SOURCE[0]}")"/versions/$sem_version.nix}
|
|
echo "Generating $output"
|
|
|
|
# Make sure the x.y version is properly passed to .NET release metadata url.
|
|
# Then get the json file and parse it to find the latest patch release.
|
|
local major_minor content major_minor_patch
|
|
major_minor=$(sed 's/^\([0-9]*\.[0-9]*\).*$/\1/' <<< "$sem_version")
|
|
content=$(curl -fsSL https://dotnetcli.blob.core.windows.net/dotnet/release-metadata/"$major_minor"/releases.json)
|
|
if [[ -n $sdk ]]; then
|
|
major_minor_patch=$(
|
|
jq -r --arg version "$sem_version" '
|
|
.releases[] |
|
|
select(.sdks[].version == $version) |
|
|
."release-version"' <<< "$content")
|
|
else
|
|
major_minor_patch=$([ "$patch_specified" == true ] && echo "$sem_version" || jq -r '."latest-release"' <<< "$content")
|
|
fi
|
|
local major_minor_underscore=${major_minor/./_}
|
|
|
|
local release_content aspnetcore_version runtime_version
|
|
local -a sdk_versions
|
|
|
|
release_content=$(release "$content" "$major_minor_patch")
|
|
aspnetcore_version=$(jq -r '."aspnetcore-runtime".version' <<< "$release_content")
|
|
runtime_version=$(jq -r '.runtime.version' <<< "$release_content")
|
|
|
|
if [[ -n $sdk ]]; then
|
|
sdk_versions=("$sem_version")
|
|
else
|
|
mapfile -t sdk_versions < <(jq -r '.sdks[] | .version' <<< "$release_content" | sort -rn)
|
|
fi
|
|
|
|
# If patch was not specified, check if the package is already the latest version
|
|
# If it is, exit early
|
|
if [ "$patch_specified" == false ] && [ -f "$output" ]; then
|
|
local current_version
|
|
current_version=$(nix-instantiate --eval -E "(import $output { \
|
|
buildAspNetCore = { ... }: {}; \
|
|
buildNetSdk = { ... }: {}; \
|
|
buildNetRuntime = { ... }: {}; \
|
|
fetchNupkg = { ... }: {}; \
|
|
}).release_${major_minor_underscore}" | jq -r)
|
|
|
|
if [[ "$current_version" == "$major_minor_patch" ]]; then
|
|
echo "Nothing to update."
|
|
return
|
|
fi
|
|
fi
|
|
|
|
local aspnetcore_files runtime_files
|
|
aspnetcore_files="$(release_files "$release_content" .\"aspnetcore-runtime\")"
|
|
runtime_files="$(release_files "$release_content" .runtime)"
|
|
|
|
local channel_version support_phase
|
|
channel_version=$(jq -r '."channel-version"' <<< "$content")
|
|
support_phase=$(jq -r '."support-phase"' <<< "$content")
|
|
|
|
local aspnetcore_sources runtime_sources
|
|
aspnetcore_sources="$(platform_sources "$aspnetcore_files")"
|
|
runtime_sources="$(platform_sources "$runtime_files")"
|
|
|
|
result=$(mktemp)
|
|
trap "rm -f $result" TERM INT EXIT
|
|
|
|
(
|
|
echo "{ buildAspNetCore, buildNetRuntime, buildNetSdk, fetchNupkg }:
|
|
|
|
# v$channel_version ($support_phase)
|
|
|
|
let
|
|
commonPackages = ["
|
|
aspnetcore_packages "${aspnetcore_version}"
|
|
netcore_packages "${runtime_version}"
|
|
echo " ];
|
|
|
|
hostPackages = {"
|
|
for rid in "${rids[@]}"; do
|
|
echo " $rid = ["
|
|
netcore_host_packages "${runtime_version}" "$rid"
|
|
echo " ];"
|
|
done
|
|
echo " };
|
|
|
|
targetPackages = {"
|
|
for rid in "${rids[@]}"; do
|
|
echo " $rid = ["
|
|
aspnetcore_target_packages "${aspnetcore_version}" "$rid"
|
|
netcore_target_packages "${runtime_version}" "$rid"
|
|
echo " ];"
|
|
done
|
|
echo " };
|
|
|
|
in rec {
|
|
release_$major_minor_underscore = \"$major_minor_patch\";
|
|
|
|
aspnetcore_$major_minor_underscore = buildAspNetCore {
|
|
version = \"${aspnetcore_version}\";
|
|
$aspnetcore_sources
|
|
};
|
|
|
|
runtime_$major_minor_underscore = buildNetRuntime {
|
|
version = \"${runtime_version}\";
|
|
$runtime_sources
|
|
};"
|
|
|
|
local -A feature_bands
|
|
unset latest_sdk
|
|
|
|
for sdk_version in "${sdk_versions[@]}"; do
|
|
local sdk_base_version=${sdk_version%-*}
|
|
local feature_band=${sdk_base_version:0:-2}xx
|
|
# sometimes one release has e.g. both 8.0.202 and 8.0.203
|
|
[[ ! ${feature_bands[$feature_band]+true} ]] || continue
|
|
feature_bands[$feature_band]=$sdk_version
|
|
local sdk_files sdk_sources
|
|
sdk_files="$(release_files "$release_content" ".sdks[] | select(.version == \"$sdk_version\")")"
|
|
sdk_sources="$(platform_sources "$sdk_files")"
|
|
local sdk_attrname=sdk_${feature_band//./_}
|
|
[[ -v latest_sdk ]] || local latest_sdk=$sdk_attrname
|
|
|
|
echo "
|
|
$sdk_attrname = buildNetSdk {
|
|
version = \"${sdk_version}\";
|
|
$sdk_sources
|
|
inherit commonPackages hostPackages targetPackages;
|
|
};"
|
|
done
|
|
|
|
if [[ -n $sdk ]]; then
|
|
echo "
|
|
sdk = sdk_$major_minor_underscore;
|
|
"
|
|
fi
|
|
|
|
echo "
|
|
sdk_$major_minor_underscore = $latest_sdk;
|
|
}"
|
|
)> "${result}"
|
|
|
|
cp "${result}" "$output"
|
|
echo "Generated $output"
|
|
}
|
|
|
|
main () {
|
|
local pname sdk output
|
|
pname=$(basename "$0")
|
|
|
|
sdk=
|
|
output=
|
|
|
|
while [ $# -gt 0 ]; do
|
|
case $1 in
|
|
--sdk)
|
|
shift
|
|
sdk=1
|
|
;;
|
|
-o)
|
|
shift
|
|
output=$1
|
|
shift
|
|
;;
|
|
*)
|
|
update "$1" "$sdk" "$output"
|
|
shift
|
|
;;
|
|
esac
|
|
done
|
|
}
|
|
|
|
main "$@"
|