pUnit 1.2.26__tar.gz → 1.3.1__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (41) hide show
  1. {punit-1.2.26/src/pUnit.egg-info → punit-1.3.1}/PKG-INFO +1 -1
  2. {punit-1.2.26 → punit-1.3.1}/pyproject.toml +1 -1
  3. {punit-1.2.26 → punit-1.3.1/src/pUnit.egg-info}/PKG-INFO +1 -1
  4. {punit-1.2.26 → punit-1.3.1}/src/pUnit.egg-info/SOURCES.txt +1 -0
  5. {punit-1.2.26 → punit-1.3.1}/src/punit/TestResult.py +1 -1
  6. {punit-1.2.26 → punit-1.3.1}/src/punit/__init__.py +2 -2
  7. {punit-1.2.26 → punit-1.3.1}/src/punit/__main__.py +3 -1
  8. {punit-1.2.26 → punit-1.3.1}/src/punit/cli.py +1 -1
  9. {punit-1.2.26 → punit-1.3.1}/src/punit/reports/HtmlReportGenerator.py +7 -3
  10. punit-1.3.1/src/punit/reports/JsonReportGenerator.py +38 -0
  11. {punit-1.2.26 → punit-1.3.1}/src/punit/reports/__init__.py +8 -0
  12. {punit-1.2.26 → punit-1.3.1}/.scripts/punit +0 -0
  13. {punit-1.2.26 → punit-1.3.1}/LICENSE +0 -0
  14. {punit-1.2.26 → punit-1.3.1}/README.md +0 -0
  15. {punit-1.2.26 → punit-1.3.1}/setup.cfg +0 -0
  16. {punit-1.2.26 → punit-1.3.1}/src/pUnit.egg-info/dependency_links.txt +0 -0
  17. {punit-1.2.26 → punit-1.3.1}/src/pUnit.egg-info/requires.txt +0 -0
  18. {punit-1.2.26 → punit-1.3.1}/src/pUnit.egg-info/top_level.txt +0 -0
  19. {punit-1.2.26 → punit-1.3.1}/src/punit/assertions/__init__.py +0 -0
  20. {punit-1.2.26 → punit-1.3.1}/src/punit/assertions/collections.py +0 -0
  21. {punit-1.2.26 → punit-1.3.1}/src/punit/assertions/exceptions.py +0 -0
  22. {punit-1.2.26 → punit-1.3.1}/src/punit/assertions/strings.py +0 -0
  23. {punit-1.2.26 → punit-1.3.1}/src/punit/discovery/TestModuleDiscovery.py +0 -0
  24. {punit-1.2.26 → punit-1.3.1}/src/punit/discovery/__init__.py +0 -0
  25. {punit-1.2.26 → punit-1.3.1}/src/punit/facts/Fact.py +0 -0
  26. {punit-1.2.26 → punit-1.3.1}/src/punit/facts/FactManager.py +0 -0
  27. {punit-1.2.26 → punit-1.3.1}/src/punit/facts/__init__.py +0 -0
  28. {punit-1.2.26 → punit-1.3.1}/src/punit/filters/Filter.py +0 -0
  29. {punit-1.2.26 → punit-1.3.1}/src/punit/filters/FilterManager.py +0 -0
  30. {punit-1.2.26 → punit-1.3.1}/src/punit/filters/__init__.py +0 -0
  31. {punit-1.2.26 → punit-1.3.1}/src/punit/metadata/CallableMetadata.py +0 -0
  32. {punit-1.2.26 → punit-1.3.1}/src/punit/metadata/__init__.py +0 -0
  33. {punit-1.2.26 → punit-1.3.1}/src/punit/py.typed +0 -0
  34. {punit-1.2.26 → punit-1.3.1}/src/punit/reports/JUnitReportGenerator.py +0 -0
  35. {punit-1.2.26 → punit-1.3.1}/src/punit/runner.py +0 -0
  36. {punit-1.2.26 → punit-1.3.1}/src/punit/theories/Theory.py +0 -0
  37. {punit-1.2.26 → punit-1.3.1}/src/punit/theories/TheoryManager.py +0 -0
  38. {punit-1.2.26 → punit-1.3.1}/src/punit/theories/__init__.py +0 -0
  39. {punit-1.2.26 → punit-1.3.1}/src/punit/traits/Trait.py +0 -0
  40. {punit-1.2.26 → punit-1.3.1}/src/punit/traits/TraitManager.py +0 -0
  41. {punit-1.2.26 → punit-1.3.1}/src/punit/traits/__init__.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: pUnit
3
- Version: 1.2.26
3
+ Version: 1.3.1
4
4
  Summary: A modernized unit-test framework for Python.
5
5
  Author-email: Shaun Wilson <mrshaunwilson@msn.com>
6
6
  License: MIT License
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "pUnit"
3
- version = "1.2.26"
3
+ version = "1.3.1"
4
4
  description = "A modernized unit-test framework for Python."
5
5
  keywords = ["test", "unittest", "unit-test", "xUnit", "nUnit", "pytest"]
6
6
  authors = [
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: pUnit
3
- Version: 1.2.26
3
+ Version: 1.3.1
4
4
  Summary: A modernized unit-test framework for Python.
5
5
  Author-email: Shaun Wilson <mrshaunwilson@msn.com>
6
6
  License: MIT License
@@ -29,6 +29,7 @@ src/punit/metadata/CallableMetadata.py
29
29
  src/punit/metadata/__init__.py
30
30
  src/punit/reports/HtmlReportGenerator.py
31
31
  src/punit/reports/JUnitReportGenerator.py
32
+ src/punit/reports/JsonReportGenerator.py
32
33
  src/punit/reports/__init__.py
33
34
  src/punit/theories/Theory.py
34
35
  src/punit/theories/TheoryManager.py
@@ -60,7 +60,7 @@ class TestResult:
60
60
 
61
61
  @property
62
62
  def className(self) -> Optional[str]:
63
- return self.__className
63
+ return self.__className if self.__className is not None and len(self.__className) > 0 else None
64
64
 
65
65
  @className.setter
66
66
  def className(self, value:Optional[str]) -> None:
@@ -7,8 +7,8 @@ from .theories import *
7
7
  from .traits import *
8
8
 
9
9
 
10
- __version__ = '1.2.26'
11
- __commit__ = 'aeff39e'
10
+ __version__ = '1.3.1'
11
+ __commit__ = '4b5fbcb'
12
12
  __all__ = [
13
13
  '__version__', '__commit__',
14
14
  'assertions',
@@ -6,7 +6,7 @@ import os
6
6
  import time
7
7
  from .cli import CommandLineInterface
8
8
  from .discovery import TestModuleDiscovery
9
- from .reports import HtmlReportGenerator, JUnitReportGenerator
9
+ from .reports import HtmlReportGenerator, JUnitReportGenerator, JsonReportGenerator
10
10
  from .runner import TestRunner
11
11
 
12
12
 
@@ -42,6 +42,8 @@ async def async_main() -> None:
42
42
  report = HtmlReportGenerator().generate(results)
43
43
  case 'junit':
44
44
  report = JUnitReportGenerator().generate(results)
45
+ case 'json':
46
+ report = JsonReportGenerator().generate(results)
45
47
  if len(report) > 0:
46
48
  if cli.outputFilename is None:
47
49
  print(report)
@@ -95,7 +95,7 @@ class CommandLineInterface:
95
95
  extractReportFormat = False
96
96
  self.__reportFormat = arg.lower()
97
97
  match self.__reportFormat:
98
- case 'html' | 'junit':
98
+ case 'html' | 'junit' | 'json':
99
99
  pass
100
100
  case _:
101
101
  print(f'Unsupported value "{arg}" for --report argument, aborting.')
@@ -60,14 +60,18 @@ class HtmlReportGenerator:
60
60
  lines.append('</div>')
61
61
  lines.append('<div class="testresults-module">')
62
62
  lines.append(f'<h2 class="module-name">{testResult.packageName}/{testResult.moduleName}</h2>')
63
- currentModuleName = testResult.moduleName
63
+ currentModuleName = testResult.moduleName
64
64
  passfailstyle = '-pass' if testResult.isSuccess else '-fail'
65
65
  passfailglyph = '🟩' if testResult.isSuccess else '🟥'
66
66
  lines.append(f'<div class="testresult testresult{passfailstyle}">')
67
67
  lines.append('<div class="testresult-heading">')
68
68
  lines.append(f'<span class="glyph glyph{passfailstyle}">{passfailglyph}</span>')
69
- lines.append(f'<span class="test-class">{"" if testResult.className is None else testResult.className}</span>')
70
- lines.append(f'<span class="test-name">{testResult.testName}</span>')
69
+ className = "" if testResult.className is None else f"{testResult.className}."
70
+ data = testResult.properties.get('data')
71
+ if data is not None and len(data) > 0:
72
+ lines.append(f'<span class="test-class">{className}</span><span class="test-name">{testResult.testName}{data}</span>')
73
+ else:
74
+ lines.append(f'<span class="test-class">{className}</span><span class="test-name">{testResult.testName}</span>')
71
75
  lines.append(f'<span class="test-time test-time{passfailstyle}">{testResult.tookPretty}</span>')
72
76
  lines.append('</div>')
73
77
  if not testResult.isSuccess or testResult.stdout is not None or testResult.stderr is not None:
@@ -0,0 +1,38 @@
1
+ # SPDX-FileCopyrightText: © 2026 Shaun Wilson
2
+ # SPDX-License-Identifier: MIT
3
+
4
+ import json
5
+ import traceback
6
+ from typing import Any
7
+
8
+ from ..TestResult import TestResult
9
+
10
+
11
+
12
+ class JsonReportGenerator:
13
+
14
+ def generate(self, testResults:list[TestResult]) -> str:
15
+ testResults.sort(key=lambda e : e.moduleName)
16
+ results = list[dict[str, Any]]()
17
+ for testResult in testResults:
18
+ filterName:str = f'{testResult.packageName}/{testResult.moduleName}'
19
+ if testResult.className is not None:
20
+ filterName = f'{filterName}/{testResult.className}'
21
+ filterName = f'{filterName}/{testResult.testName}'
22
+ data = testResult.properties.get('data')
23
+ if data is not None and len(data) > 0:
24
+ datastr = f'({",".join([str(e) for e in data])})'
25
+ filterName = f'{filterName}{datastr}'
26
+ result = dict[str, Any]({
27
+ 'status': 'pass' if testResult.isSuccess else 'fail',
28
+ 'name': filterName,
29
+ })
30
+ if testResult.took is not None:
31
+ result['took'] = round(testResult.took * 1000, 3)
32
+ if not testResult.isSuccess:
33
+ if testResult.exception is not None:
34
+ result['message'] = f'{testResult.exception}\n{"".join(traceback.format_tb(testResult.exception.__traceback__))}'
35
+ else:
36
+ result['message'] = 'Unknown Error'
37
+ results.append(result)
38
+ return json.dumps(results)
@@ -3,3 +3,11 @@
3
3
 
4
4
  from .HtmlReportGenerator import HtmlReportGenerator
5
5
  from .JUnitReportGenerator import JUnitReportGenerator
6
+ from .JsonReportGenerator import JsonReportGenerator
7
+
8
+
9
+ __all__ = [
10
+ 'HtmlReportGenerator',
11
+ 'JUnitReportGenerator',
12
+ 'JsonReportGenerator'
13
+ ]
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes