Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

chore: Drop Python 3.8, fix issues with CI #2566

Open
wants to merge 16 commits into
base: main
Choose a base branch
from
Open
10 changes: 1 addition & 9 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ jobs:
strategy:
matrix:
os: [ubuntu-latest]
python-version: ['3.8', '3.9', '3.10', '3.11', '3.12']
python-version: ['3.9', '3.10', '3.11', '3.12']
include:
- os: macos-latest
python-version: '3.12'
Expand All @@ -47,18 +47,10 @@ jobs:
python-version: ${{ matrix.python-version }}

- name: Install dependencies
if: matrix.python-version != '3.8'
run: |
python -m pip install uv
uv pip install --system --upgrade ".[all,test]"

# c.f. https://github.com/astral-sh/uv/issues/2062
- name: Install dependencies (Python 3.8)
if: matrix.python-version == '3.8'
run: |
python -m pip install --upgrade pip
python -m pip install --upgrade ".[all,test]"

- name: List installed Python packages
run: python -m pip list

Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/lower-bound-requirements.yml
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ jobs:
matrix:
os: [ubuntu-latest]
# minimum supported Python
python-version: ['3.8']
python-version: ['3.9']

steps:
- uses: actions/checkout@v4
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/release_tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ jobs:
strategy:
matrix:
os: [ubuntu-latest]
python-version: ['3.8', '3.9', '3.10', '3.11', '3.12']
python-version: ['3.9', '3.10', '3.11', '3.12']
include:
- os: macos-latest
python-version: '3.12'
Expand Down
5 changes: 3 additions & 2 deletions .pre-commit-config.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -55,14 +55,15 @@ repos:
- repo: https://github.com/pre-commit/mirrors-mypy
rev: v1.13.0
# check the oldest and newest supported Pythons
# except skip python 3.9 for numpy, due to poor typing
hooks:
- &mypy
id: mypy
name: mypy with Python 3.8
name: mypy with Python 3.10
files: src
additional_dependencies:
['numpy', 'types-tqdm', 'click', 'types-jsonpatch', 'types-pyyaml', 'types-jsonschema', 'importlib_metadata', 'packaging']
args: ["--python-version=3.8"]
args: ["--python-version=3.10"]
- <<: *mypy
name: mypy with Python 3.12
args: ["--python-version=3.12"]
Expand Down
9 changes: 9 additions & 0 deletions docs/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -529,6 +529,15 @@ def setup(app):
# and ReadTheDocs need to reference them
r'https://github.com/scikit-hep/pyhf/releases/tag/.*',
r'https://pyhf.readthedocs.io/en/.*',
# the following are 403s as they map to journals.aps.org or academic.oup.com
r'https://doi.org/10.1093/ptep/ptad144',
r'https://doi.org/10.1103/PhysRevD.104.055017',
r'https://doi.org/10.1103/PhysRevD.107.095021',
r'https://doi.org/10.1103/PhysRevD.108.016002',
r'https://doi.org/10.1103/PhysRevD.106.032005',
r'https://doi.org/10.1103/PhysRevLett.127.181802',
r'https://doi.org/10.1103/PhysRevLett.130.231801',
r'https://doi.org/10.1103/PhysRevLett.131.211802',
]
linkcheck_retries = 50

Expand Down
16 changes: 4 additions & 12 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ dynamic = ["version"]
description = "pure-Python HistFactory implementation with tensors and autodiff"
readme = "README.rst"
license = { text = "Apache-2.0" } # SPDX short identifier
requires-python = ">=3.8"
requires-python = ">=3.9"
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm personally fine with jumping up to Python 3.9. I think though as the 0.7.6 release is still on Python 3.7 that regardless of features we should make a 0.8.0 release after this that is the same as what we have in the 0.7.x series but just with non-EOL Python support. That will make multiple things easier to maintain and Python 3.9 is not new at all so I don't see that as an issue.

authors = [
{ name = "Lukas Heinrich", email = "[email protected]" },
{ name = "Matthew Feickert", email = "[email protected]" },
Expand All @@ -35,7 +35,6 @@ classifiers = [
"Operating System :: OS Independent",
"Programming Language :: Python :: 3",
"Programming Language :: Python :: 3 :: Only",
"Programming Language :: Python :: 3.8",
"Programming Language :: Python :: 3.9",
"Programming Language :: Python :: 3.10",
"Programming Language :: Python :: 3.11",
Expand Down Expand Up @@ -68,18 +67,10 @@ Homepage = "https://github.com/scikit-hep/pyhf"

[project.optional-dependencies]
shellcomplete = ["click_completion"]
# TODO: 'tensorflow' supports all platform_machine for tensorflow v2.16.1+
# but TensorFlow only supports python_version 3.8 up through tensorflow v2.13.1.
# So until Python 3.8 support is dropped, split requirments on python_version
# before and after 3.9.
# NOTE: macos x86 support is deprecated from tensorflow v2.17.0 onwards.
tensorflow = [
# python == 3.8
"tensorflow>=2.7.0; python_version < '3.9' and platform_machine != 'arm64'", # c.f. PR #1962, #2452
"tensorflow-macos>=2.7.0; python_version < '3.9' and platform_machine == 'arm64' and platform_system == 'Darwin'", # c.f. PR #2119, #2452
"tensorflow-probability>=0.11.0; python_version < '3.9'", # c.f. PR #1657, #2452
# python >= 3.9
"tensorflow-probability[tf]>=0.24.0; python_version >= '3.9'" # c.f. PR #2452
# see https://github.com/tensorflow/probability/issues/1854
"tensorflow-probability[tf]>=0.24.0,!=0.25.0" # c.f. PR #2452
]
torch = ["torch>=1.10.0"] # c.f. PR #1657
jax = [
Expand Down Expand Up @@ -251,6 +242,7 @@ warn_unused_configs = true
strict = true
enable_error_code = ["ignore-without-code", "redundant-expr", "truthy-bool"]
warn_unreachable = true
plugins = "numpy.typing.mypy_plugin"

[[tool.mypy.overrides]]
module = [
Expand Down
3 changes: 2 additions & 1 deletion src/pyhf/cli/infer.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,8 @@
from pyhf.infer import hypotest
from pyhf.infer import mle
from pyhf.workspace import Workspace
from pyhf import get_backend, set_backend, optimize
from pyhf.tensor.manager import get_backend, set_backend
from pyhf import optimize

log = logging.getLogger(__name__)

Expand Down
21 changes: 7 additions & 14 deletions src/pyhf/contrib/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -80,13 +80,7 @@
with open(output_directory, "wb") as archive:
archive.write(response.content)
else:
# Support for file-like objects for tarfile.is_tarfile was added
# in Python 3.9, so as pyhf is currently Python 3.8+ then can't
# do tarfile.is_tarfile(BytesIO(response.content)).
# Instead, just use a 'try except' block to determine if the
# archive is a valid tarfile.
# TODO: Simplify after pyhf is Python 3.9+ only
try:
if tarfile.is_tarfile(BytesIO(response.content)):
# Use transparent compression to allow for .tar or .tar.gz
with tarfile.open(
mode="r:*", fileobj=BytesIO(response.content)
Expand All @@ -97,13 +91,7 @@
archive.extractall(output_directory, filter="data")
else:
archive.extractall(output_directory)
except tarfile.ReadError:
if not zipfile.is_zipfile(BytesIO(response.content)):
raise exceptions.InvalidArchive(
f"The archive downloaded from {archive_url} is not a tarfile"
+ " or a zipfile and so can not be opened as one."
)

elif zipfile.is_zipfile(BytesIO(response.content)):
output_directory = Path(output_directory)
if output_directory.exists():
rmtree(output_directory)
Expand All @@ -129,6 +117,11 @@
# from creation time
rmtree(output_directory)
_tmp_path.replace(output_directory)
else:
raise exceptions.InvalidArchive(

Check warning on line 121 in src/pyhf/contrib/utils.py

View check run for this annotation

Codecov / codecov/patch

src/pyhf/contrib/utils.py#L121

Added line #L121 was not covered by tests
f"The archive downloaded from {archive_url} is not a tarfile"
+ " or a zipfile and so can not be opened as one."
)

except ModuleNotFoundError:
log.error(
Expand Down
2 changes: 1 addition & 1 deletion src/pyhf/infer/__init__.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
"""Inference for Statistical Models."""

from pyhf.infer import utils
from pyhf import get_backend
from pyhf.tensor.manager import get_backend
from pyhf import exceptions


Expand Down
6 changes: 3 additions & 3 deletions src/pyhf/infer/calculators.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
"""

from pyhf.infer.mle import fixed_poi_fit
from pyhf import get_backend
from pyhf.tensor.manager import get_backend
from pyhf.infer import utils
import tqdm

Expand Down Expand Up @@ -120,7 +120,7 @@ def cdf(self, value):
>>> import pyhf
>>> pyhf.set_backend("numpy")
>>> bkg_dist = pyhf.infer.calculators.AsymptoticTestStatDistribution(0.0)
>>> bkg_dist.cdf(0.0)
>>> print(bkg_dist.cdf(0.0))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Did something change RE: the way doctest works where the prints are now necessary? We had always treated this like REPL behavior, not script behavior.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No, numpy changed. ubuntu-latest has numpy 2.x which gives np.float64(0.0) while macos-13 uses numpy 1.x which gives 0.0 -- we can't write a test that relies on repr for two diff major versions of numpy.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

>>> import numpy as np
>>> np.__version__
'2.0.2'
>>> np.float64(0.0)
np.float64(0.0)
>>> print(np.float64(0.0))
0.0

and

>>> import numpy as np
>>> np.__version__
'1.26.4'
>>> np.float64(0.0)
0.0
>>> print(np.float64(0.0))
0.0

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we can't write a test that relies on repr for two diff major versions of numpy.

I understand what you mean, but I thought that we already did have ways to indicate to doc tests that it should only test on certain operating systems as we would already get different versions.

Let me take the afternoon today (once I'm free from running interviews for a search I'm on) to play with this and if I can't get it then we can just accept it and move on.

0.5

Args:
Expand Down Expand Up @@ -619,7 +619,7 @@ def expected_value(self, nsigma):
>>> normal = pyhf.probability.Normal(mean, std)
>>> samples = normal.sample((100,))
>>> dist = pyhf.infer.calculators.EmpiricalDistribution(samples)
>>> dist.expected_value(nsigma=1)
>>> print(dist.expected_value(nsigma=1))
6.15094381...

>>> import pyhf
Expand Down
2 changes: 1 addition & 1 deletion src/pyhf/infer/intervals/upper_limits.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
import numpy as np
from scipy.optimize import toms748

from pyhf import get_backend
from pyhf.tensor.manager import get_backend
from pyhf.infer import hypotest

__all__ = ["linear_grid_scan", "toms748_scan", "upper_limit"]
Expand Down
2 changes: 1 addition & 1 deletion src/pyhf/infer/mle.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
"""Module for Maximum Likelihood Estimation."""

from pyhf import get_backend
from pyhf.tensor.manager import get_backend
from pyhf.exceptions import UnspecifiedPOI

__all__ = ["fit", "fixed_poi_fit", "twice_nll"]
Expand Down
2 changes: 1 addition & 1 deletion src/pyhf/infer/test_statistics.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
from pyhf import get_backend
from pyhf.tensor.manager import get_backend
from pyhf.infer.mle import fixed_poi_fit, fit
from pyhf.exceptions import UnspecifiedPOI

Expand Down
3 changes: 2 additions & 1 deletion src/pyhf/mixins.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
from __future__ import annotations

import logging
from typing import Any, Sequence
from typing import Any
from collections.abc import Sequence

from pyhf.typing import Channel

Expand Down
13 changes: 9 additions & 4 deletions src/pyhf/modifiers/histosys.py
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,8 @@ class histosys_combined:
def __init__(
self, modifiers, pdfconfig, builder_data, interpcode='code0', batch_size=None
):
default_backend = pyhf.default_backend
Copy link
Member

@matthewfeickert matthewfeickert Jan 9, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this needed just to deal with the PyTorch warnings from Issue #2554?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it's needed in order to type _histosys_mask correctly as a tensor (see the below lines where it is being used). You could drop it, but then the typehints won't work due to the changes in typing in numpy.


self.batch_size = batch_size
self.interpcode = interpcode
assert self.interpcode in ['code0', 'code2', 'code4p']
Expand Down Expand Up @@ -128,10 +130,13 @@ def __init__(
]
for m in keys
]
self._histosys_mask = [
[[builder_data[m][s]['data']['mask']] for s in pdfconfig.samples]
for m in keys
]
self._histosys_mask = default_backend.astensor(
[
[[builder_data[m][s]['data']['mask']] for s in pdfconfig.samples]
for m in keys
],
dtype='bool',
)

if histosys_mods:
self.interpolator = getattr(interpolators, self.interpcode)(
Expand Down
17 changes: 12 additions & 5 deletions src/pyhf/modifiers/lumi.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
import logging

from pyhf import get_backend, events
import pyhf
from pyhf.tensor.manager import get_backend
from pyhf import events
from pyhf.parameters import ParamViewer

log = logging.getLogger(__name__)
Expand Down Expand Up @@ -60,6 +62,8 @@ class lumi_combined:
op_code = 'multiplication'

def __init__(self, modifiers, pdfconfig, builder_data, batch_size=None):
default_backend = pyhf.default_backend

self.batch_size = batch_size

keys = [f'{mtype}/{m}' for m, mtype in modifiers]
Expand All @@ -72,10 +76,13 @@ def __init__(self, modifiers, pdfconfig, builder_data, batch_size=None):
)
self.param_viewer = ParamViewer(parfield_shape, pdfconfig.par_map, lumi_mods)

self._lumi_mask = [
[[builder_data[m][s]['data']['mask']] for s in pdfconfig.samples]
for m in keys
]
self._lumi_mask = default_backend.astensor(
[
[[builder_data[m][s]['data']['mask']] for s in pdfconfig.samples]
for m in keys
],
dtype='bool',
)
self._precompute()
events.subscribe('tensorlib_changed')(self._precompute)

Expand Down
17 changes: 12 additions & 5 deletions src/pyhf/modifiers/normfactor.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
import logging

from pyhf import get_backend, events
import pyhf
from pyhf.tensor.manager import get_backend
from pyhf import events
from pyhf.parameters import ParamViewer

log = logging.getLogger(__name__)
Expand Down Expand Up @@ -58,6 +60,8 @@ class normfactor_combined:
op_code = 'multiplication'

def __init__(self, modifiers, pdfconfig, builder_data, batch_size=None):
default_backend = pyhf.default_backend

self.batch_size = batch_size

keys = [f'{mtype}/{m}' for m, mtype in modifiers]
Expand All @@ -72,10 +76,13 @@ def __init__(self, modifiers, pdfconfig, builder_data, batch_size=None):
parfield_shape, pdfconfig.par_map, normfactor_mods
)

self._normfactor_mask = [
[[builder_data[m][s]['data']['mask']] for s in pdfconfig.samples]
for m in keys
]
self._normfactor_mask = default_backend.astensor(
[
[[builder_data[m][s]['data']['mask']] for s in pdfconfig.samples]
for m in keys
],
dtype='bool',
)
self._precompute()
events.subscribe('tensorlib_changed')(self._precompute)

Expand Down
17 changes: 12 additions & 5 deletions src/pyhf/modifiers/normsys.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
import logging

from pyhf import get_backend, events
import pyhf
from pyhf.tensor.manager import get_backend
from pyhf import events
from pyhf import interpolators
from pyhf.parameters import ParamViewer

Expand Down Expand Up @@ -72,6 +74,8 @@ class normsys_combined:
def __init__(
self, modifiers, pdfconfig, builder_data, interpcode='code1', batch_size=None
):
default_backend = pyhf.default_backend

self.interpcode = interpcode
assert self.interpcode in ['code1', 'code4']

Expand All @@ -97,10 +101,13 @@ def __init__(
]
for m in keys
]
self._normsys_mask = [
[[builder_data[m][s]['data']['mask']] for s in pdfconfig.samples]
for m in keys
]
self._normsys_mask = default_backend.astensor(
[
[[builder_data[m][s]['data']['mask']] for s in pdfconfig.samples]
for m in keys
],
dtype='bool',
)

if normsys_mods:
self.interpolator = getattr(interpolators, self.interpcode)(
Expand Down
Loading
Loading