ruamel.yaml is a YAML 1.2 loader/dumper package for Python.
| version | 0.19.0 |
| updated | 2025-12-31 |
| documentation | https://yaml.dev/doc/ruamel.yaml |
| repository | https://sourceforge.net/projects/ruamel-yaml |
| pypi | https://pypi.org/project/ruamel.yaml |
If you are interested in future upgrades of ruamel.yaml
please check the documentation on installing,
since at some point I might not be able to upload a new version to PyPI with updated information.
ruamel.yaml was intentionally named as yaml in a namespace ruamel. The namespace allows the installation
name to correspond unchanged to how the package is imported, reduces the number of links I have to create
in site-packages of a Python install during development, as well as providing a recognisable set of packages
my company releases to the public.
However, after uploading version 0.18.7, I got an email from PyPI, about having to change the project name
to ruamel_yaml to comply with PEP 625, sometime in the future. The email doesn't say if namespace packages are
no longer allowed, or how to deal with the very real clash with the pre-existing package ruamel_yaml.
I might not be able to adapt ruamel.yaml, in
a way that does not negatively affect the 0.5 million daily downloads (and my own usage of the package) in time.
My experience with other such service downgrades (Bitbucket, Readthedocs), has not been entirely positive.
Starting with 0.19.0 ruamel.yaml no longer has ruamel.yaml.clib as a dependency,
as this has been replaced with ruamel.yaml.clibz.
The C sources are functionally unchanged,
but they are now always compiled (using setuptools-zig and ziglang) on your system,
instead of being downloaded as pre-compiled wheels (if available).
For this to function properly your Python (virtual) environment needs to have
an up-to-date version of setuptools and wheels pre-installed.
If you run into trouble, you can install ruamel.yaml.clib explicitly, without
using ruamel.yaml.clibz at all, using:
python -m pip install --no-deps ruamel.yaml ruamel.yaml.clib
The code to load ruamel.yaml.clib has priority over ruamel.yaml.clibz
if both are installed (so --no-deps is not strictly necessary.
This compatibility will at least be available during the 0.19 ruamel.yaml series
(so pin your usage of ruamel.yaml if necessary and report any problems).
The motivitation for this change is the availability,
and easy of use, of Zig as the toolchain
(in the form of ziglang on PyPi), so lenghty, non-optimized,
pre-compilation and uploading to PyPI,
is no longer necessary.
The time spent on creating ~60 wheels
and even more time wasted on dealing with CI providers (Appveyor
not being updated to support 3.14, Github CI being slow,
and charging for the use of your own computer,
etc).
The split out of ruamel.yaml.clib after the 0.15.100 release,
was also motivated by the time spent
on generating .whl files even if only Python code was changed.
The use of ziglang and setuptools-zig does make re-integration of the C sources
into ruamel.yaml feasable, but there are no plans yet to make this happen.
The test matrix for ruamel.yaml.clibz, of course still has many dimensions:
Python versions: 3.9 - 3.14
OS-es: Linux, Alpine (musl), macOS, Windows
Architectures: Intel/AMD, Arm (and others), in 64 and some also in 32 bit versions
Zig version: ziglang < 0.16 is taken from PyPI
I try to test as much of the combinations as possible,
trying at least all supported Python versions,
including freethreading,
on macOS-arm64,
Linux arm64 (via docker containers),
Ubuntu Linux-Intel,
Linux musl intel (docker).
And at least one Python version along each of the indicated positions of the
dimensions above (e.g. Windows10 64bit was tested with Python 3.14, but I
could not test the RISC-V architecture).
As with generating .whl files previously
(which I could not all test myself)
I partly have to rely on the process of compilation/generation being likely correct,
and feedback from actual users,
of exotic (for me) platforms, is of course welcome.
There is new section,
in the documentation,
on the security of processing unchecked input.
The potentially breaking change announced for the 0.18 series, in that YAML(typ='unsafe')
was going to be deprecated (now pending), has not yet been implemented, but is still considered.
If you only use unsafe to dump, please use the new YAML(typ='full'), the result of that can be safely
loaded with a default instance YAML(), as that will get you inspectable, tagged, scalars, instead of
executed Python functions/classes. (You should probably add constructors for what you actually need,
but I do consider adding a ruamel.yaml.unsafe package that will re-add the typ='unsafe' option.
Please adjust/pin your dependencies accordingly if necessary.
Version 0.18.16 was the last one tested to be working with Python 3.8.
Version 0.18.9 was the last one tested to be working with Python 3.7.
Version 0.17.21 was the last one tested to be working on Python 3.5 and 3.6.
The 0.16.13 release was the last that was tested to be working on Python 2.7.
There are two extra plug-in packages
(ruamel.yaml.bytes and ruamel.yaml.string)
for those not wanting to do the streaming to a
io.BytesIO/StringIO buffer themselves.
If your package uses ruamel.yaml and is not listed on PyPI, drop me an
email, preferably with some information on how you use the package (or a
link to the repository) and I'll keep you informed when the status of
the API is stable enough to make the transition.
For packaging purposes you can use a download of the tar balls of tagged source
0.19.0 (2025-12-31):
ruamel.yaml.clib to ruamel.yaml.clibz which includes support for free-threading (revisited after a bug report by Ahmed Moustafa and some insistance by Nathan Goldbaum.max_depth to YAML() instance. If set to a (positive) number this limits the recursion, so loading does throw a MaxDepthExceededError. Based on comments by Benjamin Oberdorfer via email. This also triggered the new documenation section on processing unchecked input.0.18.17 (2025-12-17):
_ruamel_yaml_clibz first.0.18.16 (2025-10-22):
0.18.15 (2025-08-19):
0.18.14 (2025-06-09):
0.18.13 (2025-06-06):
0.18.12 (2025-05-30):
0.18.11 (2025-05-19):
load_yaml_guess_indent now takes an option yaml argument so you can provide an already created/configured YAML instanceNotImplementedError. Reported by Karsten Tessarzik.0.18.10 (2025-01-06):
develop config)0.18.9 (2025-01-05):
0.18.8 (2025-01-02):
0.18.7 (2024-12-30):
0 (issue reported by (Mor Peled)[https://sourceforge.net/u/morp/profile/] and also in a question by Ravi on Stackoverflow)0.18.6 (2024-02-07):
InitVar fields when from __future__ import annotations was used to delay evaluation of typing.0.18.5 (2023-11-03):
0.18.4 (2023-11-01):
doc_infos attribute which is a cumulative list of DocInfo instances (one for load(), one per document for load_all()). DocInfo instances contain version information (requested, directive) and tag directive information0.18.3 (2023-10-29):
0.18.2 (2023-10-24):
AttributeError with the, somewhat more informative, orginal warning message. Instead of calling sys.exit(1)0.18.1 (2023-10-24):
0.18.0 (2023-10-23):
scan, parse, compose, load, emit, serialize, dump and their variants (_all, safe_, round_trip_, etc) have been deprecated (the same named methods on YAML() instances are, of course, still there.YAML(typ='unsafe') now issues a PendingDeprecationWarning. This will become deprecated in the 0.18 seriesYAML(typ='full') to dump unregistered Python classes/functions. bytes-like object is required not 'str' while dumping binary streams. This was reported, analysed and a fix provided by Vit ZikmundFor older changes see the file
CHANGES