We integrate with a number of frameworks and libraries and have a test suite for each. The tests run against different versions of the framework/library to make sure we support everything we claim to.
This populate_tox.py script is responsible for picking reasonable versions to
test automatically and generating parts of tox.ini to capture this.
You require a free-threaded interpreter with pip installed to run the script. With
a recent version of uv you can directly run the script with the following
command:
uv run --python 3.14t \
--with pip \
--with-requirements scripts/populate_tox/requirements.txt \
--with-editable . \
python scripts/populate_tox/populate_tox.py
There is a template in this directory called tox.jinja which contains a
combination of hardcoded and generated entries.
The populate_tox.py script fills out the auto-generated part of that template.
It does this by querying PyPI for each framework's package and its metadata and
then determining which versions it makes sense to test to get good coverage.
By default, the lowest supported and latest version of a framework are always tested, with a number of releases in between:
- If the package has majors, we pick the highest version of each major.
- If the package doesn't have multiple majors, we pick two versions in between lowest and highest.
Each test suite requires at least some configuration to be added to
TEST_SUITE_CONFIG in scripts/populate_tox/config.py. If you're adding a new
integration, check out the Add a new test suite section.
The TEST_SUITE_CONFIG dictionary in scripts/populate_tox/config.py defines,
for each integration test suite, the main package (framework, library) to test
with; any additional test dependencies, optionally gated behind specific
conditions; and optionally the Python versions to test on.
Constraints are defined using the format specified below. The following sections describe each key.
integration_name: {
"package": name_of_main_package_on_pypi,
"deps": {
rule1: [package1, package2, ...],
rule2: [package3, package4, ...],
},
"python": python_version_specifier | dict[package_version_specifier, python_version_specifier],
"include": package_version_specifier,
"integration_name": integration_name,
"num_versions": int,
}
When talking about version specifiers, we mean version specifiers as defined by the Python Packaging Authority. See also the actual implementation in packaging.specifiers.
The name of the third-party package as it's listed on PyPI. The script will be picking different versions of this package to test.
This key is mandatory.
The test dependencies of the test suite. They're defined as a dictionary of
rule: [package1, package2, ...] key-value pairs. All packages
in the package list of a rule will be installed as long as the rule applies.
Each rule must be one of the following:
*: packages will be always installed- a version specifier on the main package (e.g.
<=0.32): packages will only be installed if the main package falls into the version bounds specified - specific Python version(s) in the form
py3.8,py3.9: packages will only be installed if the Python version matches one from the list
Rules can be used to specify version bounds on older versions of the main package's dependencies, for example. If Flask tests generally need Werkzeug and don't care about its version, but Flask older than 3.0 needs a specific Werkzeug version to work, you can say:
"flask": {
"deps": {
"*": ["Werkzeug"],
"<3.0": ["Werkzeug<2.1.0"],
},
...
}If you need to install a specific version of a secondary dependency on specific Python versions, you can say:
"celery": {
"deps": {
"*": ["newrelic", "redis"],
"py3.7": ["importlib-metadata<5.0"],
},
...
}This key is optional.
Sometimes, the whole test suite should only run on specific Python versions.
This can be achieved via the python key. There are two variants how to define
the Python versions to run the test suite on.
If you want the test suite to only be run on specific Python versions, you can
set python to a version specifier. For example, if you want AIOHTTP tests to
only run on Python 3.7+, you can say:
"aiohttp": {
"python": ">=3.7",
...
}If the Python version to use is dependent on the version of the package under
test, you can use the more expressive dictionary variant. For instance, while
HTTPX v0.28 supports Python 3.8, a test dependency of ours, pytest-httpx,
doesn't. If you want to specify that HTTPX test suite should not be run on
a Python version older than 3.9 if the HTTPX version is 0.28 or higher, you can
say:
"httpx": {
"python": {
# run the test suite for httpx v0.28+ on Python 3.9+ only
">=0.28": ">=3.9",
},
}The python key is optional, and when possible, it should be omitted. The script
should automatically detect which Python versions the package supports. However,
if a package has broken metadata or the SDK is explicitly not supporting some
packages on specific Python versions, the python key can be used.
Sometimes we only want to test specific versions of packages. For example, the Starlite package has two alpha prereleases of version 2.0.0, but we do not want to test these, since Starlite 2.0 was renamed to Litestar.
The value of the include key expects a version specifier defining which
versions should be considered for testing. For example, since we only want to test
versions below 2.x in Starlite, we can use:
"starlite": {
"include": "<2",
...
}The include key can also be used to exclude a set of specific versions by using
!= version specifiers. For example, the Starlite restriction above could equivalently
be expressed like so:
"starlite": {
"include": "!=2.0.0a1,!=2.0.0a2",
...
}Sometimes, the name of the test suite doesn't match the name of the integration.
For example, we have the openai-base and openai-notiktoken test suites, both
of which are actually testing the openai integration. If this is the case, you
can use the integration_name key to define the name of the integration. If not
provided, it will default to the name of the test suite.
Linking an integration to a test suite allows the script to access integration
configuration like, for example, the minimum supported version defined in
sentry_sdk/integrations/__init__.py.
With this option you can tweak the default version picking behavior by specifying
how many package versions should be tested. It accepts an integer equal to or
greater than 2, as the oldest and latest supported versions will always be
picked. Additionally, if there is a recent prerelease, it'll also always be
picked (this doesn't count towards num_versions).
For instance, num_versions set to 2 will only test the first supported and
the last release of the package. num_versions equal to 3 will test the first
supported, the last release, and one release in between; num_versions set to 4
will test an additional release in between. In all these cases, if there is
a recent prerelease, it'll be picked as well in addition to the picked versions.
- Add the minimum supported version of the framework/library to
_MIN_VERSIONSinintegrations/__init__.py. This should be the lowest version of the framework that we can guarantee works with the SDK. If you've just added the integration, you should generally set this to the latest version of the framework at the time, unless you've verified the integration works for earlier versions as well. - Add the integration and any constraints to
TEST_SUITE_CONFIG. See the Test suite config section for the format. - Add the integration to one of the groups in the
GROUPSdictionary inscripts/split_tox_gh_actions/split_tox_gh_actions.py. - Run
scripts/generate-test-files.shand commit the changes.