mirror of
https://git.launchpad.net/livecd-rootfs
synced 2026-02-12 13:03:43 +00:00
Add isobuild tool to build installer ISOs
This adds a new tool, isobuild, which replaces the ISO-building functionality previously provided by live-build and cdimage. It is invoked from auto/build when MAKE_ISO=yes. The tool supports: - Layered desktop images (Ubuntu Desktop, flavors) - Non-layered images (Kubuntu, Ubuntu Unity) - Images with package pools (most installers) - Images without pools (Ubuntu Core Installer) The isobuild command has several subcommands: - init: Initialize the ISO build directory structure - setup-apt: Configure APT for package pool generation - generate-pool: Create the package pool from a seed - generate-sources: Generate cdrom.sources for the installed system - add-live-filesystem: Add squashfs and kernel/initrd to the ISO - make-bootable: Add GRUB and other boot infrastructure - make-iso: Generate the final ISO image auto/config is updated to: - Set MAKE_ISO=yes for relevant image types - Set POOL_SEED_NAME for images that need a package pool - Invoke gen-iso-ids to compute ISO metadata auto/build is updated to: - Remove old live-build ISO handling code - Invoke isobuild at appropriate points in the build lb_binary_layered is updated to create squashfs files with cdrom.sources included for use in the ISO.
This commit is contained in:
parent
3112c5f175
commit
b3fdc4e615
@ -208,6 +208,22 @@ EOF
|
|||||||
undivert_update_initramfs
|
undivert_update_initramfs
|
||||||
undivert_grub chroot
|
undivert_grub chroot
|
||||||
fi
|
fi
|
||||||
|
if [ "${MAKE_ISO}" = yes ]; then
|
||||||
|
isobuild init --disk-info "$(cat config/iso-ids/disk-info)" --series "${LB_DISTRIBUTION}" --arch "${ARCH}"
|
||||||
|
# Determine which chroot directory has the apt configuration to use.
|
||||||
|
# Layered builds (PASSES set) create overlay directories named
|
||||||
|
# "overlay.base", "overlay.live", etc. - we use the first one (base).
|
||||||
|
# Single-pass builds use the "chroot" directory directly.
|
||||||
|
if [ "${PASSES}" ]; then
|
||||||
|
CHROOT="overlay.$(set -- $PASSES; echo $1)"
|
||||||
|
else
|
||||||
|
CHROOT=chroot
|
||||||
|
fi
|
||||||
|
isobuild setup-apt --chroot $CHROOT
|
||||||
|
if [ -n "${POOL_SEED_NAME}" ]; then
|
||||||
|
isobuild generate-pool --package-list-file "config/germinate-output/${POOL_SEED_NAME}"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
if [ -d chroot/etc/apt/preferences.d.save ]; then
|
if [ -d chroot/etc/apt/preferences.d.save ]; then
|
||||||
# https://mastodon.social/@scream@botsin.space
|
# https://mastodon.social/@scream@botsin.space
|
||||||
@ -427,13 +443,6 @@ if [ -e config/manifest-minimal-remove ]; then
|
|||||||
cp config/manifest-minimal-remove "$PREFIX.manifest-minimal-remove"
|
cp config/manifest-minimal-remove "$PREFIX.manifest-minimal-remove"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
for ISO in binary.iso binary.hybrid.iso; do
|
|
||||||
[ -e "$ISO" ] || continue
|
|
||||||
ln "$ISO" "$PREFIX.iso"
|
|
||||||
chmod 644 "$PREFIX.iso"
|
|
||||||
break
|
|
||||||
done
|
|
||||||
|
|
||||||
if [ -e "binary/$INITFS/filesystem.dir" ]; then
|
if [ -e "binary/$INITFS/filesystem.dir" ]; then
|
||||||
(cd "binary/$INITFS/filesystem.dir/" && tar -c --sort=name --xattrs *) | \
|
(cd "binary/$INITFS/filesystem.dir/" && tar -c --sort=name --xattrs *) | \
|
||||||
gzip -9 --rsyncable > "$PREFIX.rootfs.tar.gz"
|
gzip -9 --rsyncable > "$PREFIX.rootfs.tar.gz"
|
||||||
@ -558,3 +567,23 @@ case $PROJECT in
|
|||||||
ubuntu-cpc)
|
ubuntu-cpc)
|
||||||
config/hooks.d/remove-implicit-artifacts
|
config/hooks.d/remove-implicit-artifacts
|
||||||
esac
|
esac
|
||||||
|
|
||||||
|
if [ "${MAKE_ISO}" = "yes" ]; then
|
||||||
|
# Link build artifacts with "for-iso." prefix for isobuild to consume.
|
||||||
|
# Layered builds create squashfs via lb_binary_layered (which already
|
||||||
|
# creates for-iso.*.squashfs files). Single-pass builds only have
|
||||||
|
# ${PREFIX}.squashfs, so we link it here.
|
||||||
|
if [ -z "$PASSES" ]; then
|
||||||
|
ln ${PREFIX}.squashfs for-iso.filesystem.squashfs
|
||||||
|
fi
|
||||||
|
# Link kernel and initrd files. The ${thing#${PREFIX}} expansion strips
|
||||||
|
# the PREFIX, so "livecd.ubuntu-server.kernel-generic" becomes
|
||||||
|
# "for-iso.kernel-generic".
|
||||||
|
for thing in ${PREFIX}.kernel-* ${PREFIX}.initrd-*; do
|
||||||
|
ln $thing for-iso${thing#${PREFIX}}
|
||||||
|
done
|
||||||
|
isobuild add-live-filesystem --artifact-prefix for-iso.
|
||||||
|
isobuild make-bootable --project "${PROJECT}" --capproject "$(cat config/iso-ids/capproject)" \
|
||||||
|
${SUBARCH:+--subarch "${SUBARCH}"}
|
||||||
|
isobuild make-iso --volid "$(cat config/iso-ids/vol-id)" --dest ${PREFIX}.iso
|
||||||
|
fi
|
||||||
|
|||||||
@ -663,10 +663,23 @@ if ! [ -e config/germinate-output/structure ]; then
|
|||||||
-s $FLAVOUR.$SUITE $GERMINATE_ARG -a ${ARCH_VARIANT:-$ARCH})
|
-s $FLAVOUR.$SUITE $GERMINATE_ARG -a ${ARCH_VARIANT:-$ARCH})
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# ISO build configuration. These defaults are overridden per-project below.
|
||||||
|
#
|
||||||
|
# MAKE_ISO: Set to "yes" to generate an installer ISO at the end of the build.
|
||||||
|
# This triggers isobuild to run in auto/build.
|
||||||
|
MAKE_ISO=no
|
||||||
|
# POOL_SEED_NAME: The germinate output file defining packages for the ISO's
|
||||||
|
# package pool (repository). Different flavors use different seeds:
|
||||||
|
# - "ship-live" for most desktop images
|
||||||
|
# - "server-ship-live" for Ubuntu Server (includes server-specific packages)
|
||||||
|
# - "" (empty) for images without a pool, like Ubuntu Core Installer
|
||||||
|
POOL_SEED_NAME=ship-live
|
||||||
|
|
||||||
# Common functionality for layered desktop images
|
# Common functionality for layered desktop images
|
||||||
common_layered_desktop_image() {
|
common_layered_desktop_image() {
|
||||||
touch config/universe-enabled
|
touch config/universe-enabled
|
||||||
PASSES_TO_LAYERS="true"
|
PASSES_TO_LAYERS="true"
|
||||||
|
MAKE_ISO=yes
|
||||||
|
|
||||||
if [ -n "$HAS_MINIMAL" ]; then
|
if [ -n "$HAS_MINIMAL" ]; then
|
||||||
if [ -z "$MINIMAL_TASKS" ]; then
|
if [ -z "$MINIMAL_TASKS" ]; then
|
||||||
@ -897,6 +910,7 @@ case $PROJECT in
|
|||||||
add_task install minimal standard
|
add_task install minimal standard
|
||||||
add_task install kubuntu-desktop
|
add_task install kubuntu-desktop
|
||||||
LIVE_TASK='kubuntu-live'
|
LIVE_TASK='kubuntu-live'
|
||||||
|
MAKE_ISO=yes
|
||||||
add_chroot_hook remove-gnome-icon-cache
|
add_chroot_hook remove-gnome-icon-cache
|
||||||
;;
|
;;
|
||||||
|
|
||||||
@ -923,6 +937,7 @@ case $PROJECT in
|
|||||||
ubuntu-unity)
|
ubuntu-unity)
|
||||||
add_task install minimal standard ${PROJECT}-desktop
|
add_task install minimal standard ${PROJECT}-desktop
|
||||||
LIVE_TASK=${PROJECT}-live
|
LIVE_TASK=${PROJECT}-live
|
||||||
|
MAKE_ISO=yes
|
||||||
;;
|
;;
|
||||||
|
|
||||||
lubuntu)
|
lubuntu)
|
||||||
@ -997,6 +1012,8 @@ case $PROJECT in
|
|||||||
live)
|
live)
|
||||||
OPTS="${OPTS:+$OPTS }--bootstrap-flavour=minimal"
|
OPTS="${OPTS:+$OPTS }--bootstrap-flavour=minimal"
|
||||||
PASSES_TO_LAYERS=true
|
PASSES_TO_LAYERS=true
|
||||||
|
MAKE_ISO=yes
|
||||||
|
POOL_SEED_NAME=server-ship-live
|
||||||
add_task ubuntu-server-minimal server-minimal
|
add_task ubuntu-server-minimal server-minimal
|
||||||
add_package ubuntu-server-minimal lxd-installer
|
add_package ubuntu-server-minimal lxd-installer
|
||||||
add_task ubuntu-server-minimal.ubuntu-server minimal standard server
|
add_task ubuntu-server-minimal.ubuntu-server minimal standard server
|
||||||
@ -1129,6 +1146,8 @@ case $PROJECT in
|
|||||||
fi
|
fi
|
||||||
OPTS="${OPTS:+$OPTS }--bootstrap-flavour=minimal"
|
OPTS="${OPTS:+$OPTS }--bootstrap-flavour=minimal"
|
||||||
PASSES_TO_LAYERS=true
|
PASSES_TO_LAYERS=true
|
||||||
|
MAKE_ISO=yes
|
||||||
|
POOL_SEED_NAME=
|
||||||
add_task base server-minimal server
|
add_task base server-minimal server
|
||||||
add_task base.live server-live
|
add_task base.live server-live
|
||||||
add_package base.live linux-image-generic
|
add_package base.live linux-image-generic
|
||||||
@ -1384,6 +1403,8 @@ echo "IMAGEFORMAT=\"$IMAGEFORMAT\"" >> config/chroot
|
|||||||
if [ -n "$PASSES" ]; then
|
if [ -n "$PASSES" ]; then
|
||||||
echo "PASSES=\"$PASSES\"" >> config/common
|
echo "PASSES=\"$PASSES\"" >> config/common
|
||||||
fi
|
fi
|
||||||
|
echo "MAKE_ISO=\"$MAKE_ISO\"" >> config/common
|
||||||
|
echo "POOL_SEED_NAME=\"$POOL_SEED_NAME\"" >> config/common
|
||||||
if [ -n "$NO_SQUASHFS_PASSES" ]; then
|
if [ -n "$NO_SQUASHFS_PASSES" ]; then
|
||||||
echo "NO_SQUASHFS_PASSES=\"$NO_SQUASHFS_PASSES\"" >> config/common
|
echo "NO_SQUASHFS_PASSES=\"$NO_SQUASHFS_PASSES\"" >> config/common
|
||||||
fi
|
fi
|
||||||
@ -1677,3 +1698,11 @@ apt-get -y download $PREINSTALL_POOL
|
|||||||
EOF
|
EOF
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
if [ "${MAKE_ISO}" = "yes" ]; then
|
||||||
|
# XXX should pass --official here.
|
||||||
|
/usr/share/livecd-rootfs/live-build/gen-iso-ids \
|
||||||
|
--project $PROJECT ${SUBPROJECT:+--subproject $SUBPROJECT} \
|
||||||
|
--arch $ARCH ${SUBARCH:+--subarch $SUBARCH} ${NOW+--serial $NOW} \
|
||||||
|
--output-dir config/iso-ids/
|
||||||
|
fi
|
||||||
|
|||||||
@ -1444,3 +1444,10 @@ gpt_root_partition_uuid() {
|
|||||||
|
|
||||||
echo "${ROOTFS_PARTITION_TYPE}"
|
echo "${ROOTFS_PARTITION_TYPE}"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Wrapper for the isobuild tool. Sets PYTHONPATH so the isobuilder module
|
||||||
|
# is importable, and uses config/iso-dir as the standard working directory
|
||||||
|
# for ISO metadata and intermediate files.
|
||||||
|
isobuild () {
|
||||||
|
PYTHONPATH=/usr/share/livecd-rootfs/live-build/ /usr/share/livecd-rootfs/live-build/isobuild --workdir config/iso-dir "$@"
|
||||||
|
}
|
||||||
|
|||||||
221
live-build/isobuild
Executable file
221
live-build/isobuild
Executable file
@ -0,0 +1,221 @@
|
|||||||
|
#!/usr/bin/python3
|
||||||
|
|
||||||
|
# Building an ISO requires knowing:
|
||||||
|
#
|
||||||
|
# * The architecture and series we are building for
|
||||||
|
# * The address of the mirror to pull packages from the pool from and the
|
||||||
|
# components of that mirror to use
|
||||||
|
# * The list of packages to include in the pool
|
||||||
|
# * Where the squashfs files that contain the rootfs and other metadata layers
|
||||||
|
# are
|
||||||
|
# * Where to put the final ISO
|
||||||
|
# * All the bits of information that end up in .disk/info on the ISO and in the
|
||||||
|
# "volume ID" for the ISO
|
||||||
|
#
|
||||||
|
# It's not completely trivial to come up with a nice feeling interface between
|
||||||
|
# livecd-rootfs and this tool. There are about 13 parameters that are needed to
|
||||||
|
# build the ISO and having a tool take 13 arguments seems a bit overwhelming. In
|
||||||
|
# addition some steps need to run before the layers are made into squashfs files
|
||||||
|
# and some after. It felt nicer to have a tool with a few subcommands (7, in the
|
||||||
|
# end) and taking arguments relevant to each step:
|
||||||
|
#
|
||||||
|
# $ isobuild --work-dir "" init --disk-id "" --series "" --arch ""
|
||||||
|
#
|
||||||
|
# Set up the work-dir for later steps. Create the skeleton file layout of the
|
||||||
|
# ISO, populate .disk/info etc, create the gpg key referred to above. Store
|
||||||
|
# series and arch somewhere that later steps can refer to.
|
||||||
|
#
|
||||||
|
# $ isobuild --work-dir "" setup-apt --chroot ""
|
||||||
|
#
|
||||||
|
# Set up aptfor use by later steps, using the configuration from the passed
|
||||||
|
# chroot.
|
||||||
|
#
|
||||||
|
# $ isobuild --work-dir "" generate-pool --package-list-file ""
|
||||||
|
#
|
||||||
|
# Create the pool from the passed germinate output file.
|
||||||
|
#
|
||||||
|
# $ isobuild --work-dir "" generate-sources --mountpoint ""
|
||||||
|
#
|
||||||
|
# Generate an apt deb822 source for the pool, assuming it is mounted at the
|
||||||
|
# passed mountpoint, and output it on stdout.
|
||||||
|
#
|
||||||
|
# $ isobuild --work-dir "" add-live-filesystem --artifact-prefix ""
|
||||||
|
#
|
||||||
|
# Copy the relevant artifacts to the casper directory (and extract the uuids
|
||||||
|
# from the initrds)
|
||||||
|
#
|
||||||
|
# $ isobuild --work-dir "" make-bootable --project "" --capitalized-project ""
|
||||||
|
# --subarch ""
|
||||||
|
#
|
||||||
|
# Set up the bootloader etc so that the ISO can boot (for this clones debian-cd
|
||||||
|
# and run the tools/boot/$series-$arch script but those should be folded into
|
||||||
|
# isobuild fairly promptly IMO).
|
||||||
|
#
|
||||||
|
# $ isobuild --work-dir "" make-iso --vol-id "" --dest ""
|
||||||
|
#
|
||||||
|
# Generate the checksum file and run xorriso to build the final ISO.
|
||||||
|
|
||||||
|
|
||||||
|
import pathlib
|
||||||
|
import shlex
|
||||||
|
|
||||||
|
import click
|
||||||
|
|
||||||
|
from isobuilder.builder import ISOBuilder
|
||||||
|
|
||||||
|
|
||||||
|
@click.group()
|
||||||
|
@click.option(
|
||||||
|
"--workdir",
|
||||||
|
type=click.Path(file_okay=False, resolve_path=True, path_type=pathlib.Path),
|
||||||
|
required=True,
|
||||||
|
help="working directory",
|
||||||
|
)
|
||||||
|
@click.pass_context
|
||||||
|
def main(ctxt, workdir):
|
||||||
|
ctxt.obj = ISOBuilder(workdir)
|
||||||
|
cwd = pathlib.Path().cwd()
|
||||||
|
if workdir.is_relative_to(cwd):
|
||||||
|
workdir = workdir.relative_to(cwd)
|
||||||
|
ctxt.obj.logger.log(f"isobuild starting, workdir: {workdir}")
|
||||||
|
|
||||||
|
|
||||||
|
def subcommand(f):
|
||||||
|
"""Decorator that converts a function into a Click subcommand with logging.
|
||||||
|
|
||||||
|
This decorator:
|
||||||
|
1. Converts function name from snake_case to kebab-case for the CLI
|
||||||
|
2. Wraps the function to log the subcommand name and all parameters
|
||||||
|
3. Registers it as a Click command under the main command group
|
||||||
|
4. Extracts the ISOBuilder instance from the context and passes it as first arg
|
||||||
|
"""
|
||||||
|
name = f.__name__.replace("_", "-")
|
||||||
|
|
||||||
|
def wrapped(ctxt, **kw):
|
||||||
|
# Build a log message showing the subcommand and all its parameters.
|
||||||
|
# We use ctxt.params (Click's resolved parameters) rather than **kw
|
||||||
|
# because ctxt.params includes path resolution and type conversion.
|
||||||
|
# Paths are converted to relative form to keep logs readable and avoid
|
||||||
|
# exposing full filesystem paths in build artifacts.
|
||||||
|
msg = f"subcommand {name}"
|
||||||
|
cwd = pathlib.Path().cwd()
|
||||||
|
for k, v in sorted(ctxt.params.items()):
|
||||||
|
if isinstance(v, pathlib.Path):
|
||||||
|
if v.is_relative_to(cwd):
|
||||||
|
v = v.relative_to(cwd)
|
||||||
|
v = shlex.quote(str(v))
|
||||||
|
msg += f" {k}={v}"
|
||||||
|
with ctxt.obj.logger.logged(msg):
|
||||||
|
f(ctxt.obj, **kw)
|
||||||
|
|
||||||
|
return main.command(name=name)(click.pass_context(wrapped))
|
||||||
|
|
||||||
|
|
||||||
|
@click.option(
|
||||||
|
"--disk-info",
|
||||||
|
type=str,
|
||||||
|
required=True,
|
||||||
|
help="contents of .disk/info",
|
||||||
|
)
|
||||||
|
@click.option(
|
||||||
|
"--series",
|
||||||
|
type=str,
|
||||||
|
required=True,
|
||||||
|
help="series being built",
|
||||||
|
)
|
||||||
|
@click.option(
|
||||||
|
"--arch",
|
||||||
|
type=str,
|
||||||
|
required=True,
|
||||||
|
help="architecture being built",
|
||||||
|
)
|
||||||
|
@subcommand
|
||||||
|
def init(builder, disk_info, series, arch):
|
||||||
|
builder.init(disk_info, series, arch)
|
||||||
|
|
||||||
|
|
||||||
|
@click.option(
|
||||||
|
"--chroot",
|
||||||
|
type=click.Path(
|
||||||
|
file_okay=False, resolve_path=True, path_type=pathlib.Path, exists=True
|
||||||
|
),
|
||||||
|
required=True,
|
||||||
|
)
|
||||||
|
@subcommand
|
||||||
|
def setup_apt(builder, chroot: pathlib.Path):
|
||||||
|
builder.setup_apt(chroot)
|
||||||
|
|
||||||
|
|
||||||
|
@click.pass_obj
|
||||||
|
@click.option(
|
||||||
|
"--package-list-file",
|
||||||
|
type=click.Path(
|
||||||
|
dir_okay=False, exists=True, resolve_path=True, path_type=pathlib.Path
|
||||||
|
),
|
||||||
|
required=True,
|
||||||
|
)
|
||||||
|
@subcommand
|
||||||
|
def generate_pool(builder, package_list_file: pathlib.Path):
|
||||||
|
builder.generate_pool(package_list_file)
|
||||||
|
|
||||||
|
|
||||||
|
@click.option(
|
||||||
|
"--mountpoint",
|
||||||
|
type=str,
|
||||||
|
required=True,
|
||||||
|
)
|
||||||
|
@subcommand
|
||||||
|
def generate_sources(builder, mountpoint: str):
|
||||||
|
builder.generate_sources(mountpoint)
|
||||||
|
|
||||||
|
|
||||||
|
@click.option(
|
||||||
|
"--artifact-prefix",
|
||||||
|
type=click.Path(dir_okay=False, resolve_path=True, path_type=pathlib.Path),
|
||||||
|
required=True,
|
||||||
|
)
|
||||||
|
@subcommand
|
||||||
|
def add_live_filesystem(builder, artifact_prefix: pathlib.Path):
|
||||||
|
builder.add_live_filesystem(artifact_prefix)
|
||||||
|
|
||||||
|
|
||||||
|
@click.option(
|
||||||
|
"--project",
|
||||||
|
type=str,
|
||||||
|
required=True,
|
||||||
|
)
|
||||||
|
@click.option("--capproject", type=str, required=True)
|
||||||
|
@click.option(
|
||||||
|
"--subarch",
|
||||||
|
type=str,
|
||||||
|
default="",
|
||||||
|
)
|
||||||
|
@subcommand
|
||||||
|
def make_bootable(builder, project: str, capproject: str | None, subarch: str):
|
||||||
|
# capproject is the "capitalized project name" used in GRUB menu entries,
|
||||||
|
# e.g. "Ubuntu" or "Kubuntu". It should come from gen-iso-ids (which uses
|
||||||
|
# project_to_capproject_map for proper formatting like "Ubuntu-MATE"), but
|
||||||
|
# we provide a simple .capitalize() fallback for cases where the caller
|
||||||
|
# doesn't have the pre-computed value.
|
||||||
|
if capproject is None:
|
||||||
|
capproject = project.capitalize()
|
||||||
|
builder.make_bootable(project, capproject, subarch)
|
||||||
|
|
||||||
|
|
||||||
|
@click.option(
|
||||||
|
"--dest",
|
||||||
|
type=click.Path(dir_okay=False, resolve_path=True, path_type=pathlib.Path),
|
||||||
|
required=True,
|
||||||
|
)
|
||||||
|
@click.option(
|
||||||
|
"--volid",
|
||||||
|
type=str,
|
||||||
|
default=None,
|
||||||
|
)
|
||||||
|
@subcommand
|
||||||
|
def make_iso(builder, dest: pathlib.Path, volid: str | None):
|
||||||
|
builder.make_iso(dest, volid)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
1
live-build/isobuilder/__init__.py
Normal file
1
live-build/isobuilder/__init__.py
Normal file
@ -0,0 +1 @@
|
|||||||
|
#
|
||||||
110
live-build/isobuilder/apt_state.py
Normal file
110
live-build/isobuilder/apt_state.py
Normal file
@ -0,0 +1,110 @@
|
|||||||
|
import dataclasses
|
||||||
|
import os
|
||||||
|
import pathlib
|
||||||
|
import shutil
|
||||||
|
import subprocess
|
||||||
|
from typing import Iterator
|
||||||
|
|
||||||
|
|
||||||
|
@dataclasses.dataclass
|
||||||
|
class PackageInfo:
|
||||||
|
package: str
|
||||||
|
filename: str
|
||||||
|
architecture: str
|
||||||
|
version: str
|
||||||
|
|
||||||
|
@property
|
||||||
|
def spec(self) -> str:
|
||||||
|
return f"{self.package}:{self.architecture}={self.version}"
|
||||||
|
|
||||||
|
|
||||||
|
def check_proc(proc, ok_codes=(0,)) -> None:
|
||||||
|
proc.wait()
|
||||||
|
if proc.returncode not in ok_codes:
|
||||||
|
raise Exception(f"{proc} failed")
|
||||||
|
|
||||||
|
|
||||||
|
class AptStateManager:
|
||||||
|
"""Maintain and use an apt state directory to access package info and debs."""
|
||||||
|
|
||||||
|
def __init__(self, logger, series: str, apt_dir: pathlib.Path):
|
||||||
|
self.logger = logger
|
||||||
|
self.series = series
|
||||||
|
self.apt_root = apt_dir.joinpath("root")
|
||||||
|
self.apt_conf_path = apt_dir.joinpath("apt.conf")
|
||||||
|
|
||||||
|
def _apt_env(self) -> dict[str, str]:
|
||||||
|
return dict(os.environ, APT_CONFIG=str(self.apt_conf_path))
|
||||||
|
|
||||||
|
def setup(self, chroot: pathlib.Path):
|
||||||
|
"""Set up the manager by copying the apt configuration from `chroot`."""
|
||||||
|
for path in "etc/apt", "var/lib/apt":
|
||||||
|
tgt = self.apt_root.joinpath(path)
|
||||||
|
tgt.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
shutil.copytree(chroot.joinpath(path), tgt)
|
||||||
|
self.apt_conf_path.write_text(f'Dir "{self.apt_root}/"; \n')
|
||||||
|
with self.logger.logged("updating apt indices"):
|
||||||
|
self.logger.run(["apt-get", "update"], env=self._apt_env())
|
||||||
|
|
||||||
|
def show(self, pkgs: list[str]) -> Iterator[PackageInfo]:
|
||||||
|
"""Return information about the binary packages named by `pkgs`.
|
||||||
|
|
||||||
|
Parses apt-cache output, which uses RFC822-like format: field names
|
||||||
|
followed by ": " and values, with multi-line values indented with
|
||||||
|
leading whitespace. We skip continuation lines (starting with space)
|
||||||
|
since PackageInfo only needs single-line fields.
|
||||||
|
|
||||||
|
The `fields` set (derived from PackageInfo's dataclass fields) acts as
|
||||||
|
a filter - we only extract fields we care about, ignoring others like
|
||||||
|
Description. This provides some forward-compatibility if apt-cache
|
||||||
|
output format changes.
|
||||||
|
"""
|
||||||
|
proc = subprocess.Popen(
|
||||||
|
["apt-cache", "-o", "APT::Cache::AllVersions=0", "show"] + pkgs,
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
encoding="utf-8",
|
||||||
|
env=self._apt_env(),
|
||||||
|
)
|
||||||
|
assert proc.stdout is not None
|
||||||
|
fields = {f.name for f in dataclasses.fields(PackageInfo)}
|
||||||
|
params: dict[str, str] = {}
|
||||||
|
for line in proc.stdout:
|
||||||
|
if line == "\n":
|
||||||
|
yield PackageInfo(**params)
|
||||||
|
params = {}
|
||||||
|
continue
|
||||||
|
if line.startswith(" "):
|
||||||
|
continue
|
||||||
|
field, value = line.split(": ", 1)
|
||||||
|
field = field.lower()
|
||||||
|
if field in fields:
|
||||||
|
params[field] = value.strip()
|
||||||
|
check_proc(proc)
|
||||||
|
if params:
|
||||||
|
yield PackageInfo(**params)
|
||||||
|
|
||||||
|
def download(self, rootdir: pathlib.Path, pkg_info: PackageInfo):
|
||||||
|
"""Download the package specified by `pkg_info` under `rootdir`.
|
||||||
|
|
||||||
|
The package is saved to the same path under `rootdir` as it is
|
||||||
|
at in the archive it comes from.
|
||||||
|
"""
|
||||||
|
target_dir = rootdir.joinpath(pkg_info.filename).parent
|
||||||
|
target_dir.mkdir(parents=True, exist_ok=True)
|
||||||
|
self.logger.run(
|
||||||
|
["apt-get", "download", pkg_info.spec],
|
||||||
|
cwd=target_dir,
|
||||||
|
check=True,
|
||||||
|
env=self._apt_env(),
|
||||||
|
)
|
||||||
|
|
||||||
|
def in_release_path(self) -> pathlib.Path:
|
||||||
|
"""Return the path to the InRelease file.
|
||||||
|
|
||||||
|
This assumes exactly one InRelease file matches the pattern.
|
||||||
|
Will raise ValueError if there are 0 or multiple matches.
|
||||||
|
"""
|
||||||
|
[path] = self.apt_root.joinpath("var/lib/apt/lists").glob(
|
||||||
|
f"*_dists_{self.series}_InRelease"
|
||||||
|
)
|
||||||
|
return path
|
||||||
365
live-build/isobuilder/builder.py
Normal file
365
live-build/isobuilder/builder.py
Normal file
@ -0,0 +1,365 @@
|
|||||||
|
import contextlib
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import pathlib
|
||||||
|
import shlex
|
||||||
|
import shutil
|
||||||
|
import subprocess
|
||||||
|
import sys
|
||||||
|
|
||||||
|
from isobuilder.apt_state import AptStateManager
|
||||||
|
from isobuilder.gpg_key import EphemeralGPGKey
|
||||||
|
from isobuilder.pool_builder import PoolBuilder
|
||||||
|
|
||||||
|
# Constants
|
||||||
|
PACKAGE_BATCH_SIZE = 200
|
||||||
|
MAX_CMD_DISPLAY_LENGTH = 80
|
||||||
|
|
||||||
|
|
||||||
|
def package_list_packages(package_list_file: pathlib.Path) -> list[str]:
|
||||||
|
# Parse germinate output to extract package names. Germinate is Ubuntu's
|
||||||
|
# package dependency resolver that outputs dependency trees for seeds (like
|
||||||
|
# "ship-live" or "server-ship-live").
|
||||||
|
#
|
||||||
|
# Germinate output format has 2 header lines at the start and 2 footer lines
|
||||||
|
# at the end (showing statistics), so we skip them with [2:-2].
|
||||||
|
# Each data line starts with the package name followed by whitespace and
|
||||||
|
# dependency info. This format is stable but if germinate ever changes its
|
||||||
|
# header/footer count, this will break silently.
|
||||||
|
lines = package_list_file.read_text().splitlines()[2:-2]
|
||||||
|
return [line.split(None, 1)[0] for line in lines]
|
||||||
|
|
||||||
|
|
||||||
|
def make_sources_text(
|
||||||
|
series: str, gpg_key: EphemeralGPGKey, components: list[str], mountpoint: str
|
||||||
|
) -> str:
|
||||||
|
"""Generate a deb822-format apt source file for the ISO's package pool.
|
||||||
|
|
||||||
|
deb822 is the modern apt sources format (see sources.list(5) and deb822(5)).
|
||||||
|
It uses RFC822-style fields where multi-line values must be indented with a
|
||||||
|
leading space, and empty lines within a value are represented as " ."
|
||||||
|
(space-dot). This format is required for inline GPG keys in the Signed-By
|
||||||
|
field.
|
||||||
|
"""
|
||||||
|
key = gpg_key.export_public()
|
||||||
|
quoted_key = []
|
||||||
|
for line in key.splitlines():
|
||||||
|
if not line:
|
||||||
|
quoted_key.append(" .")
|
||||||
|
else:
|
||||||
|
quoted_key.append(" " + line)
|
||||||
|
return f"""\
|
||||||
|
Types: deb
|
||||||
|
URIs: file://{mountpoint}
|
||||||
|
Suites: {series}
|
||||||
|
Components: {" ".join(components)}
|
||||||
|
Check-Date: no
|
||||||
|
Signed-By:
|
||||||
|
""" + "\n".join(
|
||||||
|
quoted_key
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class Logger:
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self._indent = ""
|
||||||
|
|
||||||
|
def log(self, msg):
|
||||||
|
print(self._indent + msg, file=sys.stderr)
|
||||||
|
|
||||||
|
@contextlib.contextmanager
|
||||||
|
def logged(self, msg, done_msg=None):
|
||||||
|
self.log(msg)
|
||||||
|
self._indent += " "
|
||||||
|
try:
|
||||||
|
yield
|
||||||
|
finally:
|
||||||
|
self._indent = self._indent[:-2]
|
||||||
|
if done_msg is not None:
|
||||||
|
self.log(done_msg)
|
||||||
|
|
||||||
|
def msg_for_cmd(self, cmd, cwd=None) -> str:
|
||||||
|
if cwd is None:
|
||||||
|
_cwd = pathlib.Path().cwd()
|
||||||
|
else:
|
||||||
|
_cwd = cwd
|
||||||
|
fmted_cmd = []
|
||||||
|
for arg in cmd:
|
||||||
|
if isinstance(arg, pathlib.Path):
|
||||||
|
if arg.is_relative_to(_cwd):
|
||||||
|
arg = arg.relative_to(_cwd)
|
||||||
|
arg = str(arg)
|
||||||
|
fmted_cmd.append(shlex.quote(arg))
|
||||||
|
fmted_cmd_str = " ".join(fmted_cmd)
|
||||||
|
if len(fmted_cmd_str) > MAX_CMD_DISPLAY_LENGTH:
|
||||||
|
fmted_cmd_str = fmted_cmd_str[:MAX_CMD_DISPLAY_LENGTH] + "..."
|
||||||
|
msg = f"running `{fmted_cmd_str}`"
|
||||||
|
if cwd is not None:
|
||||||
|
msg += f" in {cwd}"
|
||||||
|
return msg
|
||||||
|
|
||||||
|
def run(self, cmd: list[str | pathlib.Path], *args, check=True, **kw):
|
||||||
|
with self.logged(self.msg_for_cmd(cmd, kw.get("cwd"))):
|
||||||
|
return subprocess.run(cmd, *args, check=check, **kw)
|
||||||
|
|
||||||
|
|
||||||
|
class ISOBuilder:
|
||||||
|
|
||||||
|
def __init__(self, workdir: pathlib.Path):
|
||||||
|
self.workdir = workdir
|
||||||
|
self.logger = Logger()
|
||||||
|
self.iso_root = workdir.joinpath("iso-root")
|
||||||
|
self._series = self._arch = self._gpg_key = self._apt_state = None
|
||||||
|
|
||||||
|
# UTILITY STUFF
|
||||||
|
|
||||||
|
def _read_config(self):
|
||||||
|
with self.workdir.joinpath("config.json").open() as fp:
|
||||||
|
data = json.load(fp)
|
||||||
|
self._series = data["series"]
|
||||||
|
self._arch = data["arch"]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def arch(self):
|
||||||
|
if self._arch is None:
|
||||||
|
self._read_config()
|
||||||
|
return self._arch
|
||||||
|
|
||||||
|
@property
|
||||||
|
def series(self):
|
||||||
|
if self._series is None:
|
||||||
|
self._read_config()
|
||||||
|
return self._series
|
||||||
|
|
||||||
|
@property
|
||||||
|
def gpg_key(self):
|
||||||
|
if self._gpg_key is None:
|
||||||
|
self._gpg_key = EphemeralGPGKey(
|
||||||
|
self.logger, self.workdir.joinpath("gpg-home")
|
||||||
|
)
|
||||||
|
return self._gpg_key
|
||||||
|
|
||||||
|
@property
|
||||||
|
def apt_state(self):
|
||||||
|
if self._apt_state is None:
|
||||||
|
self._apt_state = AptStateManager(
|
||||||
|
self.logger, self.series, self.workdir.joinpath("apt-state")
|
||||||
|
)
|
||||||
|
return self._apt_state
|
||||||
|
|
||||||
|
# COMMANDS
|
||||||
|
|
||||||
|
def init(self, disk_info: str, series: str, arch: str):
|
||||||
|
self.logger.log("creating directories")
|
||||||
|
self.workdir.mkdir(exist_ok=True)
|
||||||
|
self.iso_root.mkdir()
|
||||||
|
dot_disk = self.iso_root.joinpath(".disk")
|
||||||
|
dot_disk.mkdir()
|
||||||
|
|
||||||
|
self.logger.log("saving config")
|
||||||
|
with self.workdir.joinpath("config.json").open("w") as fp:
|
||||||
|
json.dump({"arch": arch, "series": series}, fp)
|
||||||
|
|
||||||
|
self.logger.log("populating .disk")
|
||||||
|
dot_disk.joinpath("base_installable").touch()
|
||||||
|
dot_disk.joinpath("cd_type").write_text("full_cd/single\n")
|
||||||
|
dot_disk.joinpath("info").write_text(disk_info)
|
||||||
|
self.iso_root.joinpath("casper").mkdir()
|
||||||
|
|
||||||
|
self.gpg_key.create()
|
||||||
|
|
||||||
|
def setup_apt(self, chroot: pathlib.Path):
|
||||||
|
self.apt_state.setup(chroot)
|
||||||
|
|
||||||
|
def generate_pool(self, package_list_file: pathlib.Path):
|
||||||
|
# do we need any of the symlinks we create here??
|
||||||
|
self.logger.log("creating pool skeleton")
|
||||||
|
self.iso_root.joinpath("ubuntu").symlink_to(".")
|
||||||
|
if self.arch not in ("amd64", "i386"):
|
||||||
|
self.iso_root.joinpath("ubuntu-ports").symlink_to(".")
|
||||||
|
self.iso_root.joinpath("dists", self.series).mkdir(parents=True)
|
||||||
|
|
||||||
|
builder = PoolBuilder(
|
||||||
|
self.logger,
|
||||||
|
series=self.series,
|
||||||
|
rootdir=self.iso_root,
|
||||||
|
apt_state=self.apt_state,
|
||||||
|
)
|
||||||
|
pkgs = package_list_packages(package_list_file)
|
||||||
|
# XXX include 32-bit deps of 32-bit packages if needed here
|
||||||
|
with self.logger.logged("adding packages"):
|
||||||
|
for i in range(0, len(pkgs), PACKAGE_BATCH_SIZE):
|
||||||
|
builder.add_packages(
|
||||||
|
self.apt_state.show(pkgs[i : i + PACKAGE_BATCH_SIZE])
|
||||||
|
)
|
||||||
|
builder.make_packages()
|
||||||
|
release_file = builder.make_release()
|
||||||
|
self.gpg_key.sign(release_file)
|
||||||
|
for name in "stable", "unstable":
|
||||||
|
self.iso_root.joinpath("dists", name).symlink_to(self.series)
|
||||||
|
|
||||||
|
def generate_sources(self, mountpoint: str):
|
||||||
|
components = [p.name for p in self.iso_root.joinpath("pool").iterdir()]
|
||||||
|
print(
|
||||||
|
make_sources_text(
|
||||||
|
self.series, self.gpg_key, mountpoint=mountpoint, components=components
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
def _extract_casper_uuids(self):
|
||||||
|
# Extract UUID files from initrd images for casper (the live boot system).
|
||||||
|
# Each initrd contains a conf/uuid.conf with a unique identifier that
|
||||||
|
# casper uses at boot time to locate the correct root filesystem. These
|
||||||
|
# UUIDs must be placed in .disk/casper-uuid-<flavor> on the ISO so casper
|
||||||
|
# can verify it's booting from the right media.
|
||||||
|
with self.logger.logged("extracting casper uuids"):
|
||||||
|
casper_dir = self.iso_root.joinpath("casper")
|
||||||
|
prefix = "filesystem.initrd-"
|
||||||
|
dot_disk = self.iso_root.joinpath(".disk")
|
||||||
|
for initrd in casper_dir.glob(f"{prefix}*"):
|
||||||
|
initrddir = self.workdir.joinpath("initrd")
|
||||||
|
with self.logger.logged(
|
||||||
|
f"unpacking {initrd.name} ...", done_msg="... done"
|
||||||
|
):
|
||||||
|
self.logger.run(["unmkinitramfs", initrd, initrddir])
|
||||||
|
# unmkinitramfs can produce different directory structures:
|
||||||
|
# - Platforms with early firmware: subdirs like "main/" or "early/"
|
||||||
|
# containing conf/uuid.conf
|
||||||
|
# - Other platforms: conf/uuid.conf directly in the root
|
||||||
|
# Try to find uuid.conf in both locations. The [uuid_conf] = confs
|
||||||
|
# unpacking asserts exactly one match; multiple matches would
|
||||||
|
# indicate an unexpected initrd structure.
|
||||||
|
confs = list(initrddir.glob("*/conf/uuid.conf"))
|
||||||
|
if confs:
|
||||||
|
[uuid_conf] = confs
|
||||||
|
elif initrddir.joinpath("conf/uuid.conf").exists():
|
||||||
|
uuid_conf = initrddir.joinpath("conf/uuid.conf")
|
||||||
|
else:
|
||||||
|
raise Exception("uuid.conf not found")
|
||||||
|
self.logger.log(f"found {uuid_conf.relative_to(initrddir)}")
|
||||||
|
uuid_conf.rename(
|
||||||
|
dot_disk.joinpath("casper-uuid-" + initrd.name[len(prefix) :])
|
||||||
|
)
|
||||||
|
shutil.rmtree(initrddir)
|
||||||
|
|
||||||
|
def add_live_filesystem(self, artifact_prefix: pathlib.Path):
|
||||||
|
# Link build artifacts into the ISO's casper directory. We use hardlinks
|
||||||
|
# (not copies) for filesystem efficiency - they reference the same inode.
|
||||||
|
#
|
||||||
|
# Artifacts come from the layered build with names like "for-iso.base.squashfs"
|
||||||
|
# and need to be renamed for casper. The prefix is stripped, so:
|
||||||
|
# for-iso.base.squashfs -> base.squashfs
|
||||||
|
# for-iso.kernel-generic -> filesystem.kernel-generic
|
||||||
|
#
|
||||||
|
# Kernel and initrd get the extra "filesystem." prefix because casper
|
||||||
|
# expects names like filesystem.kernel-* and filesystem.initrd-*.
|
||||||
|
casper_dir = self.iso_root.joinpath("casper")
|
||||||
|
artifact_dir = artifact_prefix.parent
|
||||||
|
filename_prefix = artifact_prefix.name
|
||||||
|
|
||||||
|
def link(src, target_name):
|
||||||
|
target = casper_dir.joinpath(target_name)
|
||||||
|
self.logger.log(
|
||||||
|
f"creating link from $ISOROOT/casper/{target_name} to $src/{src.name}"
|
||||||
|
)
|
||||||
|
target.hardlink_to(src)
|
||||||
|
|
||||||
|
with self.logger.logged(
|
||||||
|
f"linking artifacts from {casper_dir} to {artifact_dir}"
|
||||||
|
):
|
||||||
|
for ext in "squashfs", "squashfs.gpg", "size", "manifest", "yaml":
|
||||||
|
for path in artifact_dir.glob(f"{filename_prefix}*.{ext}"):
|
||||||
|
newname = path.name[len(filename_prefix) :]
|
||||||
|
link(path, newname)
|
||||||
|
for item in "kernel", "initrd":
|
||||||
|
for path in artifact_dir.glob(f"{filename_prefix}{item}-*"):
|
||||||
|
newname = "filesystem." + path.name[len(filename_prefix) :]
|
||||||
|
link(path, newname)
|
||||||
|
self._extract_casper_uuids()
|
||||||
|
|
||||||
|
def make_bootable(self, project: str, capproject: str, subarch: str):
|
||||||
|
# debian-cd is Ubuntu's CD/ISO image build system. It contains
|
||||||
|
# architecture and series-specific boot configuration scripts that set up
|
||||||
|
# GRUB, syslinux, EFI boot, etc. The tools/boot/$series/boot-$arch script
|
||||||
|
# knows how to make an ISO bootable for each architecture.
|
||||||
|
#
|
||||||
|
# TODO: The boot configuration logic should eventually be ported directly
|
||||||
|
# into isobuilder to avoid this external dependency and git clone.
|
||||||
|
debian_cd_dir = self.workdir.joinpath("debian-cd")
|
||||||
|
with self.logger.logged("cloning debian-cd"):
|
||||||
|
self.logger.run(
|
||||||
|
[
|
||||||
|
"git",
|
||||||
|
"clone",
|
||||||
|
"--depth=1",
|
||||||
|
"https://git.launchpad.net/~ubuntu-cdimage/debian-cd/+git/ubuntu",
|
||||||
|
debian_cd_dir,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
# Override apt-selection to use our ISO's apt configuration instead of
|
||||||
|
# debian-cd's default. This ensures the boot scripts query packages from
|
||||||
|
# the correct repository (our pool) when determining boot requirements.
|
||||||
|
apt_selection = debian_cd_dir.joinpath("tools/apt-selection")
|
||||||
|
with self.logger.logged("overwriting apt-selection"):
|
||||||
|
apt_selection.write_text(
|
||||||
|
"#!/bin/sh\n" f"APT_CONFIG={self.apt_state.apt_conf_path} apt-get $@\n"
|
||||||
|
)
|
||||||
|
env = dict(
|
||||||
|
os.environ,
|
||||||
|
BASEDIR=str(debian_cd_dir),
|
||||||
|
DIST=self.series,
|
||||||
|
PROJECT=project,
|
||||||
|
CAPPROJECT=capproject,
|
||||||
|
SUBARCH=subarch,
|
||||||
|
)
|
||||||
|
tool_name = f"tools/boot/{self.series}/boot-{self.arch}"
|
||||||
|
with self.logger.logged(f"running {tool_name} ...", done_msg="... done"):
|
||||||
|
self.logger.run(
|
||||||
|
[
|
||||||
|
debian_cd_dir.joinpath(tool_name),
|
||||||
|
"1",
|
||||||
|
self.iso_root,
|
||||||
|
],
|
||||||
|
env=env,
|
||||||
|
)
|
||||||
|
|
||||||
|
def checksum(self):
|
||||||
|
# Generate md5sum.txt for ISO integrity verification.
|
||||||
|
# - Symlinks are excluded because their targets are already checksummed
|
||||||
|
# - Files are sorted for deterministic, reproducible output across builds
|
||||||
|
# - Paths use "./" prefix and we run md5sum from iso_root so the output
|
||||||
|
# matches what users get when they verify with "md5sum -c" from the ISO
|
||||||
|
all_files = []
|
||||||
|
for dirpath, dirnames, filenames in self.iso_root.walk():
|
||||||
|
filepaths = [dirpath.joinpath(filename) for filename in filenames]
|
||||||
|
all_files.extend(
|
||||||
|
"./" + str(filepath.relative_to(self.iso_root))
|
||||||
|
for filepath in filepaths
|
||||||
|
if not filepath.is_symlink()
|
||||||
|
)
|
||||||
|
self.iso_root.joinpath("md5sum.txt").write_bytes(
|
||||||
|
self.logger.run(
|
||||||
|
["md5sum"] + sorted(all_files),
|
||||||
|
cwd=self.iso_root,
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
).stdout
|
||||||
|
)
|
||||||
|
|
||||||
|
def make_iso(self, dest: pathlib.Path, volid: str | None):
|
||||||
|
# 1.mkisofs_opts is generated by debian-cd's make_bootable step. The "1"
|
||||||
|
# refers to "pass 1" of the build (a legacy naming convention). It contains
|
||||||
|
# architecture-specific xorriso options for boot sectors, EFI images, etc.
|
||||||
|
mkisofs_opts = shlex.split(self.workdir.joinpath("1.mkisofs_opts").read_text())
|
||||||
|
self.checksum()
|
||||||
|
# xorriso with "-as mkisofs" runs in mkisofs compatibility mode.
|
||||||
|
# -r enables Rock Ridge extensions for Unix metadata (permissions, symlinks).
|
||||||
|
# -iso-level 3 (amd64 only) allows files >4GB which some amd64 ISOs need.
|
||||||
|
cmd: list[str | pathlib.Path] = ["xorriso", "-as", "mkisofs", "-r"]
|
||||||
|
if self.arch == "amd64":
|
||||||
|
cmd.extend(["-iso-level", "3"])
|
||||||
|
if volid:
|
||||||
|
cmd += ["-V", volid]
|
||||||
|
cmd += mkisofs_opts + [self.iso_root, "-o", dest]
|
||||||
|
with self.logger.logged("running xorriso"):
|
||||||
|
self.logger.run(cmd, cwd=self.workdir, check=True)
|
||||||
58
live-build/isobuilder/gpg_key.py
Normal file
58
live-build/isobuilder/gpg_key.py
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
import pathlib
|
||||||
|
import subprocess
|
||||||
|
|
||||||
|
key_conf = """\
|
||||||
|
%no-protection
|
||||||
|
Key-Type: eddsa
|
||||||
|
Key-Curve: Ed25519
|
||||||
|
Key-Usage: sign
|
||||||
|
Name-Real: Ubuntu ISO One-Time Signing Key
|
||||||
|
Name-Email: noone@nowhere.invalid
|
||||||
|
Expire-Date: 0
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class EphemeralGPGKey:
|
||||||
|
|
||||||
|
def __init__(self, logger, gpghome):
|
||||||
|
self.logger = logger
|
||||||
|
self.gpghome = gpghome
|
||||||
|
|
||||||
|
def _run_gpg(self, cmd, **kwargs):
|
||||||
|
return self.logger.run(
|
||||||
|
["gpg", "--homedir", self.gpghome] + cmd, check=True, **kwargs
|
||||||
|
)
|
||||||
|
|
||||||
|
def create(self):
|
||||||
|
with self.logger.logged("creating gpg key ...", done_msg="... done"):
|
||||||
|
self.gpghome.mkdir(mode=0o700)
|
||||||
|
self._run_gpg(
|
||||||
|
["--gen-key", "--batch"],
|
||||||
|
input=key_conf,
|
||||||
|
text=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
def sign(self, path: pathlib.Path):
|
||||||
|
with self.logger.logged(f"signing {path}"):
|
||||||
|
with path.open("rb") as inp:
|
||||||
|
with pathlib.Path(str(path) + ".gpg").open("wb") as outp:
|
||||||
|
self._run_gpg(
|
||||||
|
[
|
||||||
|
"--no-options",
|
||||||
|
"--batch",
|
||||||
|
"--no-tty",
|
||||||
|
"--armour",
|
||||||
|
"--digest-algo",
|
||||||
|
"SHA512",
|
||||||
|
"--detach-sign",
|
||||||
|
],
|
||||||
|
stdin=inp,
|
||||||
|
stdout=outp,
|
||||||
|
)
|
||||||
|
|
||||||
|
def export_public(self) -> str:
|
||||||
|
return self._run_gpg(
|
||||||
|
["--export", "--armor"],
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
text=True,
|
||||||
|
).stdout
|
||||||
166
live-build/isobuilder/pool_builder.py
Normal file
166
live-build/isobuilder/pool_builder.py
Normal file
@ -0,0 +1,166 @@
|
|||||||
|
import pathlib
|
||||||
|
import subprocess
|
||||||
|
import tempfile
|
||||||
|
|
||||||
|
from isobuilder.apt_state import AptStateManager, PackageInfo
|
||||||
|
|
||||||
|
|
||||||
|
generate_template = """
|
||||||
|
Dir::ArchiveDir "{root}";
|
||||||
|
Dir::CacheDir "{scratch}/apt-ftparchive-db";
|
||||||
|
|
||||||
|
TreeDefault::Contents " ";
|
||||||
|
|
||||||
|
Tree "dists/{series}" {{
|
||||||
|
FileList "{scratch}/filelist_$(SECTION)";
|
||||||
|
Sections "{components}";
|
||||||
|
Architectures "{arches}";
|
||||||
|
}}
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class PoolBuilder:
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self, logger, series: str, apt_state: AptStateManager, rootdir: pathlib.Path
|
||||||
|
):
|
||||||
|
self.logger = logger
|
||||||
|
self.series = series
|
||||||
|
self.apt_state = apt_state
|
||||||
|
self.rootdir = rootdir
|
||||||
|
self.arches: set[str] = set()
|
||||||
|
self._present_components: set[str] = set()
|
||||||
|
|
||||||
|
def add_packages(self, pkglist: list[PackageInfo]):
|
||||||
|
for pkg_info in pkglist:
|
||||||
|
if pkg_info.architecture != "all":
|
||||||
|
self.arches.add(pkg_info.architecture)
|
||||||
|
self.apt_state.download(self.rootdir, pkg_info)
|
||||||
|
|
||||||
|
def make_packages(self) -> None:
|
||||||
|
with self.logger.logged("making Packages files"):
|
||||||
|
with tempfile.TemporaryDirectory() as tmpdir:
|
||||||
|
scratchdir = pathlib.Path(tmpdir)
|
||||||
|
with self.logger.logged("scanning for packages"):
|
||||||
|
for component in ["main", "restricted", "universe", "multiverse"]:
|
||||||
|
if not self.rootdir.joinpath("pool", component).is_dir():
|
||||||
|
continue
|
||||||
|
self._present_components.add(component)
|
||||||
|
for arch in self.arches:
|
||||||
|
self.rootdir.joinpath(
|
||||||
|
"dists", self.series, component, f"binary-{arch}"
|
||||||
|
).mkdir(parents=True)
|
||||||
|
proc = self.logger.run(
|
||||||
|
["find", f"pool/{component}"],
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
cwd=self.rootdir,
|
||||||
|
encoding="utf-8",
|
||||||
|
check=True,
|
||||||
|
)
|
||||||
|
scratchdir.joinpath(f"filelist_{component}").write_text(
|
||||||
|
"\n".join(sorted(proc.stdout.splitlines()))
|
||||||
|
)
|
||||||
|
with self.logger.logged("writing apt-ftparchive config"):
|
||||||
|
scratchdir.joinpath("apt-ftparchive-db").mkdir()
|
||||||
|
generate_path = scratchdir.joinpath("generate-binary")
|
||||||
|
generate_path.write_text(
|
||||||
|
generate_template.format(
|
||||||
|
arches=" ".join(self.arches),
|
||||||
|
series=self.series,
|
||||||
|
root=self.rootdir.resolve(),
|
||||||
|
scratch=scratchdir.resolve(),
|
||||||
|
components=" ".join(self._present_components),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
with self.logger.logged("running apt-ftparchive generate"):
|
||||||
|
self.logger.run(
|
||||||
|
[
|
||||||
|
"apt-ftparchive",
|
||||||
|
"--no-contents",
|
||||||
|
"--no-md5",
|
||||||
|
"--no-sha1",
|
||||||
|
"--no-sha512",
|
||||||
|
"generate",
|
||||||
|
generate_path,
|
||||||
|
],
|
||||||
|
check=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
def make_release(self) -> pathlib.Path:
|
||||||
|
# Build the Release file by merging metadata from the mirror with
|
||||||
|
# checksums for our pool. We can't just use apt-ftparchive's Release
|
||||||
|
# output directly because:
|
||||||
|
# 1. apt-ftparchive doesn't know about Origin, Label, Suite, Version,
|
||||||
|
# Codename, etc. - these come from the mirror and maintain package
|
||||||
|
# provenance
|
||||||
|
# 2. We keep the mirror's Date (when packages were released) rather than
|
||||||
|
# apt-ftparchive's Date (when we ran the command)
|
||||||
|
# 3. We need to override Architectures/Components to match our pool
|
||||||
|
#
|
||||||
|
# There may be a cleaner way (apt-get indextargets?) but this works.
|
||||||
|
with self.logger.logged("making Release file"):
|
||||||
|
in_release = self.apt_state.in_release_path()
|
||||||
|
cp_mirror_release = self.logger.run(
|
||||||
|
["gpg", "--verify", "--output", "-", in_release],
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
encoding="utf-8",
|
||||||
|
check=False,
|
||||||
|
)
|
||||||
|
if cp_mirror_release.returncode not in (0, 2):
|
||||||
|
# gpg returns code 2 when the public key the InRelease is
|
||||||
|
# signed with is not available, which is most of the time.
|
||||||
|
raise Exception("gpg failed")
|
||||||
|
mirror_release_lines = cp_mirror_release.stdout.splitlines()
|
||||||
|
release_dir = self.rootdir.joinpath("dists", self.series)
|
||||||
|
af_release_lines = self.logger.run(
|
||||||
|
[
|
||||||
|
"apt-ftparchive",
|
||||||
|
"--no-contents",
|
||||||
|
"--no-md5",
|
||||||
|
"--no-sha1",
|
||||||
|
"--no-sha512",
|
||||||
|
"release",
|
||||||
|
".",
|
||||||
|
],
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
encoding="utf-8",
|
||||||
|
cwd=release_dir,
|
||||||
|
check=True,
|
||||||
|
).stdout.splitlines()
|
||||||
|
# Build the final Release file by merging mirror metadata with pool
|
||||||
|
# checksums.
|
||||||
|
# Strategy:
|
||||||
|
# 1. Take metadata fields (Suite, Origin, etc.) from the mirror's InRelease
|
||||||
|
# 2. Override Architectures and Components to match what's actually in our
|
||||||
|
# pool
|
||||||
|
# 3. Skip the mirror's checksum sections (MD5Sum, SHA256, etc.) because they
|
||||||
|
# don't apply to our pool
|
||||||
|
# 4. Skip Acquire-By-Hash since we don't use it
|
||||||
|
# 5. Append checksums from apt-ftparchive (but not the Date field)
|
||||||
|
release_lines = []
|
||||||
|
skipping = False
|
||||||
|
for line in mirror_release_lines:
|
||||||
|
if line.startswith("Architectures:"):
|
||||||
|
line = "Architectures: " + " ".join(sorted(self.arches))
|
||||||
|
elif line.startswith("Components:"):
|
||||||
|
line = "Components: " + " ".join(sorted(self._present_components))
|
||||||
|
elif line.startswith("MD5") or line.startswith("SHA"):
|
||||||
|
# Start of a checksum section - skip this and indented lines below
|
||||||
|
# it
|
||||||
|
skipping = True
|
||||||
|
elif not line.startswith(" "):
|
||||||
|
# Non-indented line means we've left the checksum section if we were
|
||||||
|
# in one.
|
||||||
|
skipping = False
|
||||||
|
if line.startswith("Acquire-By-Hash"):
|
||||||
|
continue
|
||||||
|
if not skipping:
|
||||||
|
release_lines.append(line)
|
||||||
|
# Append checksums from apt-ftparchive, but skip its Date field
|
||||||
|
# (we want to keep the Date from the mirror release)
|
||||||
|
for line in af_release_lines:
|
||||||
|
if not line.startswith("Date"):
|
||||||
|
release_lines.append(line)
|
||||||
|
release_path = release_dir.joinpath("Release")
|
||||||
|
release_path.write_text("\n".join(release_lines))
|
||||||
|
return release_path
|
||||||
@ -184,6 +184,20 @@ build_layered_squashfs () {
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
create_squashfs "${overlay_dir}" ${squashfs_f}
|
create_squashfs "${overlay_dir}" ${squashfs_f}
|
||||||
|
# Create a "for-iso" variant of the squashfs for ISO builds. For
|
||||||
|
# the root layer (the base system) when building with a pool, we
|
||||||
|
# need to include cdrom.sources so casper can access the ISO's
|
||||||
|
# package repository. This requires regenerating the squashfs with
|
||||||
|
# that file included, then removing it (so it doesn't pollute the
|
||||||
|
# regular squashfs). Non-root layers (desktop environment, etc.)
|
||||||
|
# and builds without pools can just hardlink to the regular squashfs.
|
||||||
|
if [ -n "${POOL_SEED_NAME}" ] && $(is_root_layer $pass); then
|
||||||
|
isobuild generate-sources --mountpoint=/cdrom > ${overlay_dir}/etc/apt/sources.list.d/cdrom.sources
|
||||||
|
create_squashfs "${overlay_dir}" ${PWD}/for-iso.${pass}.squashfs
|
||||||
|
rm ${overlay_dir}/etc/apt/sources.list.d/cdrom.sources
|
||||||
|
else
|
||||||
|
ln ${squashfs_f} ${PWD}/for-iso.${pass}.squashfs
|
||||||
|
fi
|
||||||
|
|
||||||
if [ -f config/$pass.catalog-in.yaml ]; then
|
if [ -f config/$pass.catalog-in.yaml ]; then
|
||||||
echo "Expanding catalog entry template for $pass"
|
echo "Expanding catalog entry template for $pass"
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user