-
Notifications
You must be signed in to change notification settings - Fork 34
/
image-customize
executable file
·340 lines (286 loc) · 14.3 KB
/
image-customize
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
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
#!/usr/bin/env python3
# This file is part of Cockpit.
#
# Copyright (C) 2015 Red Hat, Inc.
#
# Cockpit is free software; you can redistribute it and/or modify it
# under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation; either version 2.1 of the License, or
# (at your option) any later version.
#
# Cockpit is distributed in the hope that it will be useful, but
# WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with Cockpit; If not, see <http://www.gnu.org/licenses/>.
import argparse
import os
import subprocess
import sys
from collections.abc import Sequence
from lib.constants import BOTS_DIR, TEST_DIR
from machine import testvm
opt_quick: bool = False
opt_verbose: bool = False
opt_build_options: str = ''
stdout_disposition: int | None = None
def prepare_install_image(base_image: str, install_image: str, resize: str | None, fresh: bool) -> str:
"""Create the necessary layered image for the build/install"""
if "/" not in base_image:
base_image = os.path.join(testvm.IMAGES_DIR, base_image)
if "/" not in install_image:
install_image = os.path.join(os.path.join(TEST_DIR, "images"), os.path.basename(install_image))
qcow2_image = f"{install_image}.qcow2"
# Remove existing overlay if --fresh was requested
if fresh:
for f in [install_image, qcow2_image]:
try:
os.unlink(f)
except FileNotFoundError:
# if there is no existing overlay, that's fine
pass
if not os.path.exists(install_image):
install_image_dir = os.path.dirname(install_image)
os.makedirs(install_image_dir, exist_ok=True)
base_image = os.path.realpath(base_image)
subprocess.check_call(["qemu-img", "create", "-q", "-f", "qcow2",
"-o", f"backing_file={base_image},backing_fmt=qcow2", qcow2_image])
if os.path.lexists(install_image):
os.unlink(install_image)
os.symlink(os.path.basename(qcow2_image), install_image)
if resize:
subprocess.check_call(["qemu-img", "resize", install_image, resize])
return install_image
class ActionBase(argparse.Action):
"""Keep an ordered list of actions"""
@staticmethod
def execute(machine_instance: testvm.Machine, argument: str) -> None:
raise NotImplementedError
def __call__(
self,
parser: argparse.ArgumentParser,
namespace: argparse.Namespace,
value: str | Sequence[str] | None,
option_string: object = None
) -> None:
getattr(namespace, self.dest).append((self.execute, value))
class InstallAction(ActionBase):
"""Install local rpm or distro package"""
@staticmethod
def execute(machine_instance: testvm.Machine, package: str) -> None:
# If we have a '/' in the package name, or if a file with that name
# exists in the current directory, then assume that this is a package
# we're uploading from the host.
if '/' in package or os.path.isfile(package):
dest = "/var/tmp/" + os.path.basename(package)
machine_instance.upload([os.path.abspath(package)], dest)
package = dest
# requesting install of Python wheel?
if package.endswith('.whl'):
machine_instance.execute(f"python3 -m pip install --no-index --prefix=/usr/local {package}", timeout=120)
return
# this will fail if neither is available -- exception is clear enough, this is a developer tool
out = machine_instance.execute("which dnf || which yum || which apt-get")
if 'dnf' in out:
install_command = "dnf install -y"
elif 'yum' in out:
install_command = "yum --setopt=skip_missing_names_on_install=False -y install"
else:
install_command = "apt-get install -y"
machine_instance.execute(f"{install_command} {package}", timeout=1800)
class BuildAction(ActionBase):
"""Build and install distribution package(s) from dist tarball or source RPM"""
@staticmethod
def execute(machine_instance: testvm.Machine, source: str) -> None:
# upload the tarball or srpm
sourcename = os.path.basename(source)
vm_source = os.path.join("/var/tmp", sourcename)
machine_instance.upload([source], vm_source, relative_dir=".")
# this will fail if neither is available -- exception is clear enough, this is a developer tool
out = machine_instance.execute("(which pbuilder || which mock || which pacman) 2>/dev/null")
if 'pbuilder' in out:
BuildAction.build_deb(machine_instance, vm_source)
elif 'mock' in out:
BuildAction.build_rpm(machine_instance, vm_source)
elif 'pacman' in out:
BuildAction.build_arch(machine_instance, vm_source)
else:
raise NotImplementedError(f"unknown build platform: {out}")
@staticmethod
def build_deb(machine: testvm.Machine, vm_source: str) -> None:
build_opts = 'nocheck' if opt_quick else ''
# build source packge
machine.execute(f"""
set -eu
rm -rf /var/tmp/build
mkdir -p /var/tmp/build
tar -C /var/tmp/build -xf '{vm_source}'
cd "$(ls -d /var/tmp/build/*)"
# find and copy debian packaging directory
cp -r "$(dirname $(find -path '*/debian/control'))" .
# create orig.tar link for building dsc
source=$(awk '/^Source: / {{ print $2 }}' debian/control)
version=$(dpkg-parsechangelog -SVersion)
# cut off Debian revision
ln -s '{vm_source}' ../${{source}}_${{version%-*}}.orig.tar.xz
dpkg-buildpackage -S -us -uc -nc""")
# build binary packages
machine.execute(f"cd /var/tmp/build; DEB_BUILD_OPTIONS='{build_opts}' pbuilder build --buildresult . "
f"{opt_build_options} *.dsc", timeout=1800, stdout=stdout_disposition)
# install packages
machine.execute("dpkg -i /var/tmp/build/*.deb")
@staticmethod
def build_rpm(machine: testvm.Machine, vm_source: str) -> None:
mock_opts = ''
if opt_verbose:
mock_opts += ' --verbose'
if opt_quick:
mock_opts += ' --nocheck'
if opt_build_options:
mock_opts += ' ' + opt_build_options
# HACK: SELinux b0rkage https://issues.redhat.com/browse/RHEL-49567
if machine.image in ['rhel-10-0', 'centos-10']:
machine.execute("setenforce 0")
# build source package, unless this is running against an srpm already
if vm_source.endswith(".src.rpm"):
srpm = vm_source
else:
machine.execute(f'''su builder -c 'rpmbuild --define "_topdir /var/tmp/build" -ts "{vm_source}"' ''')
srpm = "/var/tmp/build/SRPMS/*.src.rpm"
# HACK: mock in openSUSE must be called through sudo (but still originally as builder)
if "opensuse" in machine.execute("cat /etc/os-release"):
mock_sudo = "sudo"
else:
mock_sudo = ""
# build binary RPMs from srpm; disable all repositorys as mock insists on
# calling `dnf builddep`, which insists on a cache; our test VMs don't have a cache,
# as the mock is offline and pre-installed
machine.execute(f"su builder -c '{mock_sudo} mock --no-clean --no-cleanup-after --disablerepo=* "
f"--offline --resultdir /var/tmp/build {mock_opts} --rebuild {srpm}'",
timeout=1800, stdout=stdout_disposition)
# install RPMs
machine.execute('packages=$(find /var/tmp/build -name "*.rpm" -not -name "*.src.rpm"); '
f'rpm -U --force --verbose {"--nodigest --nosignature" if opt_quick else ""} $packages')
@staticmethod
def build_arch(machine: testvm.Machine, vm_source: str) -> None:
# unpack source tree's arch packaging directory (PKGBUILD refers to some files)
# and set PKGBUILD variables
machine.write("/var/tmp/mkbuild.sh", f"""#!/bin/sh
set -eu
rm -rf /var/tmp/build
mkdir -p /var/tmp/build
tar -C /var/tmp/build -xf '{vm_source}'
cd /var/tmp/build/
unpackdir="$(ls)"
archdir=$(dirname $(find -path '*/arch/PKGBUILD'))
cp "$archdir"/* .
# tarball must be in same directory as PKGBUILD
cp '{vm_source}' /var/tmp/build/
""", perm="755")
machine.execute(f"su builder {opt_build_options} /var/tmp/mkbuild.sh")
# build binaries
machine.execute("cd /var/tmp/build; makechrootpkg -r /var/lib/archbuild/cockpit -U builder",
timeout=1800, stdout=stdout_disposition)
# install packages
machine.execute("pacman -U --noconfirm /var/tmp/build/*.pkg.tar.zst")
class RunCommandAction(ActionBase):
@staticmethod
def execute(machine_instance: testvm.Machine, command: str) -> None:
try:
machine_instance.execute(command, timeout=1800)
except subprocess.CalledProcessError as e:
sys.stderr.write("%s\n" % e)
sys.exit(e.returncode)
class ScriptAction(ActionBase):
@staticmethod
def execute(machine_instance: testvm.Machine, script: str) -> None:
uploadpath = "/var/tmp/" + os.path.basename(script)
machine_instance.upload([os.path.abspath(script)], uploadpath)
machine_instance.execute("chmod a+x %s" % uploadpath)
try:
machine_instance.execute(uploadpath, timeout=1800)
except subprocess.CalledProcessError as e:
sys.stderr.write("%s\n" % e)
sys.exit(e.returncode)
class UploadAction(ActionBase):
@staticmethod
def execute(machine_instance: testvm.Machine, srcdest: str) -> None:
src, dest = srcdest.split(":")
abssrc = os.path.abspath(src)
# preserve trailing / for rsync compatibility
if src.endswith('/'):
abssrc += '/'
machine_instance.upload([abssrc], dest)
def main() -> None:
parser = argparse.ArgumentParser(
description=('Run command inside or install packages into a Cockpit virtual machine. '
'All actions can be specified multiple times and run in the given order.'))
# actions (at least one must be given, executed in order)
parser.add_argument('-i', '--install', action=InstallAction, metavar="PACKAGE", dest='actions', default=[],
help='Install package')
parser.add_argument('-b', '--build', action=BuildAction, metavar="TAR-OR-SRPM", dest='actions', default=[],
help='Build and install distribution package(s) from dist tarball or source RPM')
parser.add_argument('-r', '--run-command', action=RunCommandAction, dest='actions',
help='Run command inside virtual machine')
parser.add_argument('-s', '--script', action=ScriptAction, dest='actions',
help='Run selected script inside virtual machine')
parser.add_argument('-u', '--upload', action=UploadAction, metavar="SRC:DEST", dest='actions',
help='Upload file/dir to destination file/dir separated by ":" example: -u file.txt:/var/lib')
# options
parser.add_argument('--base-image',
help='Base image name, if "image" does not match a standard Cockpit VM image name')
parser.add_argument('--fresh', action='store_true',
help="Start fresh from the base image; by default, image-customize calls are additive")
parser.add_argument('--build-options', default="",
help="Additional options for mock/pbuilder/arch builder")
parser.add_argument('--resize', help="Resize the image. Size in bytes with using K, M, or G suffix.")
parser.add_argument('-n', '--no-network', action='store_true', help='Do not connect the machine to the Internet')
parser.add_argument('--cpus', type=int, default=None,
help="Number of CPUs for the virtual machine")
parser.add_argument('--memory-mb', type=int, default=2048,
help="RAM size for the virtual machine")
parser.add_argument('-v', '--verbose', action='store_true',
help='Display verbose progress details')
parser.add_argument('-q', '--quick', action='store_true',
help='Disable tests during package build with --build')
parser.add_argument('image', help='The image to use (destination name when using --base-image)')
parser.add_argument('--sit', action='store_true', help='Sit and wait if any VM action fails')
args = parser.parse_args()
if not args.actions and not args.resize:
parser.error("Must specify at least one operation")
if not args.base_image:
args.base_image = os.path.basename(args.image)
args.base_image = testvm.get_test_image(args.base_image)
global opt_quick, opt_verbose, opt_build_options, stdout_disposition
opt_quick = args.quick
opt_verbose = args.verbose
opt_build_options = args.build_options
if not args.verbose:
stdout_disposition = subprocess.DEVNULL
if '/' not in args.base_image:
subprocess.check_call([os.path.join(BOTS_DIR, "image-download"), args.base_image])
network = testvm.VirtNetwork(0, image=args.base_image)
machine = testvm.VirtMachine(maintain=True,
verbose=args.verbose,
networking=network.host(restrict=args.no_network),
image=prepare_install_image(args.base_image, args.image, args.resize, args.fresh),
cpus=args.cpus,
memory_mb=args.memory_mb)
machine.start()
machine.wait_boot()
try:
for (handler, arg) in args.actions:
handler(machine, arg)
except Exception as e:
if args.sit:
print(e, file=sys.stderr)
print(machine.diagnose(), file=sys.stderr)
print("Press RET to continue...")
sys.stdin.readline()
raise e
finally:
machine.stop()
if __name__ == '__main__':
main()