From 1d444b67e90335b860430cba2c2360d09e8eff66 Mon Sep 17 00:00:00 2001 From: Ben Greiner Date: Fri, 30 Aug 2024 19:46:20 +0200 Subject: [PATCH 01/40] simplify assert all true --- tvtk/tests/test_array_handler.py | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/tvtk/tests/test_array_handler.py b/tvtk/tests/test_array_handler.py index 556ad0483..b34e30f6c 100644 --- a/tvtk/tests/test_array_handler.py +++ b/tvtk/tests/test_array_handler.py @@ -201,8 +201,7 @@ def test_arr2cell_array(self): cells = array_handler.array2vtkCellArray(a) arr = array_handler.vtk2array(cells.GetData()) expect = numpy.array([3, 0, 1, 2]*3, int) - self.assertEqual(numpy.alltrue(numpy.equal(arr, expect)), - True) + self.assertTrue(numpy.all(numpy.equal(arr, expect))) self.assertEqual(cells.GetNumberOfCells(), N) # Test if a list of Numeric arrays of different cell lengths works. @@ -210,8 +209,7 @@ def test_arr2cell_array(self): cells = array_handler.array2vtkCellArray(l_a) arr = array_handler.vtk2array(cells.GetData()) expect = numpy.array([1, 0]*3 + [3, 0, 1, 2]*3 + [2, 0,1]*2, int) - self.assertEqual(numpy.alltrue(numpy.equal(arr, expect)), - True) + self.assertTrue(numpy.all(numpy.equal(arr, expect))) self.assertEqual(cells.GetNumberOfCells(), N*2 + 2) # This should not take a long while. This merely tests if a From 294df464044b41a04605fa3def9e4fc5fd6ae176 Mon Sep 17 00:00:00 2001 From: Ben Greiner Date: Fri, 30 Aug 2024 20:17:37 +0200 Subject: [PATCH 02/40] remove numpy.sctypes --- tvtk/tests/test_array_handler.py | 17 ++++++++++------- 1 file changed, 10 insertions(+), 7 deletions(-) diff --git a/tvtk/tests/test_array_handler.py b/tvtk/tests/test_array_handler.py index b34e30f6c..96ba65278 100644 --- a/tvtk/tests/test_array_handler.py +++ b/tvtk/tests/test_array_handler.py @@ -160,13 +160,16 @@ def test_array2vtk(self): self.assertEqual(vtk_arr.GetValue(2), 0) self.assertEqual(vtk_arr.GetValue(3), 1) - # Make sure the code at least runs for all the non-complex - # numerical dtypes in numpy. - float_types = [x for x in numpy.sctypes['float'] - if x().dtype.name not in ('float16', 'float128')] - for dtype in (numpy.sctypes['int'] + numpy.sctypes['uint'] + - float_types): - array_handler.array2vtk(numpy.zeros((1,), dtype=dtype)) + # Make sure the code at least runs for all + # numerical dtypes in numpy + # except for half, longdouble and complexfloating + int_types = ['byte', 'short', 'intc', 'int_', 'long', 'longlong'] + uint_types = ['ubyte', 'ushort', 'uintc', 'uint', 'ulong', + 'ulonglong'] + float_types = ['single', 'double'] + for dtype in int_types + uint_types + float_types: + array_handler.array2vtk(numpy.zeros((1,), + dtype=numpy.dtype(dtype))) def test_arr2cell_array(self): """Test Numeric array to vtkCellArray conversion.""" From 423fb10b0f6d17920787291b689a6f0c412e91d0 Mon Sep 17 00:00:00 2001 From: Ben Greiner Date: Fri, 30 Aug 2024 20:25:17 +0200 Subject: [PATCH 03/40] replace complex_ --- mayavi/tools/data_wizards/loadtxt.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/mayavi/tools/data_wizards/loadtxt.py b/mayavi/tools/data_wizards/loadtxt.py index 80702b0e0..5278c368b 100644 --- a/mayavi/tools/data_wizards/loadtxt.py +++ b/mayavi/tools/data_wizards/loadtxt.py @@ -23,7 +23,7 @@ def _getconv(dtype): return lambda x: int(float(x)) elif issubclass(typ, np.floating): return float - elif issubclass(typ, np.complex_): + elif issubclass(typ, np.complex128): return complex else: return str From 7b7f8057d286c5c48413a6e308a063dd9d5f2e22 Mon Sep 17 00:00:00 2001 From: Ben Greiner Date: Fri, 30 Aug 2024 20:39:45 +0200 Subject: [PATCH 04/40] handle new numpy2 nan repr --- mayavi/tests/test_csv_sniff.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/mayavi/tests/test_csv_sniff.py b/mayavi/tests/test_csv_sniff.py index 30e2802d2..f50a573fe 100644 --- a/mayavi/tests/test_csv_sniff.py +++ b/mayavi/tests/test_csv_sniff.py @@ -12,7 +12,7 @@ import tempfile from unittest import SkipTest -from numpy import array, ndarray +from numpy import array, ndarray, isnan from mayavi.tools.data_wizards.csv_sniff import \ Sniff, loadtxt, loadtxt_unknown, array2dict @@ -33,8 +33,8 @@ def assertAllClose(self, x, y): def assertClose(self, a, b): if isinstance(a, (int, float)): - if repr(a) == 'nan': - self.assertTrue(repr(b) == 'nan') + if isnan(a): + self.assertTrue(isnan(b), '%r != %r' % (a ,b)) else: self.assertTrue(abs(a - b) < 1e-6 * max(1, abs(a)), '%r != %r %r' % (a, b, abs(a - b))) From 17e934442bcf070a479d0787e08123f913c1b24c Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 6 Sep 2024 12:28:18 -0400 Subject: [PATCH 05/40] FIX: configobj --- .github/workflows/run-mayavi-tests.yml | 2 +- .gitignore | 1 + mayavi/tools/data_wizards/loadtxt.py | 2 +- tvtk/code_gen.py | 4 ++-- tvtk/tests/test_array_ext.py | 6 +++++- tvtk/wrapper_gen.py | 14 +++++++------- 6 files changed, 17 insertions(+), 12 deletions(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 8d4553fca..2539e32e7 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -73,7 +73,7 @@ jobs: run: | set -exo pipefail python -m pip install --upgrade pip setuptools wheel - python -m pip install --upgrade "${{ matrix.qt-api }}" numpy "${{ matrix.vtk }}" pillow pytest traits traitsui + python -m pip install --upgrade "${{ matrix.qt-api }}" numpy "${{ matrix.vtk }}" pillow pytest traits traitsui configobj - name: Install mayavi and tvtk run: python -um pip install -ve .[app] - name: Test Mayavi package diff --git a/.gitignore b/.gitignore index a57949113..3b903dad4 100644 --- a/.gitignore +++ b/.gitignore @@ -16,4 +16,5 @@ docs/build/ docs/html.zip tvtk/tvtk_classes.zip +tvtk/tvtk_classes/ mayavi/images/m2_about.jpg diff --git a/mayavi/tools/data_wizards/loadtxt.py b/mayavi/tools/data_wizards/loadtxt.py index 5278c368b..b4677dee8 100644 --- a/mayavi/tools/data_wizards/loadtxt.py +++ b/mayavi/tools/data_wizards/loadtxt.py @@ -23,7 +23,7 @@ def _getconv(dtype): return lambda x: int(float(x)) elif issubclass(typ, np.floating): return float - elif issubclass(typ, np.complex128): + elif issubclass(typ, np.complexfloating): return complex else: return str diff --git a/tvtk/code_gen.py b/tvtk/code_gen.py index 8ad3ab6a9..264a2e9be 100644 --- a/tvtk/code_gen.py +++ b/tvtk/code_gen.py @@ -1,10 +1,10 @@ """This module generates tvtk (Traited VTK) classes from the VTK-Python API. -This can be evoked for example by: +This can be evoked for example by running from the ``mayavi`` root: ..code-block:: console - $ python -ic "from tvtk.code_gen import main; main()" -szv + $ python -m tvtk.code_gen -szvno $PWD/tvtk On failures you can then for example do ``import pdb; pdb.pm()`` to do post-mortem debugging. diff --git a/tvtk/tests/test_array_ext.py b/tvtk/tests/test_array_ext.py index cbb03c743..4320267e8 100644 --- a/tvtk/tests/test_array_ext.py +++ b/tvtk/tests/test_array_ext.py @@ -5,12 +5,16 @@ # Copyright (c) 2005, Enthought, Inc. # License: BSD Style. +import pytest import unittest import numpy from tvtk.array_handler import ID_TYPE_CODE, set_id_type_array_py -from tvtk.array_ext import set_id_type_array +try: + from tvtk.array_ext import set_id_type_array +except ModuleNotFoundError: # not compiled + pytest.skip("array_ext not found", allow_module_level=True) class TestArrayExt(unittest.TestCase): diff --git a/tvtk/wrapper_gen.py b/tvtk/wrapper_gen.py index 95b9810ed..1cf9f8b28 100644 --- a/tvtk/wrapper_gen.py +++ b/tvtk/wrapper_gen.py @@ -1591,22 +1591,22 @@ def _write_trait_with_range(self, klass, out, vtk_attr_name): # the code for this trait, # i.e. getattr(self, name_of_method)(...) special_traits = { - '[a-zA-Z0-9]+\.Output$': ( + r'[a-zA-Z0-9]+\.Output$': ( False, False, '_write_any_output'), - '[a-zA-Z0-9]+\.Source$': ( + r'[a-zA-Z0-9]+\.Source$': ( False, False, '_write_any_source'), - '[a-zA-Z0-9]+\.ScalarType$': ( + r'[a-zA-Z0-9]+\.ScalarType$': ( False, False, '_write_any_scalar_type'), # In VTK > 4.5, Set/GetInput have multiple signatures - '[a-zA-Z0-9]+\.Input$': ( + r'[a-zA-Z0-9]+\.Input$': ( False, False, '_write_any_input'), - '[a-zA-Z0-9]+\.InputConnection$': ( + r'[a-zA-Z0-9]+\.InputConnection$': ( False, False, '_write_any_input_connection'), - '[a-zA-Z0-9\.]+FileName$': ( + r'[a-zA-Z0-9\.]+FileName$': ( True, False, '_write_any_something_file_name'), - '[a-zA-Z0-9\.]+FilePrefix$': ( + r'[a-zA-Z0-9\.]+FilePrefix$': ( True, False, '_write_any_something_file_prefix'), 'vtkImageReader2.HeaderSize$': ( True, False, '_write_image_reader2_header_size'), From 0fc20dc93b29eddd5e157a362f6758481ddea3e4 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 6 Sep 2024 13:07:08 -0400 Subject: [PATCH 06/40] FIX: Maybe? --- tvtk/code_gen.py | 3 +++ tvtk/tests/test_array_handler.py | 2 +- tvtk/vtk_parser.py | 15 ++++++++++++++- 3 files changed, 18 insertions(+), 2 deletions(-) diff --git a/tvtk/code_gen.py b/tvtk/code_gen.py index 264a2e9be..03f7fc8f8 100644 --- a/tvtk/code_gen.py +++ b/tvtk/code_gen.py @@ -8,6 +8,9 @@ On failures you can then for example do ``import pdb; pdb.pm()`` to do post-mortem debugging. + +Exceptions to behaviors based on VTK versions and bugs etc. live in ``wrapper_gen.py`` +and ``tvtk_parser.py``. """ # Author: Prabhu Ramachandran # Copyright (c) 2004-2020, Enthought, Inc. diff --git a/tvtk/tests/test_array_handler.py b/tvtk/tests/test_array_handler.py index 96ba65278..6123b8143 100644 --- a/tvtk/tests/test_array_handler.py +++ b/tvtk/tests/test_array_handler.py @@ -163,7 +163,7 @@ def test_array2vtk(self): # Make sure the code at least runs for all # numerical dtypes in numpy # except for half, longdouble and complexfloating - int_types = ['byte', 'short', 'intc', 'int_', 'long', 'longlong'] + int_types = ['byte', 'short', 'int', 'intc', 'int_', 'long', 'longlong'] uint_types = ['ubyte', 'ushort', 'uintc', 'uint', 'ulong', 'ulonglong'] float_types = ['single', 'double'] diff --git a/tvtk/vtk_parser.py b/tvtk/vtk_parser.py index 031af8f33..fa9b57da8 100644 --- a/tvtk/vtk_parser.py +++ b/tvtk/vtk_parser.py @@ -14,7 +14,7 @@ # Local imports (these are relative imports for a good reason). from . import class_tree from . import vtk_module as vtk -from .common import is_version_9 +from .common import is_version_9, vtk_minor_version class VTKMethodParser: @@ -727,6 +727,19 @@ def _find_get_set_methods(self, klass, methods): "UseAxisOrigin", "UseOrientedBounds", "UseTextActor3D", ): default = int(bool(default)) + elif ( + ( + klass_name == "vtkAbstractPolygonalHandleRepresentation3D" + or klass_name.endswith(( + "HandleRepresentation", + "HandleRepresentation2D", + "HandleRepresentation3D", + )) + ) + and key == "DisplayPosition" + and vtk_minor_version < 3 + ): + default = (0., 0., 0.) if value: low = getattr(obj, 'Get%sMinValue' % key)() From 7237d288a7775231c0d9eb09d595bbca15a46550 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 6 Sep 2024 13:24:20 -0400 Subject: [PATCH 07/40] FIX: No build isolation, breaks VTK --- .github/workflows/run-mayavi-tests.yml | 2 +- tvtk/vtk_parser.py | 13 ------------- 2 files changed, 1 insertion(+), 14 deletions(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 2539e32e7..13007811c 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -75,7 +75,7 @@ jobs: python -m pip install --upgrade pip setuptools wheel python -m pip install --upgrade "${{ matrix.qt-api }}" numpy "${{ matrix.vtk }}" pillow pytest traits traitsui configobj - name: Install mayavi and tvtk - run: python -um pip install -ve .[app] + run: python -um pip install --no-build-isolation -ve .[app] - name: Test Mayavi package run: pytest -v mayavi - name: Test tvtk package diff --git a/tvtk/vtk_parser.py b/tvtk/vtk_parser.py index fa9b57da8..27b265e6c 100644 --- a/tvtk/vtk_parser.py +++ b/tvtk/vtk_parser.py @@ -727,19 +727,6 @@ def _find_get_set_methods(self, klass, methods): "UseAxisOrigin", "UseOrientedBounds", "UseTextActor3D", ): default = int(bool(default)) - elif ( - ( - klass_name == "vtkAbstractPolygonalHandleRepresentation3D" - or klass_name.endswith(( - "HandleRepresentation", - "HandleRepresentation2D", - "HandleRepresentation3D", - )) - ) - and key == "DisplayPosition" - and vtk_minor_version < 3 - ): - default = (0., 0., 0.) if value: low = getattr(obj, 'Get%sMinValue' % key)() From b1ab51ed124ea46cdd4c42c60a8c0aeff545d002 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 6 Sep 2024 14:59:26 -0400 Subject: [PATCH 08/40] IFX: More info --- .github/workflows/headless-tests.yml | 8 ++++---- .github/workflows/run-mayavi-tests.yml | 6 +++--- tvtk/vtk_parser.py | 2 +- 3 files changed, 8 insertions(+), 8 deletions(-) diff --git a/.github/workflows/headless-tests.yml b/.github/workflows/headless-tests.yml index 96c0ca2fd..d980f728c 100644 --- a/.github/workflows/headless-tests.yml +++ b/.github/workflows/headless-tests.yml @@ -31,10 +31,10 @@ jobs: - name: Install dependencies run: | python -m pip install --upgrade pip setuptools wheel - python -m pip install numpy "vtk<9.3" pillow pytest traitsui + python -m pip install numpy "vtk<9.3" pillow pytest pytest-timeout traitsui - name: Install mayavi and tvtk - run: python -m pip install -v . + run: python -m pip install --no-build-isolation -v . - name: Test tvtk package - run: pytest -v --pyargs tvtk + run: pytest -v --timeout=10 --pyargs tvtk - name: Test Mayavi package - run: pytest -v --pyargs mayavi + run: pytest -v --timeout=10 --pyargs mayavi diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 13007811c..1cd688eee 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -73,10 +73,10 @@ jobs: run: | set -exo pipefail python -m pip install --upgrade pip setuptools wheel - python -m pip install --upgrade "${{ matrix.qt-api }}" numpy "${{ matrix.vtk }}" pillow pytest traits traitsui configobj + python -m pip install --upgrade "${{ matrix.qt-api }}" numpy "${{ matrix.vtk }}" pillow pytest pytest-timeout traits traitsui configobj - name: Install mayavi and tvtk run: python -um pip install --no-build-isolation -ve .[app] - name: Test Mayavi package - run: pytest -v mayavi + run: pytest -v --timeout=10 mayavi - name: Test tvtk package - run: pytest -sv tvtk + run: pytest -sv --timeout=10 tvtk diff --git a/tvtk/vtk_parser.py b/tvtk/vtk_parser.py index 27b265e6c..031af8f33 100644 --- a/tvtk/vtk_parser.py +++ b/tvtk/vtk_parser.py @@ -14,7 +14,7 @@ # Local imports (these are relative imports for a good reason). from . import class_tree from . import vtk_module as vtk -from .common import is_version_9, vtk_minor_version +from .common import is_version_9 class VTKMethodParser: From 667b2c9218570897f2b56e91c4b34e7315d7319d Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 6 Sep 2024 15:03:00 -0400 Subject: [PATCH 09/40] FIX: Lenient --- .github/workflows/run-mayavi-tests.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 1cd688eee..73cd1e524 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -79,4 +79,4 @@ jobs: - name: Test Mayavi package run: pytest -v --timeout=10 mayavi - name: Test tvtk package - run: pytest -sv --timeout=10 tvtk + run: pytest -sv --timeout=60 tvtk From 9e5a5796cc8b896d50f7ac945bef9519afeb37d6 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 6 Sep 2024 15:07:34 -0400 Subject: [PATCH 10/40] FIX: Time --- .github/workflows/headless-tests.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/headless-tests.yml b/.github/workflows/headless-tests.yml index d980f728c..8ed9362a8 100644 --- a/.github/workflows/headless-tests.yml +++ b/.github/workflows/headless-tests.yml @@ -34,7 +34,7 @@ jobs: python -m pip install numpy "vtk<9.3" pillow pytest pytest-timeout traitsui - name: Install mayavi and tvtk run: python -m pip install --no-build-isolation -v . - - name: Test tvtk package - run: pytest -v --timeout=10 --pyargs tvtk - name: Test Mayavi package run: pytest -v --timeout=10 --pyargs mayavi + - name: Test tvtk package + run: pytest -v --timeout=60 --pyargs tvtk From a6eac998571ffda13715ec022c04bd5240db8d53 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 6 Sep 2024 15:12:13 -0400 Subject: [PATCH 11/40] FIX: Dup --- .github/workflows/headless-tests.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/headless-tests.yml b/.github/workflows/headless-tests.yml index 8ed9362a8..7d4667168 100644 --- a/.github/workflows/headless-tests.yml +++ b/.github/workflows/headless-tests.yml @@ -31,7 +31,7 @@ jobs: - name: Install dependencies run: | python -m pip install --upgrade pip setuptools wheel - python -m pip install numpy "vtk<9.3" pillow pytest pytest-timeout traitsui + python -m pip install numpy "vtk<9.3" pillow pytest pytest-timeout traitsui configobj - name: Install mayavi and tvtk run: python -m pip install --no-build-isolation -v . - name: Test Mayavi package From e2077e5573db4c53b6cdfa3539bd718d6adaddd9 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Thu, 12 Sep 2024 14:48:56 -0400 Subject: [PATCH 12/40] FIX: Reqs --- .github/workflows/headless-tests.yml | 2 +- .github/workflows/run-mayavi-tests.yml | 2 +- mayavi/__init__.py | 1 + setup.py | 4 ++-- 4 files changed, 5 insertions(+), 4 deletions(-) diff --git a/.github/workflows/headless-tests.yml b/.github/workflows/headless-tests.yml index 7d4667168..8ed9362a8 100644 --- a/.github/workflows/headless-tests.yml +++ b/.github/workflows/headless-tests.yml @@ -31,7 +31,7 @@ jobs: - name: Install dependencies run: | python -m pip install --upgrade pip setuptools wheel - python -m pip install numpy "vtk<9.3" pillow pytest pytest-timeout traitsui configobj + python -m pip install numpy "vtk<9.3" pillow pytest pytest-timeout traitsui - name: Install mayavi and tvtk run: python -m pip install --no-build-isolation -v . - name: Test Mayavi package diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 73cd1e524..7d680e1ce 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -73,7 +73,7 @@ jobs: run: | set -exo pipefail python -m pip install --upgrade pip setuptools wheel - python -m pip install --upgrade "${{ matrix.qt-api }}" numpy "${{ matrix.vtk }}" pillow pytest pytest-timeout traits traitsui configobj + python -m pip install --upgrade "${{ matrix.qt-api }}" numpy "${{ matrix.vtk }}" pillow pytest pytest-timeout traits traitsui - name: Install mayavi and tvtk run: python -um pip install --no-build-isolation -ve .[app] - name: Test Mayavi package diff --git a/mayavi/__init__.py b/mayavi/__init__.py index 8e89a722c..e48b65d2f 100644 --- a/mayavi/__init__.py +++ b/mayavi/__init__.py @@ -9,6 +9,7 @@ __requires__ = [ 'apptools', + 'configobj', 'envisage', 'numpy', 'pyface>=6.1.1', diff --git a/setup.py b/setup.py index 2b81501c2..6f978b4c4 100644 --- a/setup.py +++ b/setup.py @@ -130,8 +130,8 @@ def example_files(self): mlab_ref_dir = join(DEFAULT_INPUT_DIR, 'mayavi', 'auto') source_path = join('examples', 'mayavi') - sources = '(\.py)|(\.rst)$' - excluded_dirs = '^\.' + sources = r'(\.py)|(\.rst)$' + excluded_dirs = r'^\.' target_path = mlab_ref_dir target_time = self.latest_modified(target_path, ignore_dirs=excluded_dirs)[0] From b5d215cf161920921638af5b48042e1dba105167 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 12:10:36 -0400 Subject: [PATCH 13/40] FIX: Fix missing --- tvtk/array_handler.py | 1 + 1 file changed, 1 insertion(+) diff --git a/tvtk/array_handler.py b/tvtk/array_handler.py index 6f42b2e89..7e98fcaba 100644 --- a/tvtk/array_handler.py +++ b/tvtk/array_handler.py @@ -185,6 +185,7 @@ def get_vtk_array_type(numeric_array_type): numpy.dtype(numpy.int8): vtkConstants.VTK_CHAR, numpy.dtype(numpy.int16): vtkConstants.VTK_SHORT, numpy.dtype(numpy.int32): vtkConstants.VTK_INT, + numpy.dtype(numpy.int64): vtkConstants.VTK_LONG, numpy.dtype(numpy.uint32): vtkConstants.VTK_UNSIGNED_INT, numpy.dtype(numpy.uint64): vtkConstants.VTK_UNSIGNED_LONG, numpy.dtype(numpy.float32): vtkConstants.VTK_FLOAT, From c06e151ec75d2c7236cd62b39386f20496cb1d47 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 12:23:56 -0400 Subject: [PATCH 14/40] FIX: More --- tvtk/array_handler.py | 5 ++++- tvtk/tests/test_array_handler.py | 5 +++-- 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/tvtk/array_handler.py b/tvtk/array_handler.py index 7e98fcaba..708b1fd95 100644 --- a/tvtk/array_handler.py +++ b/tvtk/array_handler.py @@ -42,9 +42,11 @@ if VTK_LONG_TYPE_SIZE == 4: LONG_TYPE_CODE = numpy.int32 ULONG_TYPE_CODE = numpy.uint32 + LONG_LONG_TYPE_CODE = numpy.int64 elif VTK_LONG_TYPE_SIZE == 8: LONG_TYPE_CODE = numpy.int64 ULONG_TYPE_CODE = numpy.uint64 + LONG_LONG_TYPE_CODE = None BASE_REFERENCE_COUNT = vtk.vtkObject().GetReferenceCount() @@ -185,7 +187,6 @@ def get_vtk_array_type(numeric_array_type): numpy.dtype(numpy.int8): vtkConstants.VTK_CHAR, numpy.dtype(numpy.int16): vtkConstants.VTK_SHORT, numpy.dtype(numpy.int32): vtkConstants.VTK_INT, - numpy.dtype(numpy.int64): vtkConstants.VTK_LONG, numpy.dtype(numpy.uint32): vtkConstants.VTK_UNSIGNED_INT, numpy.dtype(numpy.uint64): vtkConstants.VTK_UNSIGNED_LONG, numpy.dtype(numpy.float32): vtkConstants.VTK_FLOAT, @@ -198,6 +199,8 @@ def get_vtk_array_type(numeric_array_type): numpy.dtype(ULONG_TYPE_CODE): vtkConstants.VTK_UNSIGNED_LONG, numpy.dtype(LONG_TYPE_CODE): vtkConstants.VTK_LONG, } + if LONG_LONG_TYPE_CODE is not None: + _extra[numpy.dtype(LONG_LONG_TYPE_CODE)] = vtkConstants.VTK_LONG_LONG for t in _extra: if t not in _arr_vtk: _arr_vtk[t] = _extra[t] diff --git a/tvtk/tests/test_array_handler.py b/tvtk/tests/test_array_handler.py index 6123b8143..bc4588dec 100644 --- a/tvtk/tests/test_array_handler.py +++ b/tvtk/tests/test_array_handler.py @@ -29,6 +29,7 @@ def mysum(arr): class TestArrayHandler(unittest.TestCase): def _check_arrays(self, arr, vtk_arr): self.assertEqual(vtk_arr.GetNumberOfTuples(), len(arr)) + msg = f"\n{vtk_arr}" if len(arr.shape) == 2: dim1 = arr.shape[1] self.assertEqual(vtk_arr.GetNumberOfComponents(), dim1) @@ -45,8 +46,7 @@ def _check_arrays(self, arr, vtk_arr): self.assertEqual(chr(int(vtk_arr.GetTuple1(i))), arr[i]) else: for i in range(len(arr)): - self.assertEqual(vtk_arr.GetTuple1(i), arr[i]) - + self.assertEqual(vtk_arr.GetTuple1(i), arr[i], msg=msg) def test_array2vtk(self): """Test Numeric array to VTK array conversion and vice-versa.""" @@ -63,6 +63,7 @@ def test_array2vtk(self): t_z.append(numpy.array([-2147483648, 0, 2147483647], numpy.int32)) t_z.append(numpy.array([ -9223372036854775808, 0, 9223372036854775807], numpy.int64)) + assert t_z[-1][0] == -9223372036854775808 t_z.append(numpy.array([0, 255], numpy.uint8)) t_z.append(numpy.array([0, 65535], numpy.uint16)) t_z.append(numpy.array([0, 4294967295], numpy.uint32)) From 3584c2c0af78e3c511245aebcb192636aaf72638 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 12:25:30 -0400 Subject: [PATCH 15/40] FIX: More jobs --- .github/workflows/run-mayavi-tests.yml | 17 ++++++++++++----- 1 file changed, 12 insertions(+), 5 deletions(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 7d680e1ce..7347063eb 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -23,10 +23,6 @@ jobs: qt-api: 'pyqt6' os: macos-14 # arm64 vtk: 'vtk>=9.3' - - python-version: '3.12' - qt-api: 'pyqt6' - os: windows-latest - vtk: 'vtk>=9.3' - python-version: '3.11' qt-api: 'pyqt6' os: ubuntu-latest @@ -43,6 +39,17 @@ jobs: qt-api: 'pyside6' os: macos-13 vtk: 'vtk<9.3' + # A couple of old NumPys + - python-version: '3.12' + qt-api: 'pyside6' + os: ubuntu-latest + vtk: 'vtk>=9.3' + numpy: 'numpy==1.26.4' + - python-version: '3.12' + qt-api: 'pyside6' + os: windows-latest + vtk: 'vtk>=9.3' + numpy: 'numpy==1.26.4' fail-fast: false defaults: run: @@ -73,7 +80,7 @@ jobs: run: | set -exo pipefail python -m pip install --upgrade pip setuptools wheel - python -m pip install --upgrade "${{ matrix.qt-api }}" numpy "${{ matrix.vtk }}" pillow pytest pytest-timeout traits traitsui + python -m pip install --upgrade "${{ matrix.qt-api }}" ${{ matrix.numpy || 'numpy' }} "${{ matrix.vtk }}" pillow pytest pytest-timeout traits traitsui - name: Install mayavi and tvtk run: python -um pip install --no-build-isolation -ve .[app] - name: Test Mayavi package From e5945b51909096e3c597ba5243dad98abe9d5087 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 12:27:33 -0400 Subject: [PATCH 16/40] FIX: Qute --- .github/workflows/run-mayavi-tests.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 7347063eb..3f7b51608 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -80,7 +80,7 @@ jobs: run: | set -exo pipefail python -m pip install --upgrade pip setuptools wheel - python -m pip install --upgrade "${{ matrix.qt-api }}" ${{ matrix.numpy || 'numpy' }} "${{ matrix.vtk }}" pillow pytest pytest-timeout traits traitsui + python -m pip install --upgrade "${{ matrix.qt-api }}" "${{ matrix.numpy || 'numpy' }}" "${{ matrix.vtk }}" pillow pytest pytest-timeout traits traitsui - name: Install mayavi and tvtk run: python -um pip install --no-build-isolation -ve .[app] - name: Test Mayavi package From 9ed4422bafd26c6a7fe9c21ee72079063e071f0d Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 12:33:38 -0400 Subject: [PATCH 17/40] FIX: Check --- .github/workflows/run-mayavi-tests.yml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 3f7b51608..a4602e3b1 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -23,6 +23,10 @@ jobs: qt-api: 'pyqt6' os: macos-14 # arm64 vtk: 'vtk>=9.3' + - python-version: '3.12' + qt-api: 'pyqt6' + os: windows-latest + vtk: 'vtk>=9.3' - python-version: '3.11' qt-api: 'pyqt6' os: ubuntu-latest From fb9bb5c89e611e8227578c67d35025bc30c745d6 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 12:38:22 -0400 Subject: [PATCH 18/40] FIX: Better --- tvtk/array_handler.py | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/tvtk/array_handler.py b/tvtk/array_handler.py index 708b1fd95..0aa3aabdf 100644 --- a/tvtk/array_handler.py +++ b/tvtk/array_handler.py @@ -46,7 +46,10 @@ elif VTK_LONG_TYPE_SIZE == 8: LONG_TYPE_CODE = numpy.int64 ULONG_TYPE_CODE = numpy.uint64 - LONG_LONG_TYPE_CODE = None + if VTK_ID_TYPE_SIZE == 4: + LONG_LONG_TYPE_CODE = numpy.int64 + else: + LONG_LONG_TYPE_CODE = None BASE_REFERENCE_COUNT = vtk.vtkObject().GetReferenceCount() From 7c924cf5cde8c3460eb727ace89b7be1fd18de74 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 14:40:50 -0400 Subject: [PATCH 19/40] FIX: Maybe --- mayavi/__init__.py | 1 + mayavi/preferences/preference_manager.py | 10 ++++------ mayavi/tests/test_mlab_source.py | 2 +- setup.cfg | 2 ++ tvtk/array_handler.py | 11 ++--------- tvtk/wrapper_gen.py | 4 ++++ 6 files changed, 14 insertions(+), 16 deletions(-) diff --git a/mayavi/__init__.py b/mayavi/__init__.py index e48b65d2f..27c028128 100644 --- a/mayavi/__init__.py +++ b/mayavi/__init__.py @@ -17,6 +17,7 @@ 'traits>=6.0.0', 'traitsui>=7.0.0', 'packaging', + 'importlib_resources; python_version<"3.11"', 'vtk' ] diff --git a/mayavi/preferences/preference_manager.py b/mayavi/preferences/preference_manager.py index efe68ef9e..bf27a046d 100644 --- a/mayavi/preferences/preference_manager.py +++ b/mayavi/preferences/preference_manager.py @@ -21,7 +21,7 @@ # Standard library imports from os.path import join -import pkg_resources +import importlib.resources # Enthought library imports. from traits.etsconfig.api import ETSConfig @@ -106,12 +106,11 @@ def _load_preferences(self): for pkg in ('mayavi.preferences', 'tvtk.plugins.scene'): pref = 'preferences.ini' - pref_file = pkg_resources.resource_stream(pkg, pref) - + pref_file = importlib.resources.files(pkg).joinpath(pref) preferences = self.preferences default = preferences.node('default/') - default.load(pref_file) - pref_file.close() + with open(pref_file, 'rb') as fid: + default.load(fid) finally: # Set back the application home. ETSConfig.application_home = app_home @@ -126,4 +125,3 @@ def _preferences_changed(self, preferences): # A Global preference manager that all other modules can use. preference_manager = PreferenceManager() - diff --git a/mayavi/tests/test_mlab_source.py b/mayavi/tests/test_mlab_source.py index 49c6e87c2..ca574e1dd 100644 --- a/mayavi/tests/test_mlab_source.py +++ b/mayavi/tests/test_mlab_source.py @@ -171,7 +171,7 @@ def test_set(self): self.check_traits() self.check_dataset() - def test_strange_shape(self): + def test_basic_strange_shape(self): " Test the MGlyphSource with strange shapes for the arguments " x, y, z, v, s, src = self.get_data() x = y = z = v = s = 0 diff --git a/setup.cfg b/setup.cfg index 212a94acb..699717f82 100644 --- a/setup.cfg +++ b/setup.cfg @@ -4,5 +4,7 @@ addopts = filterwarnings = # Currently unsatisfiable ignore:Workbench will be moved from pyface:PendingDeprecationWarning + # Should be fixed in traits + ignore: module 'sre_.+' is deprecated:DeprecationWarning # We call deprecated methods and classes in our tests, and there are many variants for how the parentheticals are formatted ignore:Call to deprecated .*. \((This|Use|Please|Deprecated|Deprecating|Removed|Part|no|renamed) .*\) -- Deprecated since version.*:DeprecationWarning diff --git a/tvtk/array_handler.py b/tvtk/array_handler.py index 0aa3aabdf..1e1a1db68 100644 --- a/tvtk/array_handler.py +++ b/tvtk/array_handler.py @@ -42,14 +42,9 @@ if VTK_LONG_TYPE_SIZE == 4: LONG_TYPE_CODE = numpy.int32 ULONG_TYPE_CODE = numpy.uint32 - LONG_LONG_TYPE_CODE = numpy.int64 elif VTK_LONG_TYPE_SIZE == 8: LONG_TYPE_CODE = numpy.int64 ULONG_TYPE_CODE = numpy.uint64 - if VTK_ID_TYPE_SIZE == 4: - LONG_LONG_TYPE_CODE = numpy.int64 - else: - LONG_LONG_TYPE_CODE = None BASE_REFERENCE_COUNT = vtk.vtkObject().GetReferenceCount() @@ -202,11 +197,9 @@ def get_vtk_array_type(numeric_array_type): numpy.dtype(ULONG_TYPE_CODE): vtkConstants.VTK_UNSIGNED_LONG, numpy.dtype(LONG_TYPE_CODE): vtkConstants.VTK_LONG, } - if LONG_LONG_TYPE_CODE is not None: - _extra[numpy.dtype(LONG_LONG_TYPE_CODE)] = vtkConstants.VTK_LONG_LONG - for t in _extra: + for t, val in _extra.items(): if t not in _arr_vtk: - _arr_vtk[t] = _extra[t] + _arr_vtk[t] = val try: return _arr_vtk[numeric_array_type] diff --git a/tvtk/wrapper_gen.py b/tvtk/wrapper_gen.py index 1cf9f8b28..7b3afe02f 100644 --- a/tvtk/wrapper_gen.py +++ b/tvtk/wrapper_gen.py @@ -679,6 +679,10 @@ def _gen_state_methods(self, klass, out): if not vtk_val: default = self._reform_name(meths[m][0][0]) + # Weirdness on NumPy 2.1 and vtk >= 9.3 that this does not show up as + # an option and creates problems + if klass.__name__ == "vtkPoints" and m == "DataType" and sys.platform == "win32": + d["int32"] = vtk.VTK_ID_TYPE if extra_val is None: t_def = """tvtk_base.RevPrefixMap(%(d)s, default_value='%(default)s')""" % locals() elif hasattr(extra_val, '__iter__'): From ea2915c9deff105beb65717180ab4c062280a3ac Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 14:45:55 -0400 Subject: [PATCH 20/40] FIX: One moer --- .github/workflows/run-mayavi-tests.yml | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index a4602e3b1..bce235ee4 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -49,6 +49,11 @@ jobs: os: ubuntu-latest vtk: 'vtk>=9.3' numpy: 'numpy==1.26.4' + - python-version: '3.12' + qt-api: 'pyside6' + os: windows-latest + vtk: 'vtk>=9.3' + numpy: 'numpy==2.0.2' - python-version: '3.12' qt-api: 'pyside6' os: windows-latest @@ -84,7 +89,7 @@ jobs: run: | set -exo pipefail python -m pip install --upgrade pip setuptools wheel - python -m pip install --upgrade "${{ matrix.qt-api }}" "${{ matrix.numpy || 'numpy' }}" "${{ matrix.vtk }}" pillow pytest pytest-timeout traits traitsui + python -m pip install --upgrade "${{ matrix.qt-api }}" "${{ matrix.numpy || 'numpy' }}" "${{ matrix.vtk }}" pillow pytest pytest-timeout traits traitsui --only-binary="numpy,vtk" - name: Install mayavi and tvtk run: python -um pip install --no-build-isolation -ve .[app] - name: Test Mayavi package From 48e9fefab52d51ac73b4df03709e9d8eaf8d5256 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 14:46:08 -0400 Subject: [PATCH 21/40] FIX: Bash --- .github/workflows/run-mayavi-tests.yml | 1 - 1 file changed, 1 deletion(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index bce235ee4..599bd7d04 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -85,7 +85,6 @@ jobs: with: python-version: ${{ matrix.python-version }} - name: Install dependencies - shell: bash run: | set -exo pipefail python -m pip install --upgrade pip setuptools wheel From e83f4da8e55538a854ec51c879f67b025fca76e7 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Fri, 13 Sep 2024 14:46:46 -0400 Subject: [PATCH 22/40] STY: Comment --- .github/workflows/run-mayavi-tests.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 599bd7d04..587adc46f 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -43,7 +43,7 @@ jobs: qt-api: 'pyside6' os: macos-13 vtk: 'vtk<9.3' - # A couple of old NumPys + # Some old NumPys - python-version: '3.12' qt-api: 'pyside6' os: ubuntu-latest From 36635975eee4a19c6ad211c2f72a6f15c37a4cb5 Mon Sep 17 00:00:00 2001 From: "Michael R. Crusoe" Date: Fri, 15 Nov 2024 15:17:20 +0100 Subject: [PATCH 23/40] docs: Some JPEGs are actually PNGs --- docs/source/mayavi/application.rst | 2 +- .../mayavi/auto/example_mlab_interactive_dialog.rst | 2 +- docs/source/mayavi/auto/examples.rst | 2 +- docs/source/mayavi/building_applications.rst | 2 +- .../mayavi/example_exploring_a_vector_field.rst | 2 +- docs/source/mayavi/example_fire.rst | 2 +- docs/source/mayavi/example_heart.rst | 2 +- docs/source/mayavi/example_parametric_surface.rst | 2 +- ...r_cut_plane.jpg => example_vector_cut_plane.png} | Bin ...alog.jpg => example_mlab_interactive_dialog.png} | Bin docs/source/mayavi/images/{heart.jpg => heart.png} | Bin .../{mayavi_ui_second.jpg => mayavi_ui_second.png} | Bin docs/source/mayavi/images/{param.jpg => param.png} | Bin .../images/{streamline.jpg => streamline.png} | Bin 14 files changed, 8 insertions(+), 8 deletions(-) rename docs/source/mayavi/{example_vector_cut_plane.jpg => example_vector_cut_plane.png} (100%) rename docs/source/mayavi/images/{example_mlab_interactive_dialog.jpg => example_mlab_interactive_dialog.png} (100%) rename docs/source/mayavi/images/{heart.jpg => heart.png} (100%) rename docs/source/mayavi/images/{mayavi_ui_second.jpg => mayavi_ui_second.png} (100%) rename docs/source/mayavi/images/{param.jpg => param.png} (100%) rename docs/source/mayavi/images/{streamline.jpg => streamline.png} (100%) diff --git a/docs/source/mayavi/application.rst b/docs/source/mayavi/application.rst index dc869daec..97ce25ed4 100644 --- a/docs/source/mayavi/application.rst +++ b/docs/source/mayavi/application.rst @@ -94,7 +94,7 @@ In addition, you can save different layouts into different Shown below is a specifically configured Mayavi user interface view. In this view the size of the various parts are changed. -.. image:: images/mayavi_ui_second.jpg +.. image:: images/mayavi_ui_second.png :alt: Figure of Mayavi's UI after being configured by a user. :align: center diff --git a/docs/source/mayavi/auto/example_mlab_interactive_dialog.rst b/docs/source/mayavi/auto/example_mlab_interactive_dialog.rst index 86645a4ed..3be5c2495 100644 --- a/docs/source/mayavi/auto/example_mlab_interactive_dialog.rst +++ b/docs/source/mayavi/auto/example_mlab_interactive_dialog.rst @@ -27,7 +27,7 @@ This example is discussed in details in the section :ref:`embedding_mayavi_traits`. -.. image:: ../images/example_mlab_interactive_dialog.jpg +.. image:: ../images/example_mlab_interactive_dialog.png :align: center diff --git a/docs/source/mayavi/auto/examples.rst b/docs/source/mayavi/auto/examples.rst index 600e41578..bf26f4b57 100644 --- a/docs/source/mayavi/auto/examples.rst +++ b/docs/source/mayavi/auto/examples.rst @@ -363,7 +363,7 @@ applications.
-.. |0106| image:: ../images/example_mlab_interactive_dialog.jpg +.. |0106| image:: ../images/example_mlab_interactive_dialog.png :width: 150 diff --git a/docs/source/mayavi/building_applications.rst b/docs/source/mayavi/building_applications.rst index ec8930b35..5e70fc90e 100644 --- a/docs/source/mayavi/building_applications.rst +++ b/docs/source/mayavi/building_applications.rst @@ -194,7 +194,7 @@ In a dialog, this would be:: This code creates the following dialog: -.. image:: images/example_mlab_interactive_dialog.jpg +.. image:: images/example_mlab_interactive_dialog.png :align: center A complete, runnable, code based on the above comments is given in the diff --git a/docs/source/mayavi/example_exploring_a_vector_field.rst b/docs/source/mayavi/example_exploring_a_vector_field.rst index e06f71ec6..fa9b0d68a 100644 --- a/docs/source/mayavi/example_exploring_a_vector_field.rst +++ b/docs/source/mayavi/example_exploring_a_vector_field.rst @@ -41,7 +41,7 @@ vector field is still too dense, therefore we go to the `Masking` tab to enable masking, mask with an `on ratio` of 6 (one arrow out of 6 is masked) and turn off the random mode. -.. image:: example_vector_cut_plane.jpg +.. image:: example_vector_cut_plane.png :scale: 50 diff --git a/docs/source/mayavi/example_fire.rst b/docs/source/mayavi/example_fire.rst index c692e07a1..d61f4c574 100644 --- a/docs/source/mayavi/example_fire.rst +++ b/docs/source/mayavi/example_fire.rst @@ -115,7 +115,7 @@ As can be seen from the example, it is quite easy to script mayavi to visualize data. An image of a resulting visualization generated from this script is shown below. -.. image:: images/streamline.jpg +.. image:: images/streamline.png :alt: Sample visualization of the ``fire_ug.vtu`` dataset. diff --git a/docs/source/mayavi/example_heart.rst b/docs/source/mayavi/example_heart.rst index 96c900639..432f7b3f1 100644 --- a/docs/source/mayavi/example_heart.rst +++ b/docs/source/mayavi/example_heart.rst @@ -96,7 +96,7 @@ if you'd like to. You should have a visualization that looks something like the one shown below. -.. image:: images/heart.jpg +.. image:: images/heart.png :alt: Sample visualization of the ``heart.vtk`` dataset. The nice thing about mayavi is that although in this case all of the diff --git a/docs/source/mayavi/example_parametric_surface.rst b/docs/source/mayavi/example_parametric_surface.rst index b9b8b9674..3f2ce5328 100644 --- a/docs/source/mayavi/example_parametric_surface.rst +++ b/docs/source/mayavi/example_parametric_surface.rst @@ -87,7 +87,7 @@ source. Note that the positioning of the different surfaces were effected by mo scene via the 'a' key. For more details on this see the section on :ref:`interaction-with-the-scene`. -.. image:: images/param.jpg +.. image:: images/param.png :alt: Sample visualization using parametric surfaces. diff --git a/docs/source/mayavi/example_vector_cut_plane.jpg b/docs/source/mayavi/example_vector_cut_plane.png similarity index 100% rename from docs/source/mayavi/example_vector_cut_plane.jpg rename to docs/source/mayavi/example_vector_cut_plane.png diff --git a/docs/source/mayavi/images/example_mlab_interactive_dialog.jpg b/docs/source/mayavi/images/example_mlab_interactive_dialog.png similarity index 100% rename from docs/source/mayavi/images/example_mlab_interactive_dialog.jpg rename to docs/source/mayavi/images/example_mlab_interactive_dialog.png diff --git a/docs/source/mayavi/images/heart.jpg b/docs/source/mayavi/images/heart.png similarity index 100% rename from docs/source/mayavi/images/heart.jpg rename to docs/source/mayavi/images/heart.png diff --git a/docs/source/mayavi/images/mayavi_ui_second.jpg b/docs/source/mayavi/images/mayavi_ui_second.png similarity index 100% rename from docs/source/mayavi/images/mayavi_ui_second.jpg rename to docs/source/mayavi/images/mayavi_ui_second.png diff --git a/docs/source/mayavi/images/param.jpg b/docs/source/mayavi/images/param.png similarity index 100% rename from docs/source/mayavi/images/param.jpg rename to docs/source/mayavi/images/param.png diff --git a/docs/source/mayavi/images/streamline.jpg b/docs/source/mayavi/images/streamline.png similarity index 100% rename from docs/source/mayavi/images/streamline.jpg rename to docs/source/mayavi/images/streamline.png From a405de160dd951b77064b92f0b434f4ea3a3143d Mon Sep 17 00:00:00 2001 From: Prabhu Ramachandran Date: Tue, 17 Dec 2024 15:31:20 +0530 Subject: [PATCH 24/40] BUG: Fix issue with vtkGenericCell.GetCellFaces. Calling this on an uninitialized object segfaults on Linux. --- tvtk/vtk_parser.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/tvtk/vtk_parser.py b/tvtk/vtk_parser.py index 031af8f33..58efcc633 100644 --- a/tvtk/vtk_parser.py +++ b/tvtk/vtk_parser.py @@ -654,6 +654,9 @@ def _find_get_set_methods(self, klass, methods): # These hang on Windows (and maybe Fedora 34) elif (klass_name in ('vtkDataEncoder', 'vtkWebApplication')): continue + # This crashes on VTK version 9.4.0 + elif (klass_name == 'vtkGenericCell' and method[3:] == 'CellFaces'): + continue # we can actually process it elif ('Get' + method[3:]) in methods: key = method[3:] From 57ab29d408795cabab8d8ce221ac4fb5a9086f3f Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Tue, 17 Dec 2024 10:03:35 -0500 Subject: [PATCH 25/40] MAINT: Compat for vtk 9.4 --- .github/workflows/run-mayavi-tests.yml | 47 +++++++++++++------------- tvtk/code_gen.py | 6 ++-- tvtk/vtk_parser.py | 39 +++++++++++++-------- 3 files changed, 51 insertions(+), 41 deletions(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 587adc46f..cd0c3510f 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -10,54 +10,53 @@ jobs: tests: strategy: matrix: - os: [ubuntu-latest, windows-latest, macos-13] - python-version: ['3.9'] - qt-api: ['pyqt5'] - vtk: ['vtk<9.3'] + # Always test against the latest VTK, NumPy, and Qt bindings + os: [ubuntu-latest, windows-latest, macos-13, macos-15] + python-version: ['3.12'] + qt-api: ['pyside6'] # prefer the modern bindings + vtk: ['vtk>=9.4'] # always prefer the latest + # Then add some backward compat checks include: - - python-version: '3.12' + # PyQt6 and vtk 9.3 + - python-version: '3.10' qt-api: 'pyqt6' os: ubuntu-latest - vtk: 'vtk>=9.3' - - python-version: '3.12' + vtk: 'vtk==9.3' + - python-version: '3.10' qt-api: 'pyqt6' os: macos-14 # arm64 - vtk: 'vtk>=9.3' - - python-version: '3.12' + vtk: 'vtk==9.3' + - python-version: '3.10' qt-api: 'pyqt6' os: windows-latest - vtk: 'vtk>=9.3' + vtk: 'vtk==9.3' - python-version: '3.11' qt-api: 'pyqt6' os: ubuntu-latest - vtk: 'vtk>=9.3' - - python-version: '3.10' - qt-api: 'pyside6' + vtk: 'vtk==9.3' + # PyQt5 and vtk 9.2 (and one oldest Python) + - python-version: '3.9' + qt-api: 'pyqt5' os: ubuntu-latest - vtk: 'vtk<9.3' + vtk: 'vtk==9.2.5' - python-version: '3.10' - qt-api: 'pyside6' + qt-api: 'pyqt5' os: windows-latest - vtk: 'vtk<9.3' + vtk: 'vtk==9.2.5' - python-version: '3.10' qt-api: 'pyside6' os: macos-13 - vtk: 'vtk<9.3' + vtk: 'vtk==9.2.5' # Some old NumPys - python-version: '3.12' qt-api: 'pyside6' os: ubuntu-latest - vtk: 'vtk>=9.3' + vtk: 'vtk==9.3' numpy: 'numpy==1.26.4' - python-version: '3.12' qt-api: 'pyside6' os: windows-latest - vtk: 'vtk>=9.3' - numpy: 'numpy==2.0.2' - - python-version: '3.12' - qt-api: 'pyside6' - os: windows-latest - vtk: 'vtk>=9.3' + vtk: 'vtk==9.3' numpy: 'numpy==1.26.4' fail-fast: false defaults: diff --git a/tvtk/code_gen.py b/tvtk/code_gen.py index 03f7fc8f8..adfb92aa6 100644 --- a/tvtk/code_gen.py +++ b/tvtk/code_gen.py @@ -4,10 +4,12 @@ ..code-block:: console - $ python -m tvtk.code_gen -szvno $PWD/tvtk + $ VTK_PARSER_VERBOSE=1 python -m tvtk.code_gen -szvno $PWD/tvtk On failures you can then for example do ``import pdb; pdb.pm()`` to do -post-mortem debugging. +post-mortem debugging. If there are segfaults, the VTK_PARSER_VERBOSE=1 should help +point to the culprit, which often needs to be worked around in +``vtk_parser.py::VTKMethodParser._find_get_set_methods``. Exceptions to behaviors based on VTK versions and bugs etc. live in ``wrapper_gen.py`` and ``tvtk_parser.py``. diff --git a/tvtk/vtk_parser.py b/tvtk/vtk_parser.py index 031af8f33..7f95c6e95 100644 --- a/tvtk/vtk_parser.py +++ b/tvtk/vtk_parser.py @@ -8,13 +8,12 @@ import collections.abc import re -import types import os # Local imports (these are relative imports for a good reason). from . import class_tree from . import vtk_module as vtk -from .common import is_version_9 +from .common import vtk_major_version, vtk_minor_version class VTKMethodParser: @@ -632,7 +631,7 @@ def _find_get_set_methods(self, klass, methods): # vtkProp.Get/SetAllocatedRenderTime is private and # SetAllocatedRenderTime takes two args, don't wrap it. continue - elif (not is_version_9()) and ( + elif vtk_major_version < 9 and ( (klass_name == 'vtkGenericAttributeCollection' and method[3:] == 'AttributesToInterpolate') or (klass_name == 'vtkOverlappingAMR' and @@ -668,20 +667,18 @@ def _find_get_set_methods(self, klass, methods): # Find the default and range of the values. if gsm: - if self._verbose: - print(f'Instantiating {klass}') obj = self._get_instance(klass) # print('got instance', obj.__class__) if obj: for key, value in gsm.items(): - if not is_version_9() and ( + if vtk_major_version < 9 and ( # Evil hack, these classes segfault! (klass_name in ['vtkPolyData', 'vtkContext2D']) or # On VTK 8.1.0 this segfaults when uninitialized. (klass_name == 'vtkContextMouseEvent' and key == 'Interactor')): default = None - elif not is_version_9() and ( + elif vtk_major_version < 9 and ( klass_name == 'vtkHyperOctree' and key == 'Dimension'): # This class breaks standard VTK conventions. @@ -695,17 +692,26 @@ def _find_get_set_methods(self, klass, methods): # vtkGenericAttributeCollection.GetAttributesToInterpolate # might only be a problem if VTK is built in debug mode, # but let's keep it just to be safe. - elif is_version_9() and ( + elif vtk_major_version == 9 and ( + # Still broken on 9.4 (klass_name == 'vtkHigherOrderTetra' and - key == 'ParametricCoords') or + key == 'ParametricCoords' and vtk_minor_version < 5) or (klass_name == 'vtkGenericAttributeCollection' and - key == 'AttributesToInterpolate') or + key == 'AttributesToInterpolate' and vtk_minor_version < 4) or (klass_name == 'vtkPlotBar' and - key == 'LookupTable') or + key == 'LookupTable' and vtk_minor_version < 4) or (klass_name == 'vtkLagrangianParticleTracker' and - key == 'IntegrationModel') or + key == 'IntegrationModel' and vtk_minor_version < 4) or False): # just to simplify indentation/updates default = None + + # vtkGenericCell().GetCellFaces() segfaults on 9.4 + elif ( + (vtk_major_version, vtk_minor_version) == (9, 4) + and klass_name == "vtkGenericCell" + and key == "CellFaces" + ): + default = None else: try: if self._verbose: @@ -715,7 +721,7 @@ def _find_get_set_methods(self, klass, methods): default = None # If we don't turn these into integers, they won't instantiate - if is_version_9(): + if vtk_major_version == 9: if klass_name == "vtkAxisActor": if key in ( "AxisOnOrigin", "Use2DMode", "UseTextActor3D", @@ -765,7 +771,7 @@ def _find_get_methods(self, klass, methods): meths.remove(method) return meths - def _get_instance(self, klass): + def _get_instance(self, klass, *, do_print=True): """Given a VTK class, `klass`, returns an instance of the class. @@ -774,6 +780,8 @@ def _get_instance(self, klass): the 'state' methods and the ranges for the Get/Set methods. """ + if self._verbose and do_print: + print(f'Instantiating {klass}') obj = None try: obj = klass() @@ -783,7 +791,8 @@ def _get_instance(self, klass): n = t.get_node(klass.__name__) if n is not None: for c in n.children: - obj = self._get_instance(t.get_class(c.name)) + obj = self._get_instance(t.get_class(c.name), do_print=False) if obj: + print(f" Using super {t.get_class(c.name)} instead of {klass}") break return obj From 74837909d4f6a287080309457d8edf8348e7a163 Mon Sep 17 00:00:00 2001 From: Prabhu Ramachandran Date: Thu, 19 Dec 2024 00:21:36 +0530 Subject: [PATCH 26/40] MAINT: Fix various VTK-9.4 related issues. This fixes TVTK build and test issues. --- docs/source/tvtk/README.txt | 18 ++++++++---------- tvtk/code_gen.py | 7 ++++--- tvtk/messenger.py | 10 +++++----- tvtk/pipeline/browser.py | 4 ++-- tvtk/tests/test_messenger.py | 4 ++-- tvtk/tests/test_tvtk.py | 27 ++++++++++++++++----------- tvtk/vtk_module.py | 10 +++++++++- tvtk/wrapper_gen.py | 18 ++++++++++++++++++ 8 files changed, 64 insertions(+), 34 deletions(-) diff --git a/docs/source/tvtk/README.txt b/docs/source/tvtk/README.txt index 8d71456de..0aa39ee53 100644 --- a/docs/source/tvtk/README.txt +++ b/docs/source/tvtk/README.txt @@ -260,19 +260,17 @@ tvtk wrapper object is created. The following illustrates this:: >>> cs = tvtk.ConeSource() >>> o = cs.output >>> m = tvtk.PolyDataMapper() - >>> m.input = o - >>> print(hash(o)) - 1109012188 - >>> print(hash(m.input)) - 1109012188 + >>> m.input_connection = cs.output_port + >>> print(id(o)) + 126526931186080 + >>> print(id(m.input)) + 126526931186080 >>> del o - >>> print(hash(m.input)) - 1119694156 + >>> print(id(m.input)) + 126526931186080 Thus, after `o` is garbage collected `m.input` no longer refers to the -original tvtk object and a new one is created. This is very similar -to VTK's behaviour. Changing this behaviour is tricky and there are no -plans currently to change this. +original tvtk object the old one is cached and returned. tvtk and traits diff --git a/tvtk/code_gen.py b/tvtk/code_gen.py index 03f7fc8f8..e3a16e58a 100644 --- a/tvtk/code_gen.py +++ b/tvtk/code_gen.py @@ -110,7 +110,7 @@ def generate_code(self): # Write the wrapper files. tree = wrap_gen.get_tree().tree - classes = [] + classes = ['vtkObjectBase'] # This is another class we should not wrap and exists # in version 8.1.0. ignore = ['vtkOpenGLGL2PSHelperImpl'] + [ @@ -124,11 +124,12 @@ def generate_code(self): if (name not in include and not name.startswith('vtk')) or \ name.startswith('vtkQt'): continue - if not hasattr(vtk, name) or not hasattr(getattr(vtk, name), 'IsA'): # noqa + if not hasattr(vtk, name) or \ + not hasattr(getattr(vtk, name), 'AddObserver'): # noqa # We need to wrap VTK classes that are derived # from vtkObjectBase, the others are # straightforward VTK classes that can be used as - # such. All of these have an 'IsA' method so we + # such. All of these have an 'AddObserver' method so we # check for that. Only the vtkObjectBase # subclasses support observers etc. and hence only # those make sense to wrap into TVTK. diff --git a/tvtk/messenger.py b/tvtk/messenger.py index a11f84722..63f495251 100644 --- a/tvtk/messenger.py +++ b/tvtk/messenger.py @@ -145,7 +145,7 @@ def connect(self, obj, event, callback): """ typ = type(callback) - key = hash(obj) + key = id(obj) if not key in self._signals: self._signals[key] = {} signals = self._signals[key] @@ -200,7 +200,7 @@ def disconnect(self, obj, event=None, callback=None, obj_is_hash=False): if obj_is_hash: key = obj else: - key = hash(obj) + key = id(obj) if not key in signals: return if callback is None: @@ -282,11 +282,11 @@ def _get_signals(self, obj): object. """ - ret = self._signals.get(hash(obj)) + ret = self._signals.get(id(obj)) if ret is None: raise MessengerError( - "No such object: %s, has registered itself "\ - "with the messenger."%obj + "No such object: %s, has registered itself " + "with the messenger." % obj ) else: return ret diff --git a/tvtk/pipeline/browser.py b/tvtk/pipeline/browser.py index d2afb5b4f..5c8c8a425 100644 --- a/tvtk/pipeline/browser.py +++ b/tvtk/pipeline/browser.py @@ -447,7 +447,7 @@ class TVTKLeafNode(TreeNodeObject): __ = Python def __hash__(self): - return hash(tvtk.to_vtk(self.object)) + return id(tvtk.to_vtk(self.object)) def _get_name(self): return self.object.__class__.__name__ @@ -496,7 +496,7 @@ def __del__(self): pass def __hash__(self): - return hash(tvtk.to_vtk(self.object)) + return id(tvtk.to_vtk(self.object)) def _get_children_from_cache(self): return [x for x in self.children_cache.values() if x is not None] diff --git a/tvtk/tests/test_messenger.py b/tvtk/tests/test_messenger.py index 66d497953..a92f2344a 100644 --- a/tvtk/tests/test_messenger.py +++ b/tvtk/tests/test_messenger.py @@ -133,12 +133,12 @@ def foo(self, o, e): # Test if things behave sanely if a message was sent and one # of the callbacks has been gc'd. m = messenger.Messenger() - l1 = len(m._signals[hash(c1)]['foo']) + l1 = len(m._signals[id(c1)]['foo']) # del c messenger.send(c1, 'foo') # - l2 = len(m._signals[hash(c1)]['foo']) + l2 = len(m._signals[id(c1)]['foo']) # Since 'c' is gc'd this callback should have been cleared # out. self.assertEqual(l2, l1 - 1) diff --git a/tvtk/tests/test_tvtk.py b/tvtk/tests/test_tvtk.py index 3cba92c56..bb37bee2f 100644 --- a/tvtk/tests/test_tvtk.py +++ b/tvtk/tests/test_tvtk.py @@ -202,22 +202,22 @@ def test_help_trait(self): def test_object_cache(self): """Test if object cache works.""" cs = tvtk.ConeSource() - hash1 = hash(cs) + hash1 = id(cs) o = cs.output if hasattr(o, 'producer_port'): src = o.producer_port.producer else: src = cs.executive.algorithm self.assertEqual(src, cs) - self.assertEqual(hash1, hash(src)) + self.assertEqual(hash1, id(src)) del cs, src gc.collect() # The test sometimes fails as VTK seems to generate objects with the - # same memory address and hash, we try to force it to allocate more - # objects so as to not end up reusing the same address and hash. + # same memory address and hash/id, we try to force it to allocate more + # objects so as to not end up reusing the same address and id. junk = [tvtk.ConeSource() for i in range(50)] - # Now get another ConeSource and ensure the hash is different. + # Now get another ConeSource and ensure the id is different. cs = tvtk.ConeSource() o = cs.output if hasattr(o, 'producer_port'): @@ -230,8 +230,8 @@ def test_object_cache(self): # For VTK 5.x this test is inconsistent, hence skipeed for 5.x # See http://review.source.kitware.com/#/c/15095/ ############################################################## - self.assertEqual(hash1 != hash(src), True) - self.assertEqual(hash(cs), hash(src)) + self.assertEqual(hash1 != id(src), True) + self.assertEqual(id(cs), id(src)) # Test for a bug with collections and the object cache. r = tvtk.Renderer() @@ -565,7 +565,7 @@ def test_information_keys(self): s = tvtk.StructuredPoints() x = s.FIELD_ARRAY_TYPE() y = tvtk.Information() - x.get(y) + y.get(x) def test_parent_child_bounds(self): """CubeAxesActor2D's bounds should be writable.""" @@ -858,7 +858,13 @@ def get_min_max_value(vtk_klass, vtk_attr_name): for name in self.names: vtk_klass = getattr(vtk, name) tvtk_klass_name = get_tvtk_name(name) - + if vtk.vtk_version == '9.4.0': + if tvtk_klass_name.endswith('View'): + continue + if tvtk_klass_name in ['ImageViewer', 'ImageViewer2', + 'OpenGLRenderWindow', + 'RenderWindow']: + continue try: obj = getattr(tvtk, tvtk_klass_name)() except Exception: @@ -882,8 +888,7 @@ def get_min_max_value(vtk_klass, vtk_attr_name): # tvtk.tvtk_classes.open_gl_cell_grid_render_request.shapes_to_draw # uses strings if isinstance(min_value, str): - name = "tvtk.tvtk_classes.open_gl_cell_grid_render_request" - assert name in repr(obj), (obj, trait_name) + assert 'cell_grid_render_request' in repr(obj), (obj, trait_name) continue with self.assertRaises(TraitError): setattr(obj, trait_name, (min_value-1, max_value)) diff --git a/tvtk/vtk_module.py b/tvtk/vtk_module.py index 69dad0eb1..36aaa5249 100644 --- a/tvtk/vtk_module.py +++ b/tvtk/vtk_module.py @@ -52,4 +52,12 @@ try: del vtkDGBoundsResponder, vtkDGOpenGLRenderer, vtkDGSidesResponder except NameError: - pass \ No newline at end of file + pass + +if vtk_version == '9.4.0': + # Instantiating these using TVTK causes a crash on VTK 9.4.0 so skipping. + SKIP = ['vtkIOSSReader', 'vtkIOSSCellGridReader'] + try: + del vtkIOSSReader, vtkIOSSCellGridReader + except NameError: + pass diff --git a/tvtk/wrapper_gen.py b/tvtk/wrapper_gen.py index 7b3afe02f..47cffa08a 100644 --- a/tvtk/wrapper_gen.py +++ b/tvtk/wrapper_gen.py @@ -1654,6 +1654,11 @@ def _write_trait_with_range(self, klass, out, vtk_attr_name): 'vtkLineIntegralConvolution2D.MaxNoiseValue$': ( True, True, '_write_line_integral_conv_2d_max_noise_value' ), + # In VTK 9.4, CellGridSidesQuery's Get/OutputDimensionControl is initialized + # to some random value this happens mostly on MacOS. + 'vtkCellGridSidesQuery.OutputDimensionControl$': ( + True, True, '_write_cell_grid_sides_query_od_control' + ), # In VTK 9.3, vtkCylinderSource's GetLatLongTesselation gives random values # https://gitlab.kitware.com/vtk/vtk/-/issues/19252 'vtkCylinderSource.LatLongTessellation$': ( @@ -1927,6 +1932,19 @@ def _write_line_integral_conv_2d_max_noise_value( vtk_set_meth = getattr(klass, 'Set' + vtk_attr_name) self._write_trait(out, name, t_def, vtk_set_meth, mapped=False) + def _write_cell_grid_sides_query_od_control(self, klass, out, vtk_attr_name): + if vtk_attr_name != 'OutputDimensionControl': + raise RuntimeError(f"Wrong attribute name: {vtk_attr_name}") + if vtk_major_version >= 9: + message = ("vtkCellGridSidesQuery: " + "OutputDimensionControl not updatable " + "(VTK 9.4 bug - value not properly initialized)") + print(message) + t_def = 'tvtk_base.true_bool_trait' + name = self._reform_name(vtk_attr_name) + vtk_set_meth = getattr(klass, 'Set' + vtk_attr_name) + self._write_trait(out, name, t_def, vtk_set_meth, mapped=True) + def _write_cylinder_source_lat_long_tessellation( self, klass, out, vtk_attr_name ): From 8d8a9ff96b6de99583e1ee2e861b4daec56d95dd Mon Sep 17 00:00:00 2001 From: Prabhu Ramachandran Date: Thu, 19 Dec 2024 00:52:51 +0530 Subject: [PATCH 27/40] More fixes to get builds working. --- mayavi/components/actor.py | 9 +-------- tvtk/wrapper_gen.py | 4 ++-- 2 files changed, 3 insertions(+), 10 deletions(-) diff --git a/mayavi/components/actor.py b/mayavi/components/actor.py index 97bc06327..6fc43c8fb 100644 --- a/mayavi/components/actor.py +++ b/mayavi/components/actor.py @@ -115,14 +115,7 @@ def update_data(self): sends a `data_changed` event. """ # Invoke render to update any changes. - from mayavi.modules.outline import Outline - from mayavi.components.glyph import Glyph - #FIXME: A bad hack, but without these checks results in seg fault - input = self.inputs[0] - if isinstance(input, Outline) or isinstance(input, Glyph): - self.mapper.update(0) - else: - self.mapper.update() + self.mapper.update() self.render() ###################################################################### diff --git a/tvtk/wrapper_gen.py b/tvtk/wrapper_gen.py index 47cffa08a..b9314e4ec 100644 --- a/tvtk/wrapper_gen.py +++ b/tvtk/wrapper_gen.py @@ -54,7 +54,7 @@ def get_trait_def(value, **kwargs): Example ------- >>> get_trait_def([100., 200.], enter_set=True, auto_set=False) - ('traits.Array', '', 'auto_set=False, enter_set=True, shape=(2,), dtype=float, value=[100.0, 200.0], cols=2') + ('traits.Array', '', 'auto_set=False, enter_set=True, shape=(None,), dtype=float, value=[100.0, 200.0], cols=2') >>> get_trait_def(100, enter_set=True, auto_set=False) ('traits.Int', '100', 'auto_set=False, enter_set=True') >>> get_trait_def(u'something', enter_set=True, auto_set=False) @@ -80,7 +80,7 @@ def get_trait_def(value, **kwargs): return 'traits.String', '{!r}'.format(value), kwargs_code elif type_ in (tuple, list): - shape = (len(value),) + shape = (None,) dtypes = set(type(element) for element in value) dtype = dtypes.pop().__name__ if len(dtypes) == 1 else None if dtype == 'int' and sys.platform.startswith('win'): From 2f4b816bcb0e6ba0b0919e2b689edd321e4245f6 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 14:33:36 -0500 Subject: [PATCH 28/40] FIX: Move --- tvtk/vtk_parser.py | 3 --- 1 file changed, 3 deletions(-) diff --git a/tvtk/vtk_parser.py b/tvtk/vtk_parser.py index 43ef23cbf..7f95c6e95 100644 --- a/tvtk/vtk_parser.py +++ b/tvtk/vtk_parser.py @@ -653,9 +653,6 @@ def _find_get_set_methods(self, klass, methods): # These hang on Windows (and maybe Fedora 34) elif (klass_name in ('vtkDataEncoder', 'vtkWebApplication')): continue - # This crashes on VTK version 9.4.0 - elif (klass_name == 'vtkGenericCell' and method[3:] == 'CellFaces'): - continue # we can actually process it elif ('Get' + method[3:]) in methods: key = method[3:] From 90e81bccb6fbe7c7f06cfeb0f3756df715a2c59b Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:00:32 -0500 Subject: [PATCH 29/40] MAINT: 9+ --- README.rst | 13 +++--- mayavi/components/ui/actor.py | 16 ++----- mayavi/tests/test_builtin_image.py | 3 -- mayavi/tools/figure.py | 11 +---- pyproject.toml | 2 +- tvtk/common.py | 5 -- tvtk/tests/test_class_tree.py | 13 +----- tvtk/tests/test_vtk_parser.py | 41 ++++++----------- tvtk/vtk_parser.py | 73 ++++++++++-------------------- tvtk/wrapper_gen.py | 46 +++++++++---------- 10 files changed, 73 insertions(+), 150 deletions(-) diff --git a/README.rst b/README.rst index 5d8426703..f6829bd14 100644 --- a/README.rst +++ b/README.rst @@ -86,12 +86,11 @@ By itself Mayavi is not a difficult package to install but its dependencies are unfortunately rather heavy. However, many of these dependencies are now available as wheels on PyPI. The two critical dependencies are, - 1. VTK_ - 2. A GUI toolkit, either PyQt4_, PySide_, PySide2_, PyQt5_ or wxPython_. + 1. VTK_ >= 9.0 + 2. A GUI toolkit, either PySide6_, PyQt6_, PySide2_, PyQt5_, or wxPython_. The latest VTK wheels are available on all the major platforms (Windows, -MacOS, and Linux), but only for 64 bit machines. Python 3.x is fully supported -on all these operating systems and Python 2.7.x on MacOS and Linux. If you are +MacOS, and Linux). If you are out of luck, and your platform is not supported then you will need to install VTK yourself using your particular distribution as discussed in the `General Build and Installation instructions @@ -133,7 +132,7 @@ If you are unable to do this, read the documentation above and find a way to install VTK and a suitable UI toolkit and then repeat the above. If you are interested in the jupyter notebook support as well, do the -following (after ensuring that you have jupyter installed of course. +following (after ensuring that you have jupyter installed of course. **Note, the Jupyter notebook function is only supported starting mayavi version 4.5.0**):: @@ -200,8 +199,8 @@ Test suite The basic test suites for tvtk and mayavi can be run using nose:: - nosetests -v tvtk/tests - nosetests -v mayavi + pytest -v tvtk/tests + pytest -v mayavi The integration tests:: diff --git a/mayavi/components/ui/actor.py b/mayavi/components/ui/actor.py index d58386749..7f0c6899f 100644 --- a/mayavi/components/ui/actor.py +++ b/mayavi/components/ui/actor.py @@ -14,8 +14,6 @@ from traitsui.api import (View, Group, Item, InstanceEditor, DropEditor, Tabbed) -from tvtk.api import tvtk -from tvtk.common import vtk_major_version # The properties view group. _prop_base_group = Group(Item(name='representation'), @@ -48,16 +46,10 @@ ) # The Texture's view group -if vtk_major_version > 7: - _texture_group = Group(Item(name='interpolate'), - Item(name='color_mode'), - Item(name='repeat'), - show_border=True) -else: - _texture_group = Group(Item(name='interpolate'), - Item(name='map_color_scalars_through_lookup_table'), - Item(name='repeat'), - show_border=True) +_texture_group = Group(Item(name='interpolate'), + Item(name='color_mode'), + Item(name='repeat'), + show_border=True) # The Actor's view group. _actor_base_group = Group(Item(name='visibility')) diff --git a/mayavi/tests/test_builtin_image.py b/mayavi/tests/test_builtin_image.py index 0af5a6eca..0ba015076 100644 --- a/mayavi/tests/test_builtin_image.py +++ b/mayavi/tests/test_builtin_image.py @@ -11,7 +11,6 @@ from numpy import array # Enthought library imports. -from tvtk.common import vtk_major_version from mayavi.core.null_engine import NullEngine from mayavi.sources.builtin_image import BuiltinImage from mayavi.modules.surface import Surface @@ -40,8 +39,6 @@ def setUp(self): image_data.data_source.radius = array([80., 80., 80.]) image_data.data_source.center = array([150., 150., 0.]) image_data.data_source.whole_extent = array([10, 245, 10, 245, 0, 0]) - if vtk_major_version < 8: - image_data.data_source.set_update_extent_to_whole_extent() self.e = e self.scene = e.current_scene diff --git a/mayavi/tools/figure.py b/mayavi/tools/figure.py index a196f1fd9..8890b470c 100644 --- a/mayavi/tools/figure.py +++ b/mayavi/tools/figure.py @@ -19,7 +19,6 @@ # imports from tvtk.api import tvtk -from tvtk.common import vtk_major_version from mayavi.core.scene import Scene from mayavi.core.registry import registry from .camera import view @@ -313,19 +312,13 @@ def screenshot(figure=None, mode='rgb', antialiased=False): out = tvtk.UnsignedCharArray() shape = (y, x, 3) pixel_getter = figure.scene.render_window.get_pixel_data - if vtk_major_version > 7: - pg_args = (0, 0, x - 1, y - 1, 1, out, 0) - else: - pg_args = (0, 0, x - 1, y - 1, 1, out) + pg_args = (0, 0, x - 1, y - 1, 1, out, 0) elif mode == 'rgba': out = tvtk.FloatArray() shape = (y, x, 4) pixel_getter = figure.scene.render_window.get_rgba_pixel_data - if vtk_major_version > 7: - pg_args = (0, 0, x - 1, y - 1, 1, out, 0) - else: - pg_args = (0, 0, x - 1, y - 1, 1, out) + pg_args = (0, 0, x - 1, y - 1, 1, out, 0) else: raise ValueError('mode type not understood') diff --git a/pyproject.toml b/pyproject.toml index 9da1141ed..d6cad3fee 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -2,6 +2,6 @@ requires = [ "numpy>=2.0.0rc2,<3", "setuptools", - "vtk", + "vtk>=9", "wheel" ] diff --git a/tvtk/common.py b/tvtk/common.py index bad08ce2c..0b6a618d1 100644 --- a/tvtk/common.py +++ b/tvtk/common.py @@ -5,7 +5,6 @@ # License: BSD Style. from contextlib import contextmanager -import string import re import vtk @@ -67,10 +66,6 @@ def _sanitize_name(name): return name -def is_version_9(): - return vtk_major_version > 8 - - def configure_connection(obj, inp): """ Configure topology for vtk pipeline obj.""" if hasattr(inp, 'output_port'): diff --git a/tvtk/tests/test_class_tree.py b/tvtk/tests/test_class_tree.py index f71e43bf2..5d9ad976f 100644 --- a/tvtk/tests/test_class_tree.py +++ b/tvtk/tests/test_class_tree.py @@ -7,7 +7,6 @@ """ import builtins -import sys import unittest from contextlib import contextmanager @@ -19,8 +18,6 @@ _cache = class_tree.ClassTree(vtk) _cache.create() -vtk_major_version = vtk.vtkVersion.GetVTKMajorVersion() - def get_level(klass): """Gets the inheritance level of a given class.""" if not klass.__bases__: @@ -54,15 +51,7 @@ def test_basic_vtk(self): if (hasattr(vtk, 'vtkTuple')): names = [x.name for x in t.tree[0]] names.sort() - if vtk_major_version == 7: - expect = ['object', 'vtkColor3', 'vtkColor4', 'vtkDenseArray', - 'vtkQuaternion', 'vtkRect', - 'vtkSparseArray', 'vtkTuple', - 'vtkTypedArray','vtkVector', - 'vtkVector2', 'vtkVector3'] - else: - self.assertGreaterEqual(vtk_major_version, 8) - expect = ['object'] + expect = ['object'] self.assertEqual(names, expect) elif (hasattr(vtk, 'vtkVector')): self.assertEqual(len(t.tree[0]), 11) diff --git a/tvtk/tests/test_vtk_parser.py b/tvtk/tests/test_vtk_parser.py index c97635f58..ce88ce9e8 100644 --- a/tvtk/tests/test_vtk_parser.py +++ b/tvtk/tests/test_vtk_parser.py @@ -16,6 +16,7 @@ import unittest from tvtk import vtk_parser from tvtk import vtk_module as vtk +from tvtk.common import vtk_major_version, vtk_minor_version import time # Only used when timing. import sys # Only used when debugging. @@ -23,8 +24,7 @@ # This is a little expensive to create so we cache it. _cache = vtk_parser.VTKMethodParser() -vtk_major_version = vtk.vtkVersion.GetVTKMajorVersion() -vtk_minor_version = vtk.vtkVersion.GetVTKMinorVersion() + class TestVTKParser(unittest.TestCase): def setUp(self): @@ -44,13 +44,9 @@ def test_parse(self): p = self.p # Simple case of a vtkObject. p.parse(vtk.vtkObject()) - if vtk_major_version < 9: - self.assertEqual(p.get_toggle_methods(), - {'Debug': 0, 'GlobalWarningDisplay': 1}) - else: - self.assertIn(p.get_toggle_methods(), - [{'Debug': False, 'GlobalWarningDisplay': 1}, - {'Debug': False, 'GlobalWarningDisplay': 0}]) + self.assertIn(p.get_toggle_methods(), + [{'Debug': False, 'GlobalWarningDisplay': 1}, + {'Debug': False, 'GlobalWarningDisplay': 0}]) self.assertEqual(p.get_state_methods(), {}) self.assertEqual(p.get_get_methods(), ['GetCommand', 'GetMTime']) @@ -87,8 +83,7 @@ def test_parse(self): ['Gouraud', 1], ['Phong', 2]], 'Representation': [['Surface', 2], ['Points', 0], ['Surface', 2], ['Wireframe', 1]]} - if vtk_major_version >= 9: - res['Interpolation'].insert(-1, ['PBR', 3]) + res['Interpolation'].insert(-1, ['PBR', 3]) self.assertEqual(p.get_state_methods(), res) self.assertEqual(p.state_meths, p.get_state_methods()) @@ -118,15 +113,13 @@ def test_parse(self): 'SpecularPower': (1.0, (0.0, 100.0))} if ('ReferenceCount' not in p.get_get_set_methods()): del res['ReferenceCount'] - if vtk_major_version > 7: - res['MaterialName'] = (None, None) - res['VertexColor'] = ((0.5, 1.0, 0.5), None) - if vtk_major_version >= 9: - res['EmissiveFactor'] = ((1.0, 1.0, 1.0), None) - res['Metallic'] = (0., float_max) - res['NormalScale'] = (1., None) - res['OcclusionStrength'] = (1., float_max) - res['Roughness'] = (0.5, float_max) + res['MaterialName'] = (None, None) + res['VertexColor'] = ((0.5, 1.0, 0.5), None) + res['EmissiveFactor'] = ((1.0, 1.0, 1.0), None) + res['Metallic'] = (0., float_max) + res['NormalScale'] = (1., None) + res['OcclusionStrength'] = (1., float_max) + res['Roughness'] = (0.5, float_max) if (vtk_major_version, vtk_minor_version) >= (9, 1): res['Anisotropy'] = (0.0, (0.0, 1.0)) res['AnisotropyRotation'] = (0.0, (0.0, 1.0)) @@ -180,12 +173,8 @@ def test_parse(self): res = ['AddShaderVariable', 'BackfaceRender', 'DeepCopy', 'ReleaseGraphicsResources', 'RemoveAllTextures', 'RemoveTexture', 'Render'] - if (vtk_major_version >= 7 or vtk_minor_version >= 2) and \ - vtk_major_version < 9: - res.append('VTKTextureUnit') - if vtk_major_version == 9: - res.extend(['SetBaseColorTexture', 'SetEmissiveTexture', - 'SetNormalTexture', 'SetORMTexture']) + res.extend(['SetBaseColorTexture', 'SetEmissiveTexture', + 'SetNormalTexture', 'SetORMTexture']) if vtk_major_version == 9 and vtk_minor_version > 0: res.extend([ 'ComputeIORFromReflectance', 'ComputeReflectanceFromIOR', diff --git a/tvtk/vtk_parser.py b/tvtk/vtk_parser.py index 7f95c6e95..b14b197d6 100644 --- a/tvtk/vtk_parser.py +++ b/tvtk/vtk_parser.py @@ -631,18 +631,6 @@ def _find_get_set_methods(self, klass, methods): # vtkProp.Get/SetAllocatedRenderTime is private and # SetAllocatedRenderTime takes two args, don't wrap it. continue - elif vtk_major_version < 9 and ( - (klass_name == 'vtkGenericAttributeCollection' and - method[3:] == 'AttributesToInterpolate') or - (klass_name == 'vtkOverlappingAMR' and - method[3:] == 'Origin') or - (klass_name == 'vtkOrientationMarkerWidget' and - method[3:] in ['OutlineColor', 'Viewport']) or - (klass_name == 'vtkImageDataGeometryFilter' and - method[3:] == 'Extent') or - (klass_name == 'vtkVolumeMapper' and - method[3:] == 'CroppingRegionPlanes')): - continue elif (klass_name == 'vtkContextMouseEvent' and method[3:] == 'Interactor'): continue @@ -668,55 +656,40 @@ def _find_get_set_methods(self, klass, methods): # Find the default and range of the values. if gsm: obj = self._get_instance(klass) - # print('got instance', obj.__class__) if obj: for key, value in gsm.items(): - if vtk_major_version < 9 and ( - # Evil hack, these classes segfault! - (klass_name in ['vtkPolyData', 'vtkContext2D']) or - # On VTK 8.1.0 this segfaults when uninitialized. - (klass_name == 'vtkContextMouseEvent' and - key == 'Interactor')): + # Broken in <= 9.3 + if ( + (vtk_major_version, vtk_minor_version) <= (9, 3) + and f"{klass_name}.Get{key}" in ( + "vtkGenericAttributeCollection.GetAttributesToInterpolate", + "vtkPlotBar.GetLookupTable", + "vtkLagrangianParticleTracker.GetIntegrationModel", + ) + ): default = None - elif vtk_major_version < 9 and ( - klass_name == 'vtkHyperOctree' and - key == 'Dimension'): - # This class breaks standard VTK conventions. - gsm[key] = (3, (1, 3)) - continue - # On VTK 9.0.0 vtkHigherOrderTetra.GetParametricCorods - # segfauts when uninitialized, see: - # - # https://gitlab.kitware.com/vtk/vtk/-/merge_requests/6729#note_732848 # noqa: E501 - # - # vtkGenericAttributeCollection.GetAttributesToInterpolate - # might only be a problem if VTK is built in debug mode, - # but let's keep it just to be safe. - elif vtk_major_version == 9 and ( - # Still broken on 9.4 - (klass_name == 'vtkHigherOrderTetra' and - key == 'ParametricCoords' and vtk_minor_version < 5) or - (klass_name == 'vtkGenericAttributeCollection' and - key == 'AttributesToInterpolate' and vtk_minor_version < 4) or - (klass_name == 'vtkPlotBar' and - key == 'LookupTable' and vtk_minor_version < 4) or - (klass_name == 'vtkLagrangianParticleTracker' and - key == 'IntegrationModel' and vtk_minor_version < 4) or - False): # just to simplify indentation/updates + # Broken in <= 9.4 + # https://gitlab.kitware.com/vtk/vtk/-/merge_requests/6729#note_732848 + elif ( + (vtk_major_version, vtk_minor_version) <= (9, 4) + and f"{klass_name}.Get{key}" in ( + "vtkHigherOrderTetra.GetParametricCoords", + ) + ): default = None - - # vtkGenericCell().GetCellFaces() segfaults on 9.4 + # Broken in 9.4 elif ( (vtk_major_version, vtk_minor_version) == (9, 4) - and klass_name == "vtkGenericCell" - and key == "CellFaces" + and f"{klass_name}.Get{key}" in ( + "vtkGenericCell.GetCellFaces", + ) ): default = None else: try: if self._verbose: - print(f' Calling {klass_name}.Get{key}()') - default = getattr(obj, 'Get%s' % key)() + print(f" Calling {klass_name}.Get{key}()") + default = getattr(obj, f"Get{key}")() except TypeError: default = None diff --git a/tvtk/wrapper_gen.py b/tvtk/wrapper_gen.py index b9314e4ec..1c3cb0d25 100644 --- a/tvtk/wrapper_gen.py +++ b/tvtk/wrapper_gen.py @@ -17,7 +17,8 @@ # Local imports (these are relative imports because the package is not # installed when these modules are imported). -from .common import get_tvtk_name, camel2enthought, vtk_major_version, _sanitize_name +from .common import get_tvtk_name, camel2enthought, _sanitize_name +from .common import vtk_major_version, vtk_minor_version from . import vtk_parser from . import indenter @@ -1836,12 +1837,12 @@ def _write_smart_volume_mapper_vector_component(self, klass, out, default, rng = self.parser.get_get_set_methods()[vtk_attr_name] - if vtk_major_version >= 8: - message = ("vtkSmartVolumeMapper: " - "VectorComponent not updatable " - "(VTK 8.x bug - value not properly initialized)") - print(message) - default = rng[0] + # TODO: Still an issue in 9.x? + message = ("vtkSmartVolumeMapper: " + "VectorComponent not updatable " + "(VTK 8.x bug - value not properly initialized)") + print(message) + default = rng[0] t_def = ('traits.Trait({default}, traits.Range{rng}, ' 'enter_set=True, auto_set=False)').format(default=default, rng=rng) @@ -1857,12 +1858,6 @@ def _write_span_space_resolution(self, klass, out, default, rng = self.parser.get_get_set_methods()[vtk_attr_name] - if vtk_major_version == 7: - message = ("vtkSpanSpace: " - "Resolution not updatable " - "(VTK 7.x bug - value not properly initialized)") - print(message) - default = rng[0] t_def = ('traits.Trait({default}, traits.Range{rng}, ' 'enter_set=True, auto_set=False)').format(default=default, rng=rng) @@ -1876,11 +1871,11 @@ def _write_hyper_tree_grid_cell_centers_vertex_cells(self, klass, out, raise RuntimeError("Not sure why you ask for me! " "I only deal with VertexCells. Panicking.") - if vtk_major_version >= 8: - message = ("vtkHyperTreeGridCellCenters: " - "VertexCells not updatable " - "(VTK 8.x bug - value not properly initialized)") - print(message) + # TODO: Still an issue in 9.x? + message = ("vtkHyperTreeGridCellCenters: " + "VertexCells not updatable " + "(VTK 8.x bug - value not properly initialized)") + print(message) t_def = 'tvtk_base.true_bool_trait' @@ -1897,12 +1892,13 @@ def _write_euclidean_cluster_extraction_radius( default, rng = self.parser.get_get_set_methods()[vtk_attr_name] - if vtk_major_version >= 8: - message = ("vtkEuclideanClusterExtraction: " - "Radius not updatable " - "(VTK 9.1 bug - value not properly initialized)") - print(message) - default = rng[0] + # TODO: Still an issue in 9.x? + message = ("vtkEuclideanClusterExtraction: " + "Radius not updatable " + "(VTK 9.1 bug - value not properly initialized)") + print(message) + default = rng[0] + t_def = ('traits.Trait({default}, traits.Range{rng}, ' 'enter_set=True, auto_set=False)').format(default=default, rng=rng) @@ -1950,7 +1946,7 @@ def _write_cylinder_source_lat_long_tessellation( ): if vtk_attr_name != 'LatLongTessellation': raise RuntimeError(f"Wrong attribute name: {vtk_attr_name}") - if vtk_major_version >= 9: + if (vtk_major_version, vtk_minor_version) <= (9, 3): message = ("vtkCylinderSource: " "LatLongTesselation not updatable " "(VTK 9.3 bug - value not properly initialized)") From 831e09df4b8c8b8a8e0e47844d30ead597854766 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:05:39 -0500 Subject: [PATCH 30/40] CI: Test older --- .github/workflows/run-mayavi-tests.yml | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index cd0c3510f..6bf54fb2c 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -58,6 +58,15 @@ jobs: os: windows-latest vtk: 'vtk==9.3' numpy: 'numpy==1.26.4' + # Older Python and VTKs + - python-version: '3.10' + qt-api: 'pyqt5' + os: ubuntu-latest + vtk: 'vtk==9.2' + - python-version: '3.9' + qt-api: 'pyqt5' + os: ubuntu-latest + vtk: 'vtk==9.1' fail-fast: false defaults: run: From ba56f80ad42b19c48e8577bd3759092dee2220f1 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:06:21 -0500 Subject: [PATCH 31/40] FIX: Ancient --- .github/workflows/run-mayavi-tests.yml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 6bf54fb2c..22e96176f 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -67,6 +67,10 @@ jobs: qt-api: 'pyqt5' os: ubuntu-latest vtk: 'vtk==9.1' + - python-version: '3.9' + qt-api: 'pyqt5' + os: ubuntu-latest + vtk: 'vtk==9.0' fail-fast: false defaults: run: From a642f05d1e5590c378ad06e9dcd525b359d1e003 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:09:42 -0500 Subject: [PATCH 32/40] FIX: More --- .github/dependabot.yml | 10 ++++++++++ .github/workflows/run-mayavi-tests.yml | 11 ++++++----- 2 files changed, 16 insertions(+), 5 deletions(-) create mode 100644 .github/dependabot.yml diff --git a/.github/dependabot.yml b/.github/dependabot.yml new file mode 100644 index 000000000..d57929b9e --- /dev/null +++ b/.github/dependabot.yml @@ -0,0 +1,10 @@ +version: 2 +updates: + - package-ecosystem: "github-actions" + directory: "/" + schedule: + interval: "monthly" + groups: + actions: + patterns: + - "*" diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 22e96176f..f4e1eca81 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -63,14 +63,15 @@ jobs: qt-api: 'pyqt5' os: ubuntu-latest vtk: 'vtk==9.2' + # In practice this one segfaults :( + # - python-version: '3.9' + # qt-api: 'pyqt5' + # os: ubuntu-latest + # vtk: 'vtk~=9.1' - python-version: '3.9' qt-api: 'pyqt5' os: ubuntu-latest - vtk: 'vtk==9.1' - - python-version: '3.9' - qt-api: 'pyqt5' - os: ubuntu-latest - vtk: 'vtk==9.0' + vtk: 'vtk==9.0.2' fail-fast: false defaults: run: From 832453bdfbe5cea604c1dee22ed989b959e6bca4 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:09:59 -0500 Subject: [PATCH 33/40] FIX: Pt 2 --- .github/workflows/run-mayavi-tests.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index f4e1eca81..e3f173f52 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -62,7 +62,7 @@ jobs: - python-version: '3.10' qt-api: 'pyqt5' os: ubuntu-latest - vtk: 'vtk==9.2' + vtk: 'vtk==9.2.2' # In practice this one segfaults :( # - python-version: '3.9' # qt-api: 'pyqt5' From 47bc677d3cb4c042af7381f10474e69627d59d1c Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:10:15 -0500 Subject: [PATCH 34/40] FIX: pt0 --- .github/workflows/run-mayavi-tests.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index e3f173f52..e05ac52b0 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -59,10 +59,10 @@ jobs: vtk: 'vtk==9.3' numpy: 'numpy==1.26.4' # Older Python and VTKs - - python-version: '3.10' + - python-version: '3.9' qt-api: 'pyqt5' os: ubuntu-latest - vtk: 'vtk==9.2.2' + vtk: 'vtk==9.2.0' # In practice this one segfaults :( # - python-version: '3.9' # qt-api: 'pyqt5' From 82572cbd0968cc9384e3c0aee637d474b686156f Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:11:51 -0500 Subject: [PATCH 35/40] FIX: 2 --- .github/workflows/run-mayavi-tests.yml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index e05ac52b0..5660479c6 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -59,10 +59,10 @@ jobs: vtk: 'vtk==9.3' numpy: 'numpy==1.26.4' # Older Python and VTKs - - python-version: '3.9' + - python-version: '3.10' qt-api: 'pyqt5' os: ubuntu-latest - vtk: 'vtk==9.2.0' + vtk: 'vtk==9.2.2' # In practice this one segfaults :( # - python-version: '3.9' # qt-api: 'pyqt5' @@ -71,7 +71,7 @@ jobs: - python-version: '3.9' qt-api: 'pyqt5' os: ubuntu-latest - vtk: 'vtk==9.0.2' + vtk: 'vtk==9.0.2' # oldest available on 3.9 which is the oldest Python we support fail-fast: false defaults: run: From dd6f64cac849e3c31f9913759ecc466e42526484 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:19:36 -0500 Subject: [PATCH 36/40] FIX: Lets see it --- .github/workflows/run-mayavi-tests.yml | 9 ++++----- tvtk/tests/test_tvtk.py | 12 +++++++++++- 2 files changed, 15 insertions(+), 6 deletions(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 5660479c6..7424f9921 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -63,11 +63,10 @@ jobs: qt-api: 'pyqt5' os: ubuntu-latest vtk: 'vtk==9.2.2' - # In practice this one segfaults :( - # - python-version: '3.9' - # qt-api: 'pyqt5' - # os: ubuntu-latest - # vtk: 'vtk~=9.1' + - python-version: '3.9' + qt-api: 'pyqt5' + os: ubuntu-latest + vtk: 'vtk~=9.1' - python-version: '3.9' qt-api: 'pyqt5' os: ubuntu-latest diff --git a/tvtk/tests/test_tvtk.py b/tvtk/tests/test_tvtk.py index bb37bee2f..6ea058628 100644 --- a/tvtk/tests/test_tvtk.py +++ b/tvtk/tests/test_tvtk.py @@ -8,6 +8,7 @@ # Copyright (c) 2004-2020, Enthought, Inc. # License: BSD Style. +import os import unittest import pickle import weakref @@ -43,6 +44,9 @@ from tvtk.tvtk_classes import tvtk_helper +on_gha = os.getenv("GITHUB_ACTION", None) is not None + + def mysum(arr): val = arr while type(val) == numpy.ndarray: @@ -818,14 +822,20 @@ def setUpClass(cls): def test_all_instantiable(self): """Test if all the TVTK classes can be instantiated""" errors = [] + if on_gha: + print("\n::group::Instantiating TVTK classes") for name in self.names: tvtk_name = get_tvtk_name(name) tvtk_klass = getattr(tvtk, tvtk_name, None) + if on_gha: + print(f"{tvtk_name} ({name})") try: - obj = tvtk_klass() + tvtk_klass() # TypeError: super(type, obj): obj must be an instance or subtype of type except (TraitError, KeyError, TypeError): errors.append(f"\n{name}:\n{indent(traceback.format_exc(), ' ')}") + if on_gha: + print("\n::endgroup::") if len(errors) > 0: message = "Not all classes could be instantiated:\n{0}\n" raise AssertionError(message.format(''.join(errors))) From f716072ceaed7aadeff5ba543ebcb9699e695807 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:21:57 -0500 Subject: [PATCH 37/40] FIX: Fine --- .github/workflows/run-mayavi-tests.yml | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 7424f9921..807e72b8a 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -66,7 +66,7 @@ jobs: - python-version: '3.9' qt-api: 'pyqt5' os: ubuntu-latest - vtk: 'vtk~=9.1' + vtk: 'vtk==9.1.0' - python-version: '3.9' qt-api: 'pyqt5' os: ubuntu-latest @@ -87,8 +87,10 @@ jobs: steps: - uses: actions/checkout@v4 + with: + persist-credentials: false - name: Install Linux packages for Qt5/Qt6 support and start Xvfb - uses: pyvista/setup-headless-display-action@main + uses: pyvista/setup-headless-display-action@v3 with: qt: true if: startsWith(matrix.os, 'ubuntu') From 07a11fbf2e71811c3760a029a2d56c162bae2e78 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:26:34 -0500 Subject: [PATCH 38/40] FIX: Got another --- tvtk/tests/test_tvtk.py | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/tvtk/tests/test_tvtk.py b/tvtk/tests/test_tvtk.py index 6ea058628..8de94a5ab 100644 --- a/tvtk/tests/test_tvtk.py +++ b/tvtk/tests/test_tvtk.py @@ -828,7 +828,7 @@ def test_all_instantiable(self): tvtk_name = get_tvtk_name(name) tvtk_klass = getattr(tvtk, tvtk_name, None) if on_gha: - print(f"{tvtk_name} ({name})") + print(tvtk_name) try: tvtk_klass() # TypeError: super(type, obj): obj must be an instance or subtype of type @@ -865,6 +865,8 @@ def get_min_max_value(vtk_klass, vtk_attr_name): except AttributeError: return None, None + if on_gha: + print("\n::group::TVTK trait ranges") for name in self.names: vtk_klass = getattr(vtk, name) tvtk_klass_name = get_tvtk_name(name) @@ -875,11 +877,15 @@ def get_min_max_value(vtk_klass, vtk_attr_name): 'OpenGLRenderWindow', 'RenderWindow']: continue + + if on_gha: + print(tvtk_klass_name) + try: obj = getattr(tvtk, tvtk_klass_name)() except Exception: # testing for instantiation is above - pass + continue for trait_name in obj.editable_traits(): if trait_name in ['_in_set', '_vtk_obj']: From 932db26abf47424bcab22d31dbcdc59ed9c7dd09 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:29:48 -0500 Subject: [PATCH 39/40] FIX: Dont buffer --- .github/workflows/run-mayavi-tests.yml | 1 + tvtk/tests/test_tvtk.py | 4 +++- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 807e72b8a..2c9a7f7d0 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -84,6 +84,7 @@ jobs: QT_API: ${{ matrix.qt-api }} TVTK_VERBOSE: 'true' VTK_PARSER_VERBOSE: 'true' + PYTHONUNBUFFERED: '1' steps: - uses: actions/checkout@v4 diff --git a/tvtk/tests/test_tvtk.py b/tvtk/tests/test_tvtk.py index 8de94a5ab..8fe7f5b60 100644 --- a/tvtk/tests/test_tvtk.py +++ b/tvtk/tests/test_tvtk.py @@ -910,6 +910,8 @@ def get_min_max_value(vtk_klass, vtk_attr_name): setattr(obj, trait_name, (min_value-1, max_value)) with self.assertRaises(TraitError): setattr(obj, trait_name, (min_value, max_value+1)) + if on_gha: + print("::endgroup::") def test_no_trait_has_ptr_address_as_value(self): '''Test if none of the TVTK classes' traits has a value of "*_p_void" @@ -954,7 +956,7 @@ def test_all_traits_can_be_obtained(self): obj = getattr(tvtk, tvtk_klass_name)() except Exception: # testing for instantiation is above - pass + continue for trait_name in obj._full_traitnames_list_: try: From a48872ba308aedc49cddda2abfe7c21d2ae82793 Mon Sep 17 00:00:00 2001 From: Eric Larson Date: Wed, 18 Dec 2024 15:35:43 -0500 Subject: [PATCH 40/40] FIX: Skip for now --- .github/workflows/run-mayavi-tests.yml | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/.github/workflows/run-mayavi-tests.yml b/.github/workflows/run-mayavi-tests.yml index 2c9a7f7d0..af3465834 100644 --- a/.github/workflows/run-mayavi-tests.yml +++ b/.github/workflows/run-mayavi-tests.yml @@ -63,10 +63,11 @@ jobs: qt-api: 'pyqt5' os: ubuntu-latest vtk: 'vtk==9.2.2' - - python-version: '3.9' - qt-api: 'pyqt5' - os: ubuntu-latest - vtk: 'vtk==9.1.0' + # TVTK tests intermittently segfault on 9.1 + # - python-version: '3.9' + # qt-api: 'pyqt5' + # os: ubuntu-latest + # vtk: 'vtk==9.1.0' - python-version: '3.9' qt-api: 'pyqt5' os: ubuntu-latest