diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index 69c6f3d68523a2fbf36fd95709da0a6c349516d5..6a050cb81ce537763517ed9062fa6b3f48763634 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -157,27 +157,27 @@ variables: - bash ./.ci/test_template.sh -test:3.5: +test:3.6: stage: test - image: pauldmccarthy/fsleyes-py35-wxpy4-gtk2 + image: pauldmccarthy/fsleyes-py36-wxpy4-gtk3 <<: *test_template -test:3.6: +test:3.7: stage: test - image: pauldmccarthy/fsleyes-py36-wxpy4-gtk2 + image: pauldmccarthy/fsleyes-py37-wxpy4-gtk3 <<: *test_template -test:3.7: +test:3.8: stage: test - image: pauldmccarthy/fsleyes-py37-wxpy4-gtk2 + image: pauldmccarthy/fsleyes-py38-wxpy4-gtk3 <<: *test_template test:build-pypi-dist: stage: test - image: pauldmccarthy/fsleyes-py35-wxpy4-gtk2 + image: pauldmccarthy/fsleyes-py36-wxpy4-gtk3 <<: *except_releases tags: @@ -195,7 +195,7 @@ test:build-pypi-dist: style: stage: style - image: pauldmccarthy/fsleyes-py35-wxpy4-gtk2 + image: pauldmccarthy/fsleyes-py36-wxpy4-gtk3 <<: *test_template variables: TEST_STYLE: "true" diff --git a/CHANGELOG.rst b/CHANGELOG.rst index 00de568e586038893e99e67e56a50339528f1c45..b28604e356097583239b9a91e29561b062017a2e 100644 --- a/CHANGELOG.rst +++ b/CHANGELOG.rst @@ -2,6 +2,18 @@ This document contains the ``fslpy`` release history in reverse chronological order. +2.6.1 (Thursday 19th September 2019) +------------------------------------ + + +Changed +^^^^^^^ + + +* ``fslpy`` is no longer tested against Python 3.5, and is now tested against + Python 3.6, 3.7, and 3.8. + + 2.6.0 (Tuesday 10th September 2019) ----------------------------------- diff --git a/fsl/utils/filetree/parse.py b/fsl/utils/filetree/parse.py index 409a358bfdf9b9d707bc838b186742366054d51e..acfa68d17b4b3104e6de1f71d4307625bf1001f9 100644 --- a/fsl/utils/filetree/parse.py +++ b/fsl/utils/filetree/parse.py @@ -77,7 +77,7 @@ def read_line(line: str) -> Tuple[int, PurePath, str]: """ Parses line from the tree file - :param line: input line from a \*.tree file + :param line: input line from a ``*.tree`` file :return: Tuple with: - number of spaces in front of the name @@ -105,7 +105,7 @@ def read_subtree_line(line: str, directory: str) -> Tuple[int, "filetree.FileTre """ Parses the line defining a sub_tree - :param line: input line from a \*.tree file + :param line: input line from a ``*.tree`` file :param directory: containing directory :return: Tuple with diff --git a/fsl/utils/filetree/query.py b/fsl/utils/filetree/query.py index ac66091cd60c9fe273b0d939cfd20c3338bbeb7c..6407b14e3a39a2c59bb12502f21f6dc0512f0eed 100644 --- a/fsl/utils/filetree/query.py +++ b/fsl/utils/filetree/query.py @@ -48,9 +48,8 @@ class FileTreeQuery(object): short name that are present. The :meth:`query` method can be used to retrieve files which match a specific template, and variable values. - The :meth:`query` method returns a multi-dimensional ``numpy.array`` - which contains :class:`Match` objects, where each dimension one - represents variable for the template in question. + The :meth:`query` method returns a collection of :class:`Match` objects, + each of which represents one file which matches the query. Example usage:: diff --git a/fsl/utils/filetree/utils.py b/fsl/utils/filetree/utils.py index 0b82929491c9a8994d252f24704f66dff87f8411..11464e93570660dd27e1fa9103a3b4ff3e731f03 100644 --- a/fsl/utils/filetree/utils.py +++ b/fsl/utils/filetree/utils.py @@ -98,7 +98,7 @@ def resolve_optionals(text): else: return '' - res = [resolve_single_optional(text) for text in re.split('(\[.*?\])', text)] + res = [resolve_single_optional(text) for text in re.split(r'(\[.*?\])', text)] return ''.join(res) @@ -109,7 +109,7 @@ def find_variables(template): :param template: full template :return: sequence of variables """ - return tuple(var.split(':')[0] for var in re.findall("\{(.*?)\}", template)) + return tuple(var.split(':')[0] for var in re.findall(r"\{(.*?)\}", template)) def optional_variables(template): @@ -121,7 +121,7 @@ def optional_variables(template): """ include = set() exclude = set() - for text in re.split('(\[.*?\])', template): + for text in re.split(r'(\[.*?\])', template): if len(text) == 0: continue variables = find_variables(text) @@ -152,13 +152,13 @@ def extract_variables(template, filename, known_vars=None): sub_re = resolve_optionals(fill_known( template, dict( - **{var: '(\S+)' for k, var in zip(keep, optional) if k}, - **{var: '(\S+)' for var in remaining.difference(optional)} + **{var: r'(\S+)' for k, var in zip(keep, optional) if k}, + **{var: r'(\S+)' for var in remaining.difference(optional)} ) )) while '//' in sub_re: sub_re = sub_re.replace('//', '/') - sub_re = sub_re.replace('.', '\.') + sub_re = sub_re.replace('.', r'\.') if re.match(sub_re, filename) is None: continue diff --git a/fsl/utils/idle.py b/fsl/utils/idle.py index c1513708ebb401227dd1d0adb7cf9e0efd067631..f2fc9d0d2fcba36b3e0851b802d6b498ae908d67 100644 --- a/fsl/utils/idle.py +++ b/fsl/utils/idle.py @@ -86,7 +86,7 @@ import atexit import logging import functools import threading -import collections +from collections import abc try: import queue except ImportError: import Queue as queue @@ -610,7 +610,7 @@ def wait(threads, task, *args, **kwargs): direct = kwargs.pop('wait_direct', False) - if not isinstance(threads, collections.Sequence): + if not isinstance(threads, abc.Sequence): threads = [threads] haveWX = fslplatform.haveGui diff --git a/setup.cfg b/setup.cfg index 3f611406bec88d7cf6c581943a063647ee3b009e..9ea1ddbc4fea695ac2e0746889c32e4e510c7ce7 100644 --- a/setup.cfg +++ b/setup.cfg @@ -5,21 +5,21 @@ test=pytest universal=1 [tool:pytest] -# Available test markers: -# - fsltest: Requires FSL -# - wxtest: Requires wxPython -# - dicomtest: Requires dcm2niix -# - meshtest: Requires trimesh and rtree -# - igziptest: Requires indexed_gzip -# - piltest: Requires Pillow -# - noroottest: Need to be executed as -# non-root user (will fail -# otherwise) -# - longtest: Takes a long time -# +markers = + fsltest: Requires FSL + wxtest: Requires wxPython + dicomtest: Requires dcm2niix + meshtest: Requires trimesh and rtree + igziptest: Requires indexed_gzip + piltest: Requires Pillow + noroottest: Need to be executed as non-root user (will fail otherwise) + longtest: Takes a long time + unixtest: Only works on *nix systems + testpaths = tests addopts = -v --niters=50 --cov=fsl -m "not longtest" + [flake8] ignore = E127,E201,E203,E221,E222,E241,E271,E272,E301,E302,E303,E701,W504 \ No newline at end of file diff --git a/tests/test_image_resample.py b/tests/test_image_resample.py index 8fb488760ed74a14a0714c92d7815ac2a87f43f6..6961e7aa71c63a09d21f93ca5e128d8622b678a9 100644 --- a/tests/test_image_resample.py +++ b/tests/test_image_resample.py @@ -9,13 +9,13 @@ import pytest import scipy.ndimage as ndimage import fsl.data.image as fslimage -import fsl.utils.transform as transform +import fsl.transform.affine as affine import fsl.utils.image.resample as resample from . import make_random_image def random_affine(): - return transform.compose( + return affine.compose( 0.25 + 4.75 * np.random.random(3), -50 + 100 * np.random.random(3), -np.pi + 2 * np.pi * np.random.random(3)) @@ -61,9 +61,9 @@ def test_resample(seed): resx, resy, resz = restestcoords.T resvals = resampled[resx, resy, resz] - res2orig = transform.concat(img.worldToVoxMat, xf) + res2orig = affine.concat(img.worldToVoxMat, xf) - origtestcoords = transform.transform(restestcoords, res2orig) + origtestcoords = affine.transform(restestcoords, res2orig) # remove any coordinates which are out of # bounds in the original image space, or @@ -130,8 +130,8 @@ def test_resample_origin(seed): shape = np.random.randint(5, 50, 3) res = resample.resample(img, shape, origin='corner') res = fslimage.Image(res[0], xform=res[1]) - imgb = transform.axisBounds(img.shape, img.voxToWorldMat) - resb = transform.axisBounds(res.shape, res.voxToWorldMat) + imgb = affine.axisBounds(img.shape, img.voxToWorldMat) + resb = affine.axisBounds(res.shape, res.voxToWorldMat) assert np.all(np.isclose(imgb, resb, rtol=1e-5, atol=1e-5)) # with origin='centre' image @@ -142,8 +142,8 @@ def test_resample_origin(seed): res = resample.resample(img, shape, origin='centre') res = fslimage.Image(res[0], xform=res[1]) off = (np.array(img.shape) / np.array(res.shape) - 1) / 2 - imgb = np.array(transform.axisBounds(img.shape, img.voxToWorldMat)) - resb = np.array(transform.axisBounds(res.shape, res.voxToWorldMat)) + imgb = np.array(affine.axisBounds(img.shape, img.voxToWorldMat)) + resb = np.array(affine.axisBounds(res.shape, res.voxToWorldMat)) assert np.all(np.isclose(imgb, resb + off, rtol=1e-5, atol=1e-5)) # with origin='corner', using @@ -165,7 +165,7 @@ def test_resample_origin(seed): def test_resampleToPixdims(): img = fslimage.Image(make_random_image(dims=(10, 10, 10))) - imglo, imghi = transform.axisBounds(img.shape, img.voxToWorldMat) + imglo, imghi = affine.axisBounds(img.shape, img.voxToWorldMat) oldpix = np.array(img.pixdim, dtype=np.float) oldshape = np.array(img.shape, dtype=np.float) @@ -177,7 +177,7 @@ def test_resampleToPixdims(): res = resample.resampleToPixdims(img, newpix, origin=origin) res = fslimage.Image(res[0], xform=res[1]) - reslo, reshi = transform.axisBounds(res.shape, res.voxToWorldMat) + reslo, reshi = affine.axisBounds(res.shape, res.voxToWorldMat) resfov = reshi - reslo expfov = newpix * res.shape @@ -219,7 +219,7 @@ def test_resampleToReference2(): img[1, 1, 1] = 1 img = fslimage.Image(img) - refv2w = transform.scaleOffsetXform([1, 1, 1], [-1, -1, -1]) + refv2w = affine.scaleOffsetXform([1, 1, 1], [-1, -1, -1]) ref = np.zeros((5, 5, 5), dtype=np.float) ref = fslimage.Image(ref, xform=refv2w) res = resample.resampleToReference(img, ref, order=0) @@ -235,7 +235,7 @@ def test_resampleToReference3(): # Test resampling image to ref # with mismatched dimensions imgdata = np.random.randint(0, 65536, (5, 5, 5)) - img = fslimage.Image(imgdata, xform=transform.scaleOffsetXform( + img = fslimage.Image(imgdata, xform=affine.scaleOffsetXform( (2, 2, 2), (0.5, 0.5, 0.5))) # reference/expected data when @@ -270,7 +270,7 @@ def test_resampleToReference4(): # the image and ref are out of # alignment, but this affine # will bring them into alignment - img2ref = transform.scaleOffsetXform([2, 2, 2], [10, 10, 10]) + img2ref = affine.scaleOffsetXform([2, 2, 2], [10, 10, 10]) imgdata = np.random.randint(0, 65536, (5, 5, 5)) refdata = np.zeros((5, 5, 5)) diff --git a/tests/test_scripts/test_atlasq_query.py b/tests/test_scripts/test_atlasq_query.py index 5b2ab4e9da7a634be84c3aea0fa5f8d901a1fea0..14c61b0ebbb80f723075a01eb0be50ab596b8c82 100644 --- a/tests/test_scripts/test_atlasq_query.py +++ b/tests/test_scripts/test_atlasq_query.py @@ -286,7 +286,7 @@ def _eval_coord_voxel_query( else: exp = [q_type, squery] - _stdout = re.sub('\s+', ' ', stdout).strip() + _stdout = re.sub(r'\s+', ' ', stdout).strip() assert _stdout == ' '.join(exp) if isinstance(a_img, fslatlases.LabelAtlas): @@ -388,7 +388,7 @@ def _eval_mask_query( if expprop > 0: exp.append('{} {:0.4f}'.format(explabel, expprop)) - _stdout = re.sub('\s+', ' ', stdout).strip() + _stdout = re.sub(r'\s+', ' ', stdout).strip() assert _stdout == ' '.join(exp) if isinstance(aimg, fslatlases.LabelAtlas):