2020-08-28 11:13:04 -05:00
|
|
|
---
|
|
|
|
layout: doc
|
2021-08-04 22:22:12 -07:00
|
|
|
title: Template manager
|
2020-08-28 11:13:04 -05:00
|
|
|
permalink: /doc/template-manager/
|
|
|
|
---
|
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
This document discusses the designs and technical details of `qvm-template`, a
|
|
|
|
template manager application. The goal of the project is to design a new
|
|
|
|
mechanism for template distribution and a unified tool for template management.
|
2020-08-29 21:04:36 +08:00
|
|
|
|
2020-09-02 01:06:20 +08:00
|
|
|
## Motivation
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2020-09-02 01:10:49 +08:00
|
|
|
This project was originally proposed in the 2020 Google Summer of Code program.
|
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
Previously, templates were distributed by RPM packages and managed by
|
|
|
|
`yum`/`dnf`. However, tracking inherently dynamic VM images with a package
|
|
|
|
manager suited for static files creates some challenges. For example, users may
|
|
|
|
accidentally update the images, overriding local changes
|
|
|
|
([#996](https://github.com/QubesOS/qubes-issues/issues/996),
|
|
|
|
[#1647](https://github.com/QubesOS/qubes-issues/issues/1647)). (Or in the case
|
|
|
|
of [#2061](https://github.com/QubesOS/qubes-issues/issues/2061), want to
|
|
|
|
specifically override the changes.) Other operations that work well on normal
|
|
|
|
VMs are also somewhat inconsistent on RPM-managed templates. This includes
|
|
|
|
actions such as renaming
|
|
|
|
([#839](https://github.com/QubesOS/qubes-issues/issues/839)), removal
|
|
|
|
([#5509](https://github.com/QubesOS/qubes-issues/issues/5509)) and
|
|
|
|
backup/restore ([#1385](https://github.com/QubesOS/qubes-issues/issues/1385),
|
|
|
|
[#1453](https://github.com/QubesOS/qubes-issues/issues/1453), [discussion
|
|
|
|
thread
|
|
|
|
1](https://groups.google.com/forum/#!topic/qubes-devel/rwc2_miCNNE/discussion),
|
|
|
|
[discussion thread
|
|
|
|
2](https://groups.google.com/forum/#!topic/qubes-users/uQEUpv4THsY/discussion)).
|
|
|
|
In turn, this creates inconveniences and confusion for users
|
|
|
|
([#1403](https://github.com/QubesOS/qubes-issues/issues/1403),
|
|
|
|
[#4518](https://github.com/QubesOS/qubes-issues/issues/4518)).
|
|
|
|
|
|
|
|
Also, the usage of RPM packages meant that installing a template results in
|
|
|
|
arbitrary code execution, which is not ideal.
|
|
|
|
|
|
|
|
Besides distribution, users may also wish to have an integrated template
|
|
|
|
management application
|
|
|
|
([#2062](https://github.com/QubesOS/qubes-issues/issues/2062),
|
|
|
|
[#2064](https://github.com/QubesOS/qubes-issues/issues/2064),
|
|
|
|
[#2534](https://github.com/QubesOS/qubes-issues/issues/2534),
|
|
|
|
[#3040](https://github.com/QubesOS/qubes-issues/issues/3040)), as opposed to
|
|
|
|
the situation where multiple programs are required for different purposes,
|
|
|
|
e.g., `qubes-dom0-update`, `dnf`, `qvm-remove`, `qubes-manager`.
|
|
|
|
|
|
|
|
To tackle these issues, `qvm-template` was created. It strives to provide not
|
|
|
|
only a better mechanism for handling template installation but also a
|
|
|
|
consistent user-facing interface to deal with template management.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2020-09-02 01:06:20 +08:00
|
|
|
## Features
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
- Install/reinstall/downgrade/upgrade templates, either from local packages or
|
|
|
|
remote repositories
|
2020-08-28 02:52:01 +08:00
|
|
|
- Ability to install templates in alternative pools
|
2021-08-04 22:22:12 -07:00
|
|
|
- Possibility for the template package to specify options such as the kernel
|
|
|
|
or `virt_mode` used by the resulting template
|
2020-08-28 02:52:01 +08:00
|
|
|
- List and show information about local and available templates
|
|
|
|
- Machine-readable output for easy extension
|
|
|
|
- Search for templates
|
|
|
|
- Remove templates
|
2021-08-04 22:22:12 -07:00
|
|
|
- Optionally, VMs based on the template to be removed can be either removed
|
|
|
|
or "disassociated" -- namely, have their templates changed to a "dummy"
|
|
|
|
one.
|
2020-08-28 02:52:01 +08:00
|
|
|
- Show available repositories
|
|
|
|
- Works in both dom0 and management VMs by utilizing the Admin API
|
|
|
|
- Works well with existing tools
|
|
|
|
- Command-line interface with DNF-like usage
|
|
|
|
- A graphical interface also available
|
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
## Package format
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
The RPM package format is still used. However, the contents are manually
|
|
|
|
extracted instead of installing the whole package. This allows us to take
|
|
|
|
advantage of existing tools for things like repository management. We can also
|
|
|
|
avoid the burden of dealing with verification, reducing the risk of issues like
|
|
|
|
[QSB-028](/news/2016/12/19/qsb-28/).
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
The package name should be in the form `qubes-template-<TEMPLATE_NAME>`.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
The package metadata (summary, description, etc.) should not contain the `|`
|
|
|
|
character to avoid possibly cryptic errors. This is because of its use as an
|
|
|
|
internal separator. Note that as we already consider the repository metadata
|
|
|
|
untrusted. This should not result in security issues.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
The file structure should be quite similar to previous template RPMs. Namely,
|
|
|
|
there should be the following files in the package:
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
- `var/lib/qubes/vm-templates/<TEMPLATE_NAME>/root.img.part.[00,01,...]`
|
2020-08-28 02:52:01 +08:00
|
|
|
- Split tarball of template `root.img`
|
2021-08-04 22:22:12 -07:00
|
|
|
- Note that the file is still split due to tools such as `rpm2cpio` not
|
|
|
|
supporting large files. (Notably, the cpio format does not support files
|
|
|
|
over 4GiB.)
|
|
|
|
- `var/lib/qubes/vm-templates/<TEMPLATE_NAME>/template.conf`
|
2020-08-28 02:52:01 +08:00
|
|
|
- Stores custom package metadata (as RPM does not support custom attributes).
|
|
|
|
- Uses `KEY=VALUE` format.
|
2021-08-04 22:22:12 -07:00
|
|
|
- Fields (corresponding to
|
|
|
|
[qvm-prefs](https://dev.qubes-os.org/projects/core-admin-client/en/stable/manpages/qvm-prefs.html#common-properties)
|
|
|
|
and
|
|
|
|
[qvm-features](https://dev.qubes-os.org/projects/core-admin-client/en/stable/manpages/qvm-features.html#list-of-known-features)
|
|
|
|
tags with the same names)
|
2020-08-28 02:52:01 +08:00
|
|
|
- `virt_mode`
|
|
|
|
- Setting this to `pv` requires user confirmation.
|
2021-06-08 13:49:07 +08:00
|
|
|
- Permitted values: `pv`, `pvh`, `hvm`.
|
2020-08-28 02:52:01 +08:00
|
|
|
- `kernel`
|
2021-08-04 22:22:12 -07:00
|
|
|
- Only allowed to be set to "" (without quotes), i.e., "none", for
|
|
|
|
PVGrub.
|
|
|
|
- Network-related flags: (Must be set to IPv4 addresses in the form of
|
|
|
|
`x.x.x.x`.)
|
2021-06-08 13:49:07 +08:00
|
|
|
- `net.fake-ip`
|
|
|
|
- `net.fake-gateway`
|
|
|
|
- `net.fake-netmask`
|
2021-08-04 22:22:12 -07:00
|
|
|
- Boolean flags: (Permitted values are "1" and "0", denoting "true" and
|
|
|
|
"false" respectively.)
|
2020-09-15 11:56:04 +08:00
|
|
|
- `no-monitor-layout`
|
|
|
|
- `pci-e820-host`
|
|
|
|
- `linux-stubdom`
|
|
|
|
- `gui`
|
|
|
|
- `gui-emulated`
|
|
|
|
- `qrexec`
|
2021-08-04 22:22:12 -07:00
|
|
|
- `var/lib/qubes/vm-templates/<TEMPLATE_NAME>/whitelisted-appmenus.list`
|
2020-08-28 02:52:01 +08:00
|
|
|
- Contains default app menu entries of the template itself.
|
2021-08-04 22:22:12 -07:00
|
|
|
- `var/lib/qubes/vm-templates/<TEMPLATE_NAME>/vm-whitelisted-appmenus.list`
|
2020-08-28 02:52:01 +08:00
|
|
|
- Contains default app menu entries of VMs based on the template.
|
2021-08-04 22:22:12 -07:00
|
|
|
- `var/lib/qubes/vm-templates/<TEMPLATE_NAME>/netvm-whitelisted-appmenus.list`
|
2020-08-28 02:52:01 +08:00
|
|
|
- Contains default app menu entries of NetVMs based on the template.
|
|
|
|
- These three files are the same as the current format.
|
2021-08-04 22:22:12 -07:00
|
|
|
- Note that the contents of these files are stored in `qvm-features` upon
|
|
|
|
installation. See the section below for details.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
## Metadata storage
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
The template manager needs to keep metadata of installed templates such as
|
|
|
|
versions and origin. This data can be stored via `qvm-features` to keep things
|
|
|
|
consistent when, e.g., `qvm-remove` is used. Besides, backups are also more
|
|
|
|
easily handled this way.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
Also, the fields can serve as an indicator of whether a template is installed
|
|
|
|
by `qvm-template`.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2020-09-02 01:06:20 +08:00
|
|
|
### Fields
|
2020-08-28 02:52:01 +08:00
|
|
|
|
|
|
|
Most of the fields should be fairly self-explanatory.
|
|
|
|
|
|
|
|
- `template-name`
|
2021-08-04 22:22:12 -07:00
|
|
|
- Note that this field needs to be consistent with the template name to be
|
|
|
|
considered valid.
|
2020-08-28 02:52:01 +08:00
|
|
|
- `template-epoch`
|
|
|
|
- `template-version`
|
|
|
|
- `template-release`
|
|
|
|
- `template-reponame`
|
|
|
|
- `template-buildtime`
|
2020-08-30 02:04:35 +08:00
|
|
|
- `template-installtime`
|
2020-09-15 00:31:11 +08:00
|
|
|
- The times are in UTC, and are of the format `YYYY-MM-DD HH:MM:SS`.
|
2021-06-08 13:40:25 +08:00
|
|
|
- `template-license`
|
2020-08-28 02:52:01 +08:00
|
|
|
- `template-url`
|
|
|
|
- `template-summary`
|
|
|
|
- `template-description`
|
2021-08-04 22:22:12 -07:00
|
|
|
- Note that the newlines in this field are converted to `|` to work better
|
|
|
|
with existing tools like `qvm-features`.
|
2020-08-28 02:52:01 +08:00
|
|
|
- `menu-items`
|
|
|
|
- `default-menu-items`
|
|
|
|
- `netvm-menu-items`
|
2021-08-04 22:22:12 -07:00
|
|
|
- The `*menu-items` entries store the contents of
|
|
|
|
`var/lib/qubes/vm-templates/<TEMPLATE_NAME>/whitelisted-appmenus.list`,
|
|
|
|
`var/lib/qubes/vm-templates/<TEMPLATE_NAME>/vm-whitelisted-appmenus.list`,
|
|
|
|
`var/lib/qubes/vm-templates/<TEMPLATE_NAME>/netvm-whitelisted-appmenus.list`
|
|
|
|
respectively.
|
|
|
|
- Note that newlines are converted to spaces, again for it to work better
|
|
|
|
with existing tools. This should not cause ambiguity as [the FreeDesktop
|
|
|
|
specifications](https://specifications.freedesktop.org/desktop-entry-spec/desktop-entry-spec-latest.html)
|
|
|
|
forbid spaces in .desktop file names.
|
|
|
|
|
|
|
|
## Repository management
|
|
|
|
|
|
|
|
For UpdateVMs to access the repository configuration, the package
|
|
|
|
[qubes-repo-templates](https://github.com/WillyPillow/qubes-repo-templates) is
|
|
|
|
created with the following contents:
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-06-08 13:39:02 +08:00
|
|
|
- `/etc/qubes/repo-templates/*.repo`: repository configuration
|
|
|
|
- `/etc/qubes/repo-templates/keys`: PGP keys
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
As template keys may be less trusted, they are *not* added to the system RPM
|
|
|
|
keychain but instead managed separately.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
## Qrexec protocol
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
Dom0 and management VMs without network access also need to interact with
|
|
|
|
template repositories. The following qrexec calls that list and download
|
|
|
|
templates are thus proposed.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
|
|
|
- `qubes.TemplateSearch`: wraps `dnf repoquery`
|
|
|
|
- `qubes.TemplateDownload`: wraps `dnf download`
|
|
|
|
|
2020-09-02 01:06:20 +08:00
|
|
|
### Input
|
2020-08-28 02:52:01 +08:00
|
|
|
|
|
|
|
Both calls accept the following format from standard input:
|
|
|
|
|
2021-06-08 13:39:02 +08:00
|
|
|
```text
|
2020-08-28 02:52:01 +08:00
|
|
|
arg1
|
|
|
|
arg2
|
|
|
|
...
|
|
|
|
argN
|
|
|
|
package-file-spec
|
|
|
|
---
|
|
|
|
repo config
|
|
|
|
```
|
|
|
|
|
|
|
|
In other words, the input consists of two parts separated by the line `---`.
|
2021-08-04 22:22:12 -07:00
|
|
|
The first part contains some arguments and `package-file-spec` that indicates
|
|
|
|
the pattern to be queried or downloaded. The following arguments are allowed:
|
2020-08-28 02:52:01 +08:00
|
|
|
|
|
|
|
- `--enablerepo=<repoid>`
|
|
|
|
- `--disablerepo=<repoid>`
|
|
|
|
- `--repoid=<repoid>`
|
|
|
|
- `--releasever=<release>`
|
|
|
|
- `--refresh`
|
|
|
|
|
|
|
|
where the usage is identical to that of DNF.
|
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
For the exact definition of `package-file-spec`, refer to the DNF
|
|
|
|
documentation.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
|
|
|
The second part contains the repository configurations in `yum.repos.d` format.
|
|
|
|
|
2020-09-02 01:06:20 +08:00
|
|
|
### Output
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
`qubes.TemplateSearch` prints each package in
|
|
|
|
`%{name}|%{epoch}|%{version}|%{release}|%{reponame}|%{downloadsize}|%{buildtime}|%{license}|%{url}|%{summary}|%{description}|`
|
|
|
|
format to standard output, separated by newlines. Note that there is a `|` at
|
|
|
|
the end of the line. This is because `%{description}` may contain newlines, and
|
|
|
|
doing so allows us to split the entries by `|\n`. (As we are using `dnf
|
|
|
|
repoquery --qf`, we are unable to escape the newlines in advance.)
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
`qubes.TemplateDownload`, on the other hand, directly outputs the downloaded
|
|
|
|
content to standard output.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
## Machine-readable output
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
The commands `qvm-template list` and `qvm-template info` provide
|
|
|
|
machine-readable output in both pipe(`|`)-separated and JSON format. See the
|
|
|
|
`qvm-template` man page for details.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
## Interactions with existing tools
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2020-09-02 01:06:20 +08:00
|
|
|
### `qvm-remove`
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
The existing `qvm-remove` tool should behave identically to `qvm-template
|
|
|
|
remove` -- albeit without fancy features like disassociation. This is unlike
|
|
|
|
the previous situation where `qvm-remove` cannot remove RPM-installed
|
|
|
|
templates.
|
2020-09-15 11:56:04 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
Notably, the metadata needs no special handling as it is stored in VM features
|
|
|
|
and thus automatically consistent.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
### Renaming and cloning
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
A template is treated as non-manager-installed once renamed or cloned. However,
|
|
|
|
relevant metadata in the VM features is still retained for future extension and
|
|
|
|
to serve as a hint for the user.
|
2020-08-28 02:52:01 +08:00
|
|
|
|
2021-08-04 22:22:12 -07:00
|
|
|
## Further reading
|
2020-08-28 02:52:01 +08:00
|
|
|
|
|
|
|
Initial Google Summer of Code (2020) project proposal:
|
2020-08-29 17:38:01 +08:00
|
|
|
|
2020-08-28 02:52:01 +08:00
|
|
|
- <https://hackmd.io/aYauztkGR0iOIoh8fJLecw>
|
|
|
|
|
|
|
|
Previous design document:
|
2020-08-29 17:38:01 +08:00
|
|
|
|
2020-08-28 02:52:01 +08:00
|
|
|
- <https://gist.github.com/WillyPillow/b8a643ddbd9235a97bc187e6e44b16e4>
|
|
|
|
|
|
|
|
Discussion threads:
|
2020-08-29 17:38:01 +08:00
|
|
|
|
2020-08-28 02:52:01 +08:00
|
|
|
- <https://groups.google.com/forum/#!topic/qubes-devel/6Zb_WLy3GY4>
|
|
|
|
- <https://groups.google.com/forum/#!topic/qubes-devel/PyJogqT1TUg>
|
|
|
|
- <https://groups.google.com/forum/#!topic/qubes-devel/2XaMP4Us3kg>
|
|
|
|
- <https://groups.google.com/forum/#!topic/qubes-devel/wF_84b1BR0A>
|
2020-09-04 02:21:38 +08:00
|
|
|
- <https://groups.google.com/forum/#!topic/qubes-devel/pYHnihVCBM0>
|