Development environment
April 27, 2026 ยท View on GitHub
Debian/Ubuntu
Install dependencies:
:memo: Expand this section if you are on Ubuntu 22.04 (Jammy).The
The |
sudo apt install -y podman dh-python build-essential make libqt6gui6 \
pipx python3 python3-dev
Install Poetry using pipx (recommended) and add it to your $PATH:
(See also a list of alternative installation methods)
pipx ensurepath
pipx install poetry
pipx inject poetry poetry-plugin-export
After this, restart the terminal window, for the poetry command to be in your
$PATH.
Clone this repository:
git clone https://github.com/freedomofpress/dangerzone/
Change to the dangerzone folder, and install the poetry dependencies:
Note: due to an issue with poetry, if it prompts for your keyring, disable the keyring with
keyring --disableand run the command again.
cd dangerzone
poetry install
Dangerzone depends on some assets that should be downloaded in order to run (think binaries and others resources). This can be done with the following command:
poetry run mazette install
Run the following command to download the latest container image, or build it locally:
poetry run dangerzone-image prepare-archive --output share/container.tar
Run from source tree:
# run the CLI
poetry run dangerzone-cli --help
# run the GUI
poetry run dangerzone
Create a .deb:
./install/linux/build-deb.py
Fedora
Install dependencies:
sudo dnf install -y rpm-build podman python3 python3-devel python3-poetry-core \
pipx qt6-qtbase-gui
Install Poetry using pipx:
pipx install poetry
pipx inject poetry
Clone this repository:
git clone https://github.com/freedomofpress/dangerzone/
Change to the dangerzone folder, and install the poetry dependencies:
Note: due to an issue with poetry, if it prompts for your keyring, disable the keyring with
keyring --disableand run the command again.
cd dangerzone
poetry install
Dangerzone depends on some assets that should be downloaded in order to run (think binaries and others resources). This can be done with the following command:
poetry run mazette install
Run the following command to download the latest container image, or build it locally:
poetry run dangerzone-image prepare-archive --output share/container.tar
Run from source tree:
# run the CLI
poetry run dangerzone-cli --help
# run the GUI
poetry run dangerzone
Note
Prefer running the following command in a Fedora development environment,
created by ./dev_script/env.py.
Create a .rpm:
./install/linux/build-rpm.py
Qubes OS
:warning: Native Qubes support is in beta stage, so the instructions below require switching between qubes, and are subject to change.
If you want to build Dangerzone on Qubes and use containers instead of disposable qubes, please follow the instructions of Fedora / Debian instead.
Initial Setup
The following steps must be completed once. Make sure you run them in the specified qubes.
Overview of the qubes you'll create:
| qube | type | purpose |
|---|---|---|
| dz | app qube | Dangerzone development |
| dz-dvm | app qube | offline disposable template for performing conversions |
| fedora-43-dz | template | template for the other two qubes |
In dom0:
The following instructions require typing commands in a terminal in dom0.
-
Create a new Fedora template (
fedora-43-dz) for Dangerzone development:qvm-clone fedora-43 fedora-43-dz:bulb: Alternatively, you can use your base Fedora 42 template in the following instructions. In that case, skip this step and replace
fedora-43-dzwithfedora-43in the steps below. -
Create an offline disposable template (app qube) called
dz-dvm, based on thefedora-43-dztemplate. This will be the qube where the documents will be sanitized:qvm-create --class AppVM --label red --template fedora-43-dz \ --prop netvm="" --prop template_for_dispvms=True \ --prop default_dispvm='' dz-dvm -
Create an app qube (
dz) that will be used for Dangerzone development and initiating the sanitization process:qvm-create --class AppVM --label red --template fedora-43-dz dz qvm-volume resize dz:private $(numfmt --from=auto 20Gi):bulb: Alternatively, you can use a different app qube for Dangerzone development. In that case, replace
dzwith the qube of your choice in the steps below.In the commands above, we also resize the private volume of the
dzqube to 20GiB, since you may need some extra storage space when developing on Dangerzone (e.g., for container images, Tesseract data, and Python virtualenvs). -
Add an RPC policy (
/etc/qubes/policy.d/50-dangerzone.policy) that will allow launching a disposable qube (dz-dvm) when Dangerzone converts a document, with the following contents:dz.Convert * @anyvm @dispvm:dz-dvm allow dz.ConvertDev * @anyvm @dispvm:dz-dvm allow
In the dz app qube
In the following steps you'll setup the development environment and install a dangerzone build. This will make the development faster since it loads the server code dynamically each time it's run, instead of having to build and install a server package each time the developer wants to test it.
-
Follow the Fedora installation instructions up until
poetry run mazette install. -
Clone the Dangerzone image repo, which holds the server-side conversion component:
cd ~ git clone https://github.com/freedomofpress/dangerzone-image -
Install some extra build dependencies:
sudo dnf install -y python3-uv-build python3-pymupdf python3-magic -
Build the Dangerzone RPM packages for Qubes:
./dangerzone/install/linux/build-rpm.py --qubes ./dangerzone-image/qubes/build-rpm.sh -
Copy the produced
.rpmfiles intofedora-43-dz:qvm-copy ./dangerzone/dist/*.x86_64.rpm \ ./dangerzone-image/qubes/dist/*.noarch.rpm
In the fedora-43-dz template
-
Install the
.rpmpackages you just copied, in order to get the Dangerzone dependencies:sudo dnf install ~/QubesIncoming/dz/*.rpm -
Shutdown the
fedora-43-dztemplate.
Developing Dangerzone
From here on, developing Dangerzone is similar to Fedora. The only differences
are that you need to set the environment variable QUBES_CONVERSION=1 when
you wish to test the Qubes conversion, run the following commands on the dz development qube:
export DANGERZONE_DEV=1 QUBES_CONVERSION=1
# run the CLI
poetry run dangerzone-cli --help
# run the GUI
poetry run dangerzone
And when creating a .rpm you'll need to enable the --qubes flag.
Note
Prefer running the following command in a Fedora development environment,
created by ./dev_script/env.py.
./install/linux/build-rpm.py --qubes
Tip
For faster changes to the server side components, you can let Dangerzone know
about the location of dangerzone-image repo:
export DANGERZONE_INSECURE_CONVERTER_PATH=~/dangerzone-image/src
From there on, you can make changes in the dangerzone-image repo, and they
will be mirrored to the disposable qube through the dz.ConvertDev RPC call.
The only reason to build a new Qubes RPM and install it in the fedora-43-dz
template for development is if:
- The project requires new server-side components.
- The code for
qubes/dz.ConvertDevneeds to be updated.
macOS
Install the latest version of Python 3.13 from python.org, and make sure /Library/Frameworks/Python.framework/Versions/3.13/bin is in your PATH.
Clone this repository:
git clone https://github.com/freedomofpress/dangerzone/
cd dangerzone
Install Python dependencies:
python3 -m pip install poetry
poetry install
Install Homebrew dependencies:
brew install create-dmg
Dangerzone depends on some assets that should be downloaded in order to run (think binaries and others resources). This can be done with the following command:
poetry run mazette install
Run the following command to download the latest container image, or build it locally:
poetry run dangerzone-image prepare-archive --output share/container.tar
Run from source tree:
# run the CLI
poetry run dangerzone-cli --help
# run the GUI
poetry run dangerzone
To create an app bundle, use the build_app.py script:
poetry run ./install/macos/build-app.py
If you want to build for distribution, you'll need a codesigning certificate, and then run:
poetry run ./install/macos/build-app.py --with-codesign
The output is in the dist folder.
Windows
Install the latest version of Python 3.13 (64-bit) from python.org. Make sure to check the "Add Python 3.13 to PATH" checkbox on the first page of the installer.
Install Microsoft Visual C++ 14.0 or greater. Get it with "Microsoft C++ Build Tools" and make sure to select "Desktop development with C++" when installing.
Install poetry. Open PowerShell, and run:
python -m pip install poetry
Install git from here, open a Windows terminal (cmd.exe) and clone this repository:
git clone https://github.com/freedomofpress/dangerzone/
Change to the dangerzone folder, and install the poetry dependencies:
cd dangerzone
poetry install
Dangerzone depends on some assets that should be downloaded in order to run (think binaries and others resources). This can be done with the following command:
poetry run mazette install
Run the following command to download the latest container image, or build it locally:
poetry run dangerzone-image prepare-archive --output share/container.tar
After that you can launch dangerzone during development with:
# run the CLI
poetry run dangerzone-cli --help
# run the GUI
poetry run dangerzone
If you want to build the Windows installer
Install .NET SDK version 6 or later. Then, open a terminal and install the latest version of WiX Toolset .NET tool v5 with:
dotnet tool install --global wix --version 5.0.2
Install the WiX UI extension. You may need to open a new terminal in order to use the newly installed wix .NET tool:
wix extension add --global WixToolset.UI.wixext/5.0.2
Important
To avoid compatibility issues, ensure the WiX UI extension version matches the version of the WiX Toolset.
Run wix --version to check the version of WiX Toolset you have installed and replace 5.x.y with the full version number without the Git revision.
If you want to sign binaries with Authenticode
You'll need a code signing certificate.
To make a .exe
Open a command prompt, cd into the dangerzone directory, and run:
poetry run python .\setup-windows.py build
In build\exe.win32-3.13\ you will find dangerzone.exe, dangerzone-cli.exe, and all supporting files.
To build the installer
Note that you must have a codesigning certificate installed in order to use the install\windows\build-app.bat script, because it codesigns dangerzone.exe, dangerzone-cli.exe and Dangerzone.msi.
poetry run .\install\windows\build-app.bat
When you're done you will have dist\Dangerzone.msi.
Reproducing the container image
The Dangezone container image is bit-for-bit reproducible, meaning that anybody can ensure that the distributed images match the expected ones, avoiding supply-chain attacks. Read more on how to reproduce the images at docs/developer/reproducibility.md.
Building a local container image
It is possible to build the image locally for testing, without verifying its signatures against Sigstore:
python3 ./install/common/build-image.py
export DANGERZONE_BYPASS_SIG_CHECKS=1 # On Linux and macOS
set DANGERZONE_BYPASS_SIG_CHECKS 1 # On Windows
To switch back to the original behavior, remove the environment variable:
unset DANGERZONE_BYPASS_SIG_CHECKS # On Linux and macOS
set DANGERZONE_BYPASS_SIG_CHECKS= # On Windows