From 3dd7eaa9c61c966d5837a4e6d17205001570318c Mon Sep 17 00:00:00 2001 From: Simon Cross Date: Fri, 14 Jun 2024 06:14:20 +0200 Subject: [PATCH 1/7] Add Python 3.12 to CI matrix. --- .github/workflows/tests.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 2037fa0..b138d65 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -7,7 +7,7 @@ jobs: runs-on: ubuntu-20.04 strategy: matrix: - python-version: [3.6, 3.7, 3.8, 3.9, "3.10", "3.11", pypy2, pypy3] + python-version: [3.6, 3.7, 3.8, 3.9, "3.10", "3.11", "3.12", pypy2, pypy3] steps: - uses: actions/checkout@v2 From 0cb928ebafa1615a8c495f425a92b98308b64ecc Mon Sep 17 00:00:00 2001 From: Simon Cross Date: Fri, 14 Jun 2024 06:20:05 +0200 Subject: [PATCH 2/7] Explicitly add setuptools as a build requirement. --- setup.cfg | 2 ++ 1 file changed, 2 insertions(+) diff --git a/setup.cfg b/setup.cfg index b22d3a0..0536af6 100644 --- a/setup.cfg +++ b/setup.cfg @@ -41,6 +41,8 @@ packages = genshi.template.tests.templates install_requires = six +setup_requires = + setuptools [options.entry_points] babel.extractors = From c327384d60dbf6a4e1d29cff5137c6baf0edf5e3 Mon Sep 17 00:00:00 2001 From: Simon Cross Date: Fri, 14 Jun 2024 06:23:11 +0200 Subject: [PATCH 3/7] Explicitly install setuptools. --- .github/workflows/tests.yml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index b138d65..bd91988 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -16,6 +16,10 @@ jobs: with: python-version: ${{ matrix.python-version }} + - name: Install setuptools + run: | + pip install setuptools + - name: Run test suite run: | python setup.py test From a190a6d0b3d3195d8a5707488a762c53eaa7e817 Mon Sep 17 00:00:00 2001 From: Simon Cross Date: Sat, 15 Jun 2024 20:43:03 +0200 Subject: [PATCH 4/7] Apply Fedora patch for using default test loader. --- genshi/filters/tests/i18n.py | 12 ++++++------ genshi/filters/tests/test_html.py | 4 ++-- genshi/filters/tests/transform.py | 2 +- genshi/template/tests/base.py | 2 +- genshi/template/tests/directives.py | 20 ++++++++++---------- genshi/template/tests/eval.py | 4 ++-- genshi/template/tests/interpolation.py | 2 +- genshi/template/tests/loader.py | 2 +- genshi/template/tests/markup.py | 2 +- genshi/template/tests/plugin.py | 4 ++-- genshi/template/tests/text.py | 4 ++-- genshi/tests/builder.py | 2 +- genshi/tests/core.py | 10 +++++----- genshi/tests/input.py | 4 ++-- genshi/tests/output.py | 8 ++++---- genshi/tests/path.py | 2 +- genshi/tests/util.py | 2 +- 17 files changed, 43 insertions(+), 43 deletions(-) diff --git a/genshi/filters/tests/i18n.py b/genshi/filters/tests/i18n.py index 18548f2..8b4ace0 100644 --- a/genshi/filters/tests/i18n.py +++ b/genshi/filters/tests/i18n.py @@ -2203,12 +2203,12 @@ def test_translate_context_with_msg(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest_suite(Translator.__module__)) - suite.addTest(unittest.makeSuite(TranslatorTestCase, 'test')) - suite.addTest(unittest.makeSuite(MsgDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(ChooseDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(DomainDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(ExtractTestCase, 'test')) - suite.addTest(unittest.makeSuite(ContextDirectiveTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(TranslatorTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(MsgDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ChooseDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(DomainDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ExtractTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ContextDirectiveTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/filters/tests/test_html.py b/genshi/filters/tests/test_html.py index a5fc71b..c356675 100644 --- a/genshi/filters/tests/test_html.py +++ b/genshi/filters/tests/test_html.py @@ -614,8 +614,8 @@ def test_sanitize_unicode_url(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest_suite(HTMLFormFiller.__module__)) - suite.addTest(unittest.makeSuite(HTMLFormFillerTestCase, 'test')) - suite.addTest(unittest.makeSuite(HTMLSanitizerTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(HTMLFormFillerTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(HTMLSanitizerTestCase)) return suite diff --git a/genshi/filters/tests/transform.py b/genshi/filters/tests/transform.py index 6201557..baa5ae5 100644 --- a/genshi/filters/tests/transform.py +++ b/genshi/filters/tests/transform.py @@ -1492,7 +1492,7 @@ def suite(): EmptyTest, RemoveTest, UnwrapText, WrapTest, FilterTest, MapTest, SubstituteTest, RenameTest, ReplaceTest, BeforeTest, AfterTest, PrependTest, AppendTest, AttrTest, CopyTest, CutTest): - suite.addTest(unittest.makeSuite(test, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(test)) suite.addTest(doctest_suite( genshi.filters.transform, optionflags=doctest.NORMALIZE_WHITESPACE, extraglobs={'HTML': HTML, 'tag': tag, 'Markup': Markup})) diff --git a/genshi/template/tests/base.py b/genshi/template/tests/base.py index e832f3a..39257df 100644 --- a/genshi/template/tests/base.py +++ b/genshi/template/tests/base.py @@ -35,7 +35,7 @@ def test_copy(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest.DocTestSuite(Template.__module__)) - suite.addTest(unittest.makeSuite(ContextTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ContextTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/template/tests/directives.py b/genshi/template/tests/directives.py index 5a17799..389f859 100644 --- a/genshi/template/tests/directives.py +++ b/genshi/template/tests/directives.py @@ -1217,16 +1217,16 @@ def test_with_empty_value(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest.DocTestSuite(directives)) - suite.addTest(unittest.makeSuite(AttrsDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(ChooseDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(DefDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(ForDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(IfDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(MatchDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(ContentDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(ReplaceDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(StripDirectiveTestCase, 'test')) - suite.addTest(unittest.makeSuite(WithDirectiveTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(AttrsDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ChooseDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(DefDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ForDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(IfDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(MatchDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ContentDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ReplaceDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(StripDirectiveTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(WithDirectiveTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/template/tests/eval.py b/genshi/template/tests/eval.py index 5dbb348..390cca5 100644 --- a/genshi/template/tests/eval.py +++ b/genshi/template/tests/eval.py @@ -1037,8 +1037,8 @@ def __init__(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest.DocTestSuite(Expression.__module__)) - suite.addTest(unittest.makeSuite(ExpressionTestCase, 'test')) - suite.addTest(unittest.makeSuite(SuiteTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ExpressionTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(SuiteTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/template/tests/interpolation.py b/genshi/template/tests/interpolation.py index 65be21d..575c71d 100644 --- a/genshi/template/tests/interpolation.py +++ b/genshi/template/tests/interpolation.py @@ -195,7 +195,7 @@ def test_interpolate_triplequoted(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest.DocTestSuite(interpolate.__module__)) - suite.addTest(unittest.makeSuite(InterpolateTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(InterpolateTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/template/tests/loader.py b/genshi/template/tests/loader.py index 1b9612b..aa842b5 100644 --- a/genshi/template/tests/loader.py +++ b/genshi/template/tests/loader.py @@ -522,7 +522,7 @@ def test_prefix_delegation_to_directories_with_subdirs(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest.DocTestSuite(TemplateLoader.__module__)) - suite.addTest(unittest.makeSuite(TemplateLoaderTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(TemplateLoaderTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/template/tests/markup.py b/genshi/template/tests/markup.py index 8c5be56..f10d548 100644 --- a/genshi/template/tests/markup.py +++ b/genshi/template/tests/markup.py @@ -802,7 +802,7 @@ def test_directive_single_line_with_translator(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest.DocTestSuite(MarkupTemplate.__module__)) - suite.addTest(unittest.makeSuite(MarkupTemplateTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(MarkupTemplateTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/template/tests/plugin.py b/genshi/template/tests/plugin.py index 93355e0..83e1990 100644 --- a/genshi/template/tests/plugin.py +++ b/genshi/template/tests/plugin.py @@ -256,8 +256,8 @@ def test_helper_functions(self): def suite(): suite = unittest.TestSuite() - suite.addTest(unittest.makeSuite(MarkupTemplateEnginePluginTestCase, 'test')) - suite.addTest(unittest.makeSuite(TextTemplateEnginePluginTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(MarkupTemplateEnginePluginTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(TextTemplateEnginePluginTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/template/tests/text.py b/genshi/template/tests/text.py index 3973df6..d49a079 100644 --- a/genshi/template/tests/text.py +++ b/genshi/template/tests/text.py @@ -295,8 +295,8 @@ def test_include_expr(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest.DocTestSuite(NewTextTemplate.__module__)) - suite.addTest(unittest.makeSuite(OldTextTemplateTestCase, 'test')) - suite.addTest(unittest.makeSuite(NewTextTemplateTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(OldTextTemplateTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(NewTextTemplateTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/tests/builder.py b/genshi/tests/builder.py index e322d9b..1859c94 100644 --- a/genshi/tests/builder.py +++ b/genshi/tests/builder.py @@ -67,7 +67,7 @@ def test_markup_escape(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest.DocTestSuite(Element.__module__)) - suite.addTest(unittest.makeSuite(ElementFactoryTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(ElementFactoryTestCase)) return suite diff --git a/genshi/tests/core.py b/genshi/tests/core.py index e7e7906..27d33aa 100644 --- a/genshi/tests/core.py +++ b/genshi/tests/core.py @@ -256,11 +256,11 @@ def test_curly_brace_equality(self): def suite(): suite = unittest.TestSuite() - suite.addTest(unittest.makeSuite(StreamTestCase, 'test')) - suite.addTest(unittest.makeSuite(MarkupTestCase, 'test')) - suite.addTest(unittest.makeSuite(NamespaceTestCase, 'test')) - suite.addTest(unittest.makeSuite(AttrsTestCase, 'test')) - suite.addTest(unittest.makeSuite(QNameTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(StreamTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(MarkupTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(NamespaceTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(AttrsTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(QNameTestCase)) suite.addTest(doctest_suite(core)) return suite diff --git a/genshi/tests/input.py b/genshi/tests/input.py index fbee840..cddbdd4 100644 --- a/genshi/tests/input.py +++ b/genshi/tests/input.py @@ -297,8 +297,8 @@ def test_convert_ElementTree_to_markup_stream(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest_suite(XMLParser.__module__)) - suite.addTest(unittest.makeSuite(XMLParserTestCase, 'test')) - suite.addTest(unittest.makeSuite(HTMLParserTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(XMLParserTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(HTMLParserTestCase)) return suite if __name__ == '__main__': diff --git a/genshi/tests/output.py b/genshi/tests/output.py index aa4c81a..29c4fe5 100644 --- a/genshi/tests/output.py +++ b/genshi/tests/output.py @@ -469,10 +469,10 @@ def test_elem_content(self): def suite(): suite = unittest.TestSuite() - suite.addTest(unittest.makeSuite(XMLSerializerTestCase, 'test')) - suite.addTest(unittest.makeSuite(XHTMLSerializerTestCase, 'test')) - suite.addTest(unittest.makeSuite(HTMLSerializerTestCase, 'test')) - suite.addTest(unittest.makeSuite(EmptyTagFilterTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(XMLSerializerTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(XHTMLSerializerTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(HTMLSerializerTestCase)) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(EmptyTagFilterTestCase)) suite.addTest(doctest_suite(XMLSerializer.__module__)) return suite diff --git a/genshi/tests/path.py b/genshi/tests/path.py index afa3a3d..b095261 100644 --- a/genshi/tests/path.py +++ b/genshi/tests/path.py @@ -709,7 +709,7 @@ def _test_eval(self, path, equiv=None, input=None, output='', def suite(): suite = unittest.TestSuite() suite.addTest(doctest_suite(Path.__module__)) - suite.addTest(unittest.makeSuite(PathTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(PathTestCase)) return suite diff --git a/genshi/tests/util.py b/genshi/tests/util.py index 4556d11..6a18046 100644 --- a/genshi/tests/util.py +++ b/genshi/tests/util.py @@ -87,7 +87,7 @@ def test_getitem(self): def suite(): suite = unittest.TestSuite() suite.addTest(doctest_suite(util)) - suite.addTest(unittest.makeSuite(LRUCacheTestCase, 'test')) + suite.addTest(unittest.defaultTestLoader.loadTestsFromTestCase(LRUCacheTestCase)) return suite if __name__ == '__main__': From dfbd962721e26750a7ed2ce08a5c726a5d086e9c Mon Sep 17 00:00:00 2001 From: Simon Cross Date: Sat, 15 Jun 2024 22:38:07 +0200 Subject: [PATCH 5/7] Fix handling of Ellipsis in AST for Python 3.12. --- genshi/compat.py | 5 ++++- genshi/template/astutil.py | 7 +++++-- 2 files changed, 9 insertions(+), 3 deletions(-) diff --git a/genshi/compat.py b/genshi/compat.py index 2abca13..0cd9324 100644 --- a/genshi/compat.py +++ b/genshi/compat.py @@ -142,10 +142,13 @@ def build_code_chunk(code, filename, name, lineno): warnings.filterwarnings('error', category=DeprecationWarning) try: _ast_Ellipsis = ast.Ellipsis + _ast_Ellipsis_value = lambda obj: ... _ast_Str = ast.Str _ast_Str_value = lambda obj: obj.s except (AttributeError, DeprecationWarning): - _ast_Ellipsis = _ast_Str = ast.Constant + _ast_Ellipsis = ast.Constant + _ast_Ellipsis_value = lambda obj: obj.value + _ast_Str = ast.Constant _ast_Str_value = lambda obj: obj.value class _DummyASTItem(object): diff --git a/genshi/template/astutil.py b/genshi/template/astutil.py index d5b33bd..87ab397 100644 --- a/genshi/template/astutil.py +++ b/genshi/template/astutil.py @@ -14,7 +14,7 @@ """Support classes for generating code from abstract syntax trees.""" from genshi.compat import ast as _ast, _ast_Constant, IS_PYTHON2, isstring, \ - _ast_Ellipsis + _ast_Ellipsis, _ast_Ellipsis_value __docformat__ = 'restructuredtext en' @@ -721,7 +721,10 @@ def visit_Subscript(self, node): self.visit(node.value) self._write('[') def _process_slice(node): - if isinstance(node, _ast_Ellipsis): + if ( + isinstance(node, _ast_Ellipsis) + and _ast_Ellipsis_value(node) == ... + ): self._write('...') elif isinstance(node, _ast.Slice): if getattr(node, 'lower', 'None'): From 5cbcf2c4f7ffd36f5b5366884848c2b539bdcc4f Mon Sep 17 00:00:00 2001 From: Simon Cross Date: Sat, 15 Jun 2024 22:46:13 +0200 Subject: [PATCH 6/7] Use Ellipsis constant for ... -- it is available in Python 2 and 3. --- genshi/compat.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/genshi/compat.py b/genshi/compat.py index 0cd9324..bfe0373 100644 --- a/genshi/compat.py +++ b/genshi/compat.py @@ -142,7 +142,7 @@ def build_code_chunk(code, filename, name, lineno): warnings.filterwarnings('error', category=DeprecationWarning) try: _ast_Ellipsis = ast.Ellipsis - _ast_Ellipsis_value = lambda obj: ... + _ast_Ellipsis_value = lambda obj: Ellipsis _ast_Str = ast.Str _ast_Str_value = lambda obj: obj.s except (AttributeError, DeprecationWarning): From 1e5c1b99eaba77eab6c4094feb751fda7d95a50c Mon Sep 17 00:00:00 2001 From: Simon Cross Date: Sat, 15 Jun 2024 22:47:45 +0200 Subject: [PATCH 7/7] Also use the Ellipsis constant when checking the node value. --- genshi/template/astutil.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/genshi/template/astutil.py b/genshi/template/astutil.py index 87ab397..c841aeb 100644 --- a/genshi/template/astutil.py +++ b/genshi/template/astutil.py @@ -723,7 +723,7 @@ def visit_Subscript(self, node): def _process_slice(node): if ( isinstance(node, _ast_Ellipsis) - and _ast_Ellipsis_value(node) == ... + and _ast_Ellipsis_value(node) == Ellipsis ): self._write('...') elif isinstance(node, _ast.Slice):