##// END OF EJS Templates
pyoxidizer: add hooks to inject extra python packages and install files...
Augie Fackler -
r48443:11f834e7 default
parent child Browse files
Show More
@@ -1,273 +1,290 b''
1 1 # The following variables can be passed in as parameters:
2 2 #
3 3 # VERSION
4 4 # Version string of program being produced.
5 5 #
6 6 # MSI_NAME
7 7 # Root name of MSI installer.
8 8 #
9 9 # EXTRA_MSI_FEATURES
10 10 # ; delimited string of extra features to advertise in the built MSA.
11 11 #
12 12 # SIGNING_PFX_PATH
13 13 # Path to code signing certificate to use.
14 14 #
15 15 # SIGNING_PFX_PASSWORD
16 16 # Password to code signing PFX file defined by SIGNING_PFX_PATH.
17 17 #
18 18 # SIGNING_SUBJECT_NAME
19 19 # String fragment in code signing certificate subject name used to find
20 20 # code signing certificate in Windows certificate store.
21 21 #
22 22 # TIME_STAMP_SERVER_URL
23 23 # URL of time-stamp token authority (RFC 3161) servers to stamp code signatures.
24 24
25 25 ROOT = CWD + "/../.."
26 26
27 27 VERSION = VARS.get("VERSION", "5.8")
28 28 MSI_NAME = VARS.get("MSI_NAME", "mercurial")
29 29 EXTRA_MSI_FEATURES = VARS.get("EXTRA_MSI_FEATURES")
30 30 SIGNING_PFX_PATH = VARS.get("SIGNING_PFX_PATH")
31 31 SIGNING_PFX_PASSWORD = VARS.get("SIGNING_PFX_PASSWORD", "")
32 32 SIGNING_SUBJECT_NAME = VARS.get("SIGNING_SUBJECT_NAME")
33 33 TIME_STAMP_SERVER_URL = VARS.get("TIME_STAMP_SERVER_URL", "http://timestamp.digicert.com")
34 34
35 35 IS_WINDOWS = "windows" in BUILD_TARGET_TRIPLE
36 36
37 37 # Code to run in Python interpreter.
38 38 RUN_CODE = "import hgdemandimport; hgdemandimport.enable(); from mercurial import dispatch; dispatch.run()"
39 39
40 40 set_build_path(ROOT + "/build/pyoxidizer")
41 41
42 42 def make_distribution():
43 43 return default_python_distribution(python_version = "3.9")
44 44
45 45 def resource_callback(policy, resource):
46 46 if not IS_WINDOWS:
47 47 resource.add_location = "in-memory"
48 48 return
49 49
50 50 # We use a custom resource routing policy to influence where things are loaded
51 51 # from.
52 52 #
53 53 # For Python modules and resources, we load from memory if they are in
54 54 # the standard library and from the filesystem if not. This is because
55 55 # parts of Mercurial and some 3rd party packages aren't yet compatible
56 56 # with memory loading.
57 57 #
58 58 # For Python extension modules, we load from the filesystem because
59 59 # this yields greatest compatibility.
60 60 if type(resource) in ("PythonModuleSource", "PythonPackageResource", "PythonPackageDistributionResource"):
61 61 if resource.is_stdlib:
62 62 resource.add_location = "in-memory"
63 63 else:
64 64 resource.add_location = "filesystem-relative:lib"
65 65
66 66 elif type(resource) == "PythonExtensionModule":
67 67 resource.add_location = "filesystem-relative:lib"
68 68
69 69 def make_exe(dist):
70 70 """Builds a Rust-wrapped Mercurial binary."""
71 71 packaging_policy = dist.make_python_packaging_policy()
72 72
73 73 # Extension may depend on any Python functionality. Include all
74 74 # extensions.
75 75 packaging_policy.extension_module_filter = "all"
76 76 packaging_policy.resources_location = "in-memory"
77 77 if IS_WINDOWS:
78 78 packaging_policy.resources_location_fallback = "filesystem-relative:lib"
79 79 packaging_policy.register_resource_callback(resource_callback)
80 80
81 81 config = dist.make_python_interpreter_config()
82 82 config.allocator_backend = "default"
83 83 config.run_command = RUN_CODE
84 84
85 85 # We want to let the user load extensions from the file system
86 86 config.filesystem_importer = True
87 87
88 88 # We need this to make resourceutil happy, since it looks for sys.frozen.
89 89 config.sys_frozen = True
90 90 config.legacy_windows_stdio = True
91 91
92 92 exe = dist.to_python_executable(
93 93 name = "hg",
94 94 packaging_policy = packaging_policy,
95 95 config = config,
96 96 )
97 97
98 98 # Add Mercurial to resources.
99 99 exe.add_python_resources(exe.pip_install(["--verbose", ROOT]))
100 100
101 101 # On Windows, we install extra packages for convenience.
102 102 if IS_WINDOWS:
103 103 exe.add_python_resources(
104 104 exe.pip_install(["-r", ROOT + "/contrib/packaging/requirements-windows-py3.txt"]),
105 105 )
106 extra_packages = VARS.get("extra_py_packages", "")
107 if extra_packages:
108 for extra in extra_packages.split(","):
109 extra_src, pkgs = extra.split("=")
110 pkgs = pkgs.split(":")
111 exe.add_python_resources(exe.read_package_root(extra_src, pkgs))
106 112
107 113 return exe
108 114
109 115 def make_manifest(dist, exe):
110 116 m = FileManifest()
111 117 m.add_python_resource(".", exe)
112 118
113 119 return m
114 120
115 121
116 122 # This adjusts the InstallManifest produced from exe generation to provide
117 123 # additional files found in a Windows install layout.
118 124 def make_windows_install_layout(manifest):
119 125 # Copy various files to new install locations. This can go away once
120 126 # we're using the importlib resource reader.
121 127 RECURSIVE_COPIES = {
122 128 "lib/mercurial/locale/": "locale/",
123 129 "lib/mercurial/templates/": "templates/",
124 130 }
125 131 for (search, replace) in RECURSIVE_COPIES.items():
126 132 for path in manifest.paths():
127 133 if path.startswith(search):
128 134 new_path = path.replace(search, replace)
129 135 print("copy %s to %s" % (path, new_path))
130 136 file = manifest.get_file(path)
131 137 manifest.add_file(file, path = new_path)
132 138
133 139 # Similar to above, but with filename pattern matching.
134 140 # lib/mercurial/helptext/**/*.txt -> helptext/
135 141 # lib/mercurial/defaultrc/*.rc -> defaultrc/
136 142 for path in manifest.paths():
137 143 if path.startswith("lib/mercurial/helptext/") and path.endswith(".txt"):
138 144 new_path = path[len("lib/mercurial/"):]
139 145 elif path.startswith("lib/mercurial/defaultrc/") and path.endswith(".rc"):
140 146 new_path = path[len("lib/mercurial/"):]
141 147 else:
142 148 continue
143 149
144 150 print("copying %s to %s" % (path, new_path))
145 151 manifest.add_file(manifest.get_file(path), path = new_path)
146 152
153 extra_install_files = VARS.get("extra_install_files", "")
154 if extra_install_files:
155 for extra in extra_install_files.split(","):
156 print("adding extra files from %s" % extra)
157 # TODO: I expected a ** glob to work, but it didn't.
158 #
159 # TODO: I know this has forward-slash paths. As far as I can tell,
160 # backslashes don't ever match glob() expansions in
161 # tugger-starlark, even on Windows.
162 manifest.add_manifest(glob(include=[extra + "/*/*"], strip_prefix=extra+"/"))
163
147 164 # We also install a handful of additional files.
148 165 EXTRA_CONTRIB_FILES = [
149 166 "bash_completion",
150 167 "hgweb.fcgi",
151 168 "hgweb.wsgi",
152 169 "logo-droplets.svg",
153 170 "mercurial.el",
154 171 "mq.el",
155 172 "tcsh_completion",
156 173 "tcsh_completion_build.sh",
157 174 "xml.rnc",
158 175 "zsh_completion",
159 176 ]
160 177
161 178 for f in EXTRA_CONTRIB_FILES:
162 179 manifest.add_file(FileContent(path = ROOT + "/contrib/" + f), directory = "contrib")
163 180
164 181 # Individual files with full source to destination path mapping.
165 182 EXTRA_FILES = {
166 183 "contrib/hgk": "contrib/hgk.tcl",
167 184 "contrib/win32/postinstall.txt": "ReleaseNotes.txt",
168 185 "contrib/win32/ReadMe.html": "ReadMe.html",
169 186 "doc/style.css": "doc/style.css",
170 187 "COPYING": "Copying.txt",
171 188 }
172 189
173 190 for source, dest in EXTRA_FILES.items():
174 191 print("adding extra file %s" % dest)
175 192 manifest.add_file(FileContent(path = ROOT + "/" + source), path = dest)
176 193
177 194 # And finally some wildcard matches.
178 195 manifest.add_manifest(glob(
179 196 include = [ROOT + "/contrib/vim/*"],
180 197 strip_prefix = ROOT + "/"
181 198 ))
182 199 manifest.add_manifest(glob(
183 200 include = [ROOT + "/doc/*.html"],
184 201 strip_prefix = ROOT + "/"
185 202 ))
186 203
187 204 # But we don't ship hg-ssh on Windows, so exclude its documentation.
188 205 manifest.remove("doc/hg-ssh.8.html")
189 206
190 207 return manifest
191 208
192 209
193 210 def make_msi(manifest):
194 211 manifest = make_windows_install_layout(manifest)
195 212
196 213 if "x86_64" in BUILD_TARGET_TRIPLE:
197 214 platform = "x64"
198 215 else:
199 216 platform = "x86"
200 217
201 218 manifest.add_file(
202 219 FileContent(path = ROOT + "/contrib/packaging/wix/COPYING.rtf"),
203 220 path = "COPYING.rtf",
204 221 )
205 222 manifest.remove("Copying.txt")
206 223 manifest.add_file(
207 224 FileContent(path = ROOT + "/contrib/win32/mercurial.ini"),
208 225 path = "defaultrc/mercurial.rc",
209 226 )
210 227 manifest.add_file(
211 228 FileContent(filename = "editor.rc", content = "[ui]\neditor = notepad\n"),
212 229 path = "defaultrc/editor.rc",
213 230 )
214 231
215 232 wix = WiXInstaller("hg", "%s-%s.msi" % (MSI_NAME, VERSION))
216 233
217 234 # Materialize files in the manifest to the install layout.
218 235 wix.add_install_files(manifest)
219 236
220 237 # From mercurial.wxs.
221 238 wix.install_files_root_directory_id = "INSTALLDIR"
222 239
223 240 # Pull in our custom .wxs files.
224 241 defines = {
225 242 "PyOxidizer": "1",
226 243 "Platform": platform,
227 244 "Version": VERSION,
228 245 "Comments": "Installs Mercurial version %s" % VERSION,
229 246 "PythonVersion": "3",
230 247 "MercurialHasLib": "1",
231 248 }
232 249
233 250 if EXTRA_MSI_FEATURES:
234 251 defines["MercurialExtraFeatures"] = EXTRA_MSI_FEATURES
235 252
236 253 wix.add_wxs_file(
237 254 ROOT + "/contrib/packaging/wix/mercurial.wxs",
238 255 preprocessor_parameters=defines,
239 256 )
240 257
241 258 # Our .wxs references to other files. Pull those into the build environment.
242 259 for f in ("defines.wxi", "guids.wxi", "COPYING.rtf"):
243 260 wix.add_build_file(f, ROOT + "/contrib/packaging/wix/" + f)
244 261
245 262 wix.add_build_file("mercurial.ico", ROOT + "/contrib/win32/mercurial.ico")
246 263
247 264 return wix
248 265
249 266
250 267 def register_code_signers():
251 268 if not IS_WINDOWS:
252 269 return
253 270
254 271 if SIGNING_PFX_PATH:
255 272 signer = code_signer_from_pfx_file(SIGNING_PFX_PATH, SIGNING_PFX_PASSWORD)
256 273 elif SIGNING_SUBJECT_NAME:
257 274 signer = code_signer_from_windows_store_subject(SIGNING_SUBJECT_NAME)
258 275 else:
259 276 signer = None
260 277
261 278 if signer:
262 279 signer.set_time_stamp_server(TIME_STAMP_SERVER_URL)
263 280 signer.activate()
264 281
265 282
266 283 register_code_signers()
267 284
268 285 register_target("distribution", make_distribution)
269 286 register_target("exe", make_exe, depends = ["distribution"])
270 287 register_target("app", make_manifest, depends = ["distribution", "exe"], default = True)
271 288 register_target("msi", make_msi, depends = ["app"])
272 289
273 290 resolve_targets()
General Comments 0
You need to be logged in to leave comments. Login now