forked from JuliaPackaging/Yggdrasil
-
Notifications
You must be signed in to change notification settings - Fork 0
/
generate_buildjl.jl
executable file
·242 lines (207 loc) · 8.38 KB
/
generate_buildjl.jl
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
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
#!/usr/bin/env julia
using GitHub, BinaryBuilder, Pkg, Pkg.PlatformEngines, SHA
"""
extract_platform_key(path::AbstractString)
Given the path to a tarball, return the platform key of that tarball. If none
can be found, prints a warning and return the current platform suffix.
"""
function extract_platform_key(path::AbstractString)
try
return extract_name_version_platform_key(path)[3]
catch
@warn("Could not extract the platform key of $(path); continuing...")
return platform_key_abi()
end
end
"""
extract_name_version_platform_key(path::AbstractString)
Given the path to a tarball, return the name, platform key and version of that
tarball. If any of those things cannot be found, throw an error.
"""
function extract_name_version_platform_key(path::AbstractString)
m = match(r"^(.*?)\.v(.*?)\.([^\.\-]+-[^\.\-]+-([^\-]+-){0,2}[^\-]+).tar.gz$", basename(path))
if m === nothing
error("Could not parse name, platform key and version from $(path)")
end
name = m.captures[1]
version = VersionNumber(m.captures[2])
platkey = platform_key_abi(m.captures[3])
return name, version, platkey
end
function product_hashes_from_github_release(repo_name::AbstractString, tag_name::AbstractString;
verbose::Bool = true)
# Get list of files within this release
release = GitHub.gh_get_json(GitHub.DEFAULT_API, "/repos/$(repo_name)/releases/tags/$(tag_name)", auth=BinaryBuilder.github_auth())
# Try to extract the platform key from each, use that to find all tarballs
function can_extract_platform(filename)
# Short-circuit build.jl because that's quite often there. :P
if startswith(filename, "build") && endswith(filename, ".jl")
return false
end
unknown_platform = typeof(extract_platform_key(filename)) <: UnknownPlatform
if unknown_platform && verbose
@info("Ignoring file $(filename); can't extract its platform key")
end
return !unknown_platform
end
assets = [a for a in release["assets"] if can_extract_platform(a["name"])]
# Download each tarball, hash it, and reconstruct product_hashes.
product_hashes = Dict()
mktempdir() do d
for asset in assets
# For each asset (tarball), download it
filepath = joinpath(d, asset["name"])
url = asset["browser_download_url"]
download(url, filepath)
# Hash it
hash = open(filepath) do file
return bytes2hex(sha256(file))
end
# Then fit it into our product_hashes
file_triplet = triplet(extract_platform_key(asset["name"]))
product_hashes[file_triplet] = (asset["name"], hash)
if verbose
@info("Calculated $hash for $(asset["name"])")
end
end
end
return product_hashes
end
# `repr` calls `BinaryBuilder.repr` which does not print the `prefix, `
# first argument that is needed by `BinaryProvider`.
function _repr(p::Product)
return replace(repr(p), "(" => "(prefix, ")
end
# `Pkg.CompilerABI` has diverged from `BinaryProvider.CompilerABI`.
# This function fixes common differences.
function _pkey(platform)
pkey = string(platform_key_abi(platform))
for (i, j) in [(3, 4), (4, 7), (5, 8)]
pkey = replace(pkey, "CompilerABI(libgfortran_version=v\"$i.0.0\")" => "CompilerABI(:gcc$j)")
end
for i in ["03", "11"]
pkey = replace(pkey, "CompilerABI(cxxstring_abi=:cxx$i)" => "CompilerABI(:gcc_any, :cxx$i)")
end
return pkey
end
function print_buildjl(io::IO, products::Vector, product_hashes::Dict,
bin_path::AbstractString)
print(io, """
using BinaryProvider # requires BinaryProvider 0.3.0 or later
# Parse some basic command-line arguments
const verbose = "--verbose" in ARGS
const prefix = Prefix(get([a for a in ARGS if a != "--verbose"], 1, joinpath(@__DIR__, "usr")))
""")
# Print out products
print(io, "products = [\n")
for prod in products
print(io, " $(_repr(prod)),\n")
end
print(io, "]\n\n")
# Print binary locations/tarball hashes
print(io, """
# Download binaries from hosted location
bin_prefix = "$bin_path"
# Listing of files generated by BinaryBuilder:
""")
println(io, "download_info = Dict(")
for platform in sort(collect(keys(product_hashes)))
fname, hash = product_hashes[platform]
pkey = _pkey(platform)
println(io, " $(pkey) => (\"\$bin_prefix/$(fname)\", \"$(hash)\"),")
end
println(io, ")\n")
print(io, """
# Install unsatisfied or updated dependencies:
unsatisfied = any(!satisfied(p; verbose=verbose) for p in products)
dl_info = choose_download(download_info, platform_key_abi())
if dl_info === nothing && unsatisfied
# If we don't have a compatible .tar.gz to download, complain.
# Alternatively, you could attempt to install from a separate provider,
# build from source or something even more ambitious here.
error("Your platform (\\\"\$(Sys.MACHINE)\\\", parsed as \\\"\$(triplet(platform_key_abi()))\\\") is not supported by this package!")
end
# If we have a download, and we are unsatisfied (or the version we're
# trying to install is not itself installed) then load it up!
if unsatisfied || !isinstalled(dl_info...; prefix=prefix)
# Download and install binaries
install(dl_info...; prefix=prefix, force=true, verbose=verbose)
end
# Write out a deps.jl file that will contain mappings for our products
write_deps_file(joinpath(@__DIR__, "deps.jl"), products, verbose=verbose)
""")
end
if length(ARGS) < 1 || length(ARGS) > 3
@error("Usage: generate_buildjl.jl path/to/build_tarballs.jl [<repo_name> <tag_name>]")
exit(1)
end
build_tarballs_path = ARGS[1]
@info "Build tarballs script: $(build_tarballs_path)"
src_name = basename(dirname(build_tarballs_path))
if 2 <= length(ARGS) <= 3
repo_name = ARGS[2]
else
repo_name = "JuliaBinaryWrappers/$(src_name)_jll.jl"
end
@info "Repo name: $(repo_name)"
if length(ARGS) == 3
tag_name = ARGS[3]
else
ctx = Pkg.Types.Context()
# Force-update the registry here, since we may have pushed a new version recently
BinaryBuilder.update_registry(ctx)
versions = VersionNumber[]
paths = Pkg.Operations.registered_paths(ctx.env, BinaryBuilder.jll_uuid("$(src_name)_jll"))
if any(p -> isfile(joinpath(p, "Package.toml")), paths)
# Find largest version number that matches ours in the registered paths
for path in paths
append!(versions, Pkg.Compress.load_versions(joinpath(path, "Versions.toml")))
end
end
if !isempty(versions)
last_version = maximum(versions)
tag_name = "$(src_name)-v$(last_version)"
else
@error("""Unable to determine latest version of $(src_name),
please specify it as third argument to this script:
generate_buildjl.jl $(build_tarballs_path) $(repo_name) <tag_name>""")
exit(1)
end
end
@info "Tag name: $(tag_name)"
# First, snarf out the Product variables:
if !isfile(build_tarballs_path)
@error("Unable to open $(build_tarballs_path)")
exit(1)
end
m = Module(:__anon__)
Core.eval(m, quote
using BinaryBuilder, Pkg.BinaryPlatforms
# Override BinaryBuilder functionality so that it doesn't actually do anything
# it just saves the inputs so that we can mess around with them:
_name = nothing
_version = nothing
_products = nothing
function build_tarballs(A, name, version, sources, script, platforms, products, dependencies; kwargs...)
global _name = name
global _version = version
# Peel off the functionalization of Products
if isa(products, Function)
products = products(Prefix("."))
end
global _products = products
return nothing
end
end)
include_string(m, String(read(build_tarballs_path)))
name, version, products = Core.eval(m, quote
_name, _version, _products
end)
product_hashes = product_hashes_from_github_release(repo_name, tag_name)
mkpath(joinpath(@__DIR__, "build"))
buildjl_path = joinpath(@__DIR__, "build", "build_$(name).v$(version).jl")
bin_path = "https://github.com/$(repo_name)/releases/download/$(tag_name)"
@info("Writing out to $(buildjl_path)")
open(buildjl_path, "w") do io
print_buildjl(io, products, product_hashes, bin_path)
end