mirror of
https://github.com/QubesOS/qubes-doc.git
synced 2025-01-02 19:21:05 -05:00
444 lines
12 KiB
ReStructuredText
444 lines
12 KiB
ReStructuredText
|
================
|
|||
|
Template manager
|
|||
|
================
|
|||
|
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
Motivation
|
|||
|
----------
|
|||
|
|
|||
|
|
|||
|
This project was originally proposed in the 2020 Google Summer of Code
|
|||
|
program.
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
Features
|
|||
|
--------
|
|||
|
|
|||
|
|
|||
|
- Install/reinstall/downgrade/upgrade templates, either from local
|
|||
|
packages or remote repositories
|
|||
|
|
|||
|
- Ability to install templates in alternative pools
|
|||
|
|
|||
|
- Possibility for the template package to specify options such as
|
|||
|
the kernel or ``virt_mode`` used by the resulting template
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- List and show information about local and available templates
|
|||
|
|
|||
|
- Machine-readable output for easy extension
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- Search for templates
|
|||
|
|
|||
|
- Remove templates
|
|||
|
|
|||
|
- Optionally, VMs based on the template to be removed can be either
|
|||
|
removed or “disassociated” – namely, have their templates changed
|
|||
|
to a “dummy” one.
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- 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
|
|||
|
|
|||
|
|
|||
|
|
|||
|
Package format
|
|||
|
--------------
|
|||
|
|
|||
|
|
|||
|
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 <https://www.qubes-os.org/news/2016/12/19/qsb-28/>`__.
|
|||
|
|
|||
|
The package name should be in the form
|
|||
|
``qubes-template-<TEMPLATE_NAME>``.
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
The file structure should be quite similar to previous template RPMs.
|
|||
|
Namely, there should be the following files in the package:
|
|||
|
|
|||
|
- ``var/lib/qubes/vm-templates/<TEMPLATE_NAME>/root.img.part.[00,01,...]``
|
|||
|
|
|||
|
- Split tarball of template ``root.img``
|
|||
|
|
|||
|
- 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``
|
|||
|
|
|||
|
- Stores custom package metadata (as RPM does not support custom
|
|||
|
attributes).
|
|||
|
|
|||
|
- Uses ``KEY=VALUE`` format.
|
|||
|
|
|||
|
- 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)
|
|||
|
|
|||
|
- ``virt_mode``
|
|||
|
|
|||
|
- Setting this to ``pv`` requires user confirmation.
|
|||
|
|
|||
|
- Permitted values: ``pv``, ``pvh``, ``hvm``.
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- ``kernel``
|
|||
|
|
|||
|
- 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``.)
|
|||
|
|
|||
|
- ``net.fake-ip``
|
|||
|
|
|||
|
- ``net.fake-gateway``
|
|||
|
|
|||
|
- ``net.fake-netmask``
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- Boolean flags: (Permitted values are “1” and “0”, denoting
|
|||
|
“true” and “false” respectively.)
|
|||
|
|
|||
|
- ``no-monitor-layout``
|
|||
|
|
|||
|
- ``pci-e820-host``
|
|||
|
|
|||
|
- ``linux-stubdom``
|
|||
|
|
|||
|
- ``gui``
|
|||
|
|
|||
|
- ``gui-emulated``
|
|||
|
|
|||
|
- ``qrexec``
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- ``var/lib/qubes/vm-templates/<TEMPLATE_NAME>/whitelisted-appmenus.list``
|
|||
|
|
|||
|
- Contains default app menu entries of the template itself.
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- ``var/lib/qubes/vm-templates/<TEMPLATE_NAME>/vm-whitelisted-appmenus.list``
|
|||
|
|
|||
|
- Contains default app menu entries of VMs based on the template.
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- ``var/lib/qubes/vm-templates/<TEMPLATE_NAME>/netvm-whitelisted-appmenus.list``
|
|||
|
|
|||
|
- Contains default app menu entries of NetVMs based on the template.
|
|||
|
|
|||
|
- These three files are the same as the current format.
|
|||
|
|
|||
|
- Note that the contents of these files are stored in
|
|||
|
``qvm-features`` upon installation. See the section below for
|
|||
|
details.
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
Metadata storage
|
|||
|
----------------
|
|||
|
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
Also, the fields can serve as an indicator of whether a template is
|
|||
|
installed by ``qvm-template``.
|
|||
|
|
|||
|
Fields
|
|||
|
^^^^^^
|
|||
|
|
|||
|
|
|||
|
Most of the fields should be fairly self-explanatory.
|
|||
|
|
|||
|
- ``template-name``
|
|||
|
|
|||
|
- Note that this field needs to be consistent with the template name
|
|||
|
to be considered valid.
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- ``template-epoch``
|
|||
|
|
|||
|
- ``template-version``
|
|||
|
|
|||
|
- ``template-release``
|
|||
|
|
|||
|
- ``template-reponame``
|
|||
|
|
|||
|
- ``template-buildtime``
|
|||
|
|
|||
|
- ``template-installtime``
|
|||
|
|
|||
|
- The times are in UTC, and are of the format
|
|||
|
``YYYY-MM-DD HH:MM:SS``.
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- ``template-license``
|
|||
|
|
|||
|
- ``template-url``
|
|||
|
|
|||
|
- ``template-summary``
|
|||
|
|
|||
|
- ``template-description``
|
|||
|
|
|||
|
- Note that the newlines in this field are converted to ``|`` to
|
|||
|
work better with existing tools like ``qvm-features``.
|
|||
|
|
|||
|
|
|||
|
|
|||
|
- ``menu-items``
|
|||
|
|
|||
|
- ``default-menu-items``
|
|||
|
|
|||
|
- ``netvm-menu-items``
|
|||
|
|
|||
|
- 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:
|
|||
|
|
|||
|
- ``/etc/qubes/repo-templates/*.repo``: repository configuration
|
|||
|
|
|||
|
- ``/etc/qubes/repo-templates/keys``: PGP keys
|
|||
|
|
|||
|
|
|||
|
|
|||
|
As template keys may be less trusted, they are *not* added to the system
|
|||
|
RPM keychain but instead managed separately.
|
|||
|
|
|||
|
Qrexec protocol
|
|||
|
---------------
|
|||
|
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
- ``qubes.TemplateSearch``: wraps ``dnf repoquery``
|
|||
|
|
|||
|
- ``qubes.TemplateDownload``: wraps ``dnf download``
|
|||
|
|
|||
|
|
|||
|
|
|||
|
Input
|
|||
|
^^^^^
|
|||
|
|
|||
|
|
|||
|
Both calls accept the following format from standard input:
|
|||
|
|
|||
|
.. code:: bash
|
|||
|
|
|||
|
arg1
|
|||
|
arg2
|
|||
|
...
|
|||
|
argN
|
|||
|
package-file-spec
|
|||
|
---
|
|||
|
repo config
|
|||
|
|
|||
|
|
|||
|
|
|||
|
In other words, the input consists of two parts separated by the line
|
|||
|
``---``. The first part contains some arguments and
|
|||
|
``package-file-spec`` that indicates the pattern to be queried or
|
|||
|
downloaded. The following arguments are allowed:
|
|||
|
|
|||
|
- ``--enablerepo=<repoid>``
|
|||
|
|
|||
|
- ``--disablerepo=<repoid>``
|
|||
|
|
|||
|
- ``--repoid=<repoid>``
|
|||
|
|
|||
|
- ``--releasever=<release>``
|
|||
|
|
|||
|
- ``--refresh``
|
|||
|
|
|||
|
|
|||
|
|
|||
|
where the usage is identical to that of DNF.
|
|||
|
|
|||
|
For the exact definition of ``package-file-spec``, refer to the DNF
|
|||
|
documentation.
|
|||
|
|
|||
|
The second part contains the repository configurations in
|
|||
|
``yum.repos.d`` format.
|
|||
|
|
|||
|
Output
|
|||
|
^^^^^^
|
|||
|
|
|||
|
|
|||
|
``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.)
|
|||
|
|
|||
|
``qubes.TemplateDownload``, on the other hand, directly outputs the
|
|||
|
downloaded content to standard output.
|
|||
|
|
|||
|
Machine-readable output
|
|||
|
-----------------------
|
|||
|
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
Interactions with existing tools
|
|||
|
--------------------------------
|
|||
|
|
|||
|
|
|||
|
``qvm-remove``
|
|||
|
^^^^^^^^^^^^^^
|
|||
|
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
Notably, the metadata needs no special handling as it is stored in VM
|
|||
|
features and thus automatically consistent.
|
|||
|
|
|||
|
Renaming and cloning
|
|||
|
^^^^^^^^^^^^^^^^^^^^
|
|||
|
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
Further reading
|
|||
|
---------------
|
|||
|
|
|||
|
|
|||
|
Initial Google Summer of Code (2020) project proposal:
|
|||
|
|
|||
|
- https://hackmd.io/aYauztkGR0iOIoh8fJLecw
|
|||
|
|
|||
|
|
|||
|
|
|||
|
Previous design document:
|
|||
|
|
|||
|
- https://gist.github.com/WillyPillow/b8a643ddbd9235a97bc187e6e44b16e4
|
|||
|
|
|||
|
|
|||
|
|
|||
|
Discussion threads:
|
|||
|
|
|||
|
- 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
|
|||
|
|
|||
|
- https://groups.google.com/forum/#!topic/qubes-devel/pYHnihVCBM0
|
|||
|
|
|||
|
|