Skip to content

Commit 73ccb32

Browse files
author
deathaxe
authored
Refactor self tests (#244)
This commit... 1. fixes `sublime.package_path()` called on import time, causing invalid or unexpected output paths being used. 2. use `setUp()` and `tearDow()` methods to setup and cleanup packages used for self tests in a batch per `TestCase` in order to prevent or reduce risk of possible race conditions or failing API calls due to too frequent package reloading or removing events. 3. optimize availability check for ColorSchemeUnit package
1 parent 38ff3f1 commit 73ccb32

File tree

1 file changed

+79
-59
lines changed

1 file changed

+79
-59
lines changed

tests/test_3141596.py

+79-59
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import os
22
import re
33
import shutil
4-
from functools import partial, wraps
4+
from functools import wraps
55
from unittest import skipIf
66
from unittesting.utils import isiterable
77
from unittesting import DeferrableTestCase
@@ -11,75 +11,74 @@
1111
import sublime
1212

1313
BASEDIR = os.path.dirname(os.path.abspath(__file__))
14-
UUTDIR = os.path.join(
15-
sublime.packages_path(), 'User', 'UnitTesting')
1614

1715

18-
def set_package(package):
16+
def setup_package(package):
17+
packages_path = sublime.packages_path()
18+
package_path = os.path.join(packages_path, package)
1919
try:
20-
shutil.rmtree(os.path.join(sublime.packages_path(), package))
21-
except Exception:
20+
shutil.rmtree(package_path)
21+
except FileNotFoundError:
2222
pass
2323
try:
24-
shutil.copytree(
25-
os.path.join(BASEDIR, package),
26-
os.path.join(sublime.packages_path(), package))
27-
except Exception:
24+
shutil.copytree(os.path.join(BASEDIR, package), package_path)
25+
except FileNotFoundError:
2826
pass
2927
try:
30-
shutil.rmtree(os.path.join(UUTDIR, package))
31-
except Exception:
28+
shutil.rmtree(os.path.join(packages_path, "User", "UnitTesting", package))
29+
except FileNotFoundError:
3230
pass
3331

3432

3533
def cleanup_package(package):
3634
try:
3735
shutil.rmtree(os.path.join(sublime.packages_path(), package))
38-
except Exception:
36+
except FileNotFoundError:
3937
pass
4038

4139

42-
def prepare_package(package, output=None, syntax_test=False, syntax_compatibility=False,
43-
color_scheme_test=False, delay=1000, wait_timeout=5000):
40+
def with_package(package, output=None, syntax_test=False, syntax_compatibility=False,
41+
color_scheme_test=False, wait_timeout=5000):
4442
def wrapper(func):
4543
@wraps(func)
4644
def real_wrapper(self):
47-
set_package(package)
45+
packages_path = sublime.packages_path()
46+
4847
if output:
49-
# set by _Ooutput/unittesting.json
48+
# set by _Output/unittesting.json
5049
outfile = None
51-
result_file = os.path.join(sublime.packages_path(), package, output)
50+
result_file = os.path.join(packages_path, package, output)
5251
else:
53-
outfiledir = os.path.join(UUTDIR, package)
54-
outfile = os.path.join(outfiledir, "result")
52+
outfile = os.path.join(packages_path, "User", "UnitTesting", package, "result")
5553
result_file = outfile
56-
os.makedirs(outfiledir, exist_ok=True)
5754

58-
yield delay
5955
yield AWAIT_WORKER
6056

61-
args = {"package": package}
57+
kwargs = {"package": package}
6258
if outfile:
63-
# Command args have the highest precedence. Passing down
59+
# Command kwargs have the highest precedence. Passing down
6460
# 'None' is not what we want, the intention is to omit it
6561
# so that the value from 'unittesting.json' wins.
66-
args["output"] = outfile
62+
kwargs["output"] = outfile
6763

6864
if syntax_test:
69-
sublime.run_command("unit_testing_syntax", args)
65+
sublime.run_command("unit_testing_syntax", kwargs)
7066
elif syntax_compatibility:
71-
sublime.run_command("unit_testing_syntax_compatibility", args)
67+
sublime.run_command("unit_testing_syntax_compatibility", kwargs)
7268
elif color_scheme_test:
73-
sublime.run_command("unit_testing_color_scheme", args)
69+
sublime.run_command("unit_testing_color_scheme", kwargs)
7470
else:
75-
sublime.run_command("unit_testing", args)
71+
sublime.run_command("unit_testing", kwargs)
7672

77-
def condition(result_file):
78-
with open(result_file, 'r') as f:
79-
txt = f.read()
80-
return "UnitTesting: Done." in txt
73+
def condition():
74+
try:
75+
with open(result_file, 'r') as f:
76+
txt = f.read()
77+
return "UnitTesting: Done." in txt
78+
except FileNotFoundError:
79+
return False
8180

82-
yield {"condition": partial(condition, result_file), "timeout": wait_timeout}
81+
yield {"condition": condition, "period": 200, "timeout": wait_timeout}
8382

8483
with open(result_file, 'r') as f:
8584
txt = f.read()
@@ -88,14 +87,24 @@ def condition(result_file):
8887
if isiterable(deferred):
8988
yield from deferred
9089

91-
cleanup_package(package)
92-
9390
yield
91+
9492
return real_wrapper
93+
9594
return wrapper
9695

9796

9897
class UnitTestingTestCase(DeferrableTestCase):
98+
fixtures = ()
99+
100+
def setUp(self):
101+
for fixture in self.fixtures:
102+
setup_package(fixture)
103+
yield 500
104+
105+
def tearDown(self):
106+
for fixture in self.fixtures:
107+
cleanup_package(fixture)
99108

100109
def assertRegexContains(self, txt, expr, msg=None):
101110
if re.search(expr, txt, re.MULTILINE) is None:
@@ -108,75 +117,85 @@ def assertOk(self, txt, msg=None):
108117

109118

110119
class TestUnitTesting(UnitTestingTestCase):
120+
fixtures = (
121+
"_Success", "_Failure", "_Empty", "_Output", "_Deferred", "_Async"
122+
)
111123

112-
@prepare_package("_Success")
124+
@with_package("_Success")
113125
def test_success(self, txt):
114126
self.assertOk(txt)
115127

116-
@prepare_package("_Failure")
128+
@with_package("_Failure")
117129
def test_failure(self, txt):
118130
self.assertRegexContains(txt, r'^FAILED \(failures=1\)')
119131

120-
@prepare_package("_Error")
132+
@with_package("_Error")
121133
def test_error(self, txt):
122134
self.assertRegexContains(txt, r'^ERROR')
123135

124-
@prepare_package("_Empty")
136+
@with_package("_Empty")
125137
def test_empty(self, txt):
126138
self.assertRegexContains(txt, r'^No tests are found.')
127139

128-
@prepare_package("_Output", "tests/result")
140+
@with_package("_Output", "tests/result")
129141
def test_output(self, txt):
130142
self.assertOk(txt)
131143

132-
@prepare_package("_Deferred")
144+
@with_package("_Deferred")
133145
def test_deferred(self, txt):
134146
self.assertOk(txt)
135147

136-
@prepare_package("_Async")
148+
@with_package("_Async")
137149
def test_async(self, txt):
138150
self.assertOk(txt)
139151

140152

141153
class TestSyntax(UnitTestingTestCase):
142-
143-
@prepare_package("_Syntax_Failure", syntax_test=True)
154+
fixtures = (
155+
"_Syntax_Failure",
156+
"_Syntax_Success",
157+
"_Syntax_Compat_Failure",
158+
"_Syntax_Compat_Success",
159+
)
160+
161+
@with_package("_Syntax_Failure", syntax_test=True)
144162
def test_fail_syntax(self, txt):
145163
self.assertRegexContains(txt, r'^FAILED: 1 of 21 assertions in 1 file failed$')
146164

147-
@prepare_package("_Syntax_Success", syntax_test=True)
165+
@with_package("_Syntax_Success", syntax_test=True)
148166
def test_success_syntax(self, txt):
149167
self.assertOk(txt)
150168

151-
@prepare_package("_Syntax_Error", syntax_test=True)
169+
@with_package("_Syntax_Error", syntax_test=True)
152170
def test_error_syntax(self, txt):
153171
self.assertRegexContains(txt, r'^ERROR: No syntax_test')
154172

155-
@prepare_package("_Syntax_Compat_Failure", syntax_compatibility=True)
173+
@with_package("_Syntax_Compat_Failure", syntax_compatibility=True)
156174
def test_fail_syntax_compatibility(self, txt):
157175
self.assertRegexContains(txt, r'^FAILED: 3 errors in 1 of 1 syntax$')
158176

159-
@prepare_package("_Syntax_Compat_Success", syntax_compatibility=True)
177+
@with_package("_Syntax_Compat_Success", syntax_compatibility=True)
160178
def test_success_syntax_compatibility(self, txt):
161179
self.assertOk(txt)
162180

163181

164182
def has_colorschemeunit():
165-
if "ColorSchemeUnit.sublime-package" in os.listdir(sublime.installed_packages_path()):
166-
return True
167-
elif "ColorSchemeUnit" in os.listdir(sublime.packages_path()):
168-
return True
169-
return False
183+
return (
184+
os.path.isfile(os.path.join(sublime.installed_packages_path(), "ColorSchemeUnit.sublime-package")) or
185+
os.path.isdir(os.path.join(sublime.packages_path(), "ColorSchemeUnit"))
186+
)
170187

171188

172189
class TestColorScheme(UnitTestingTestCase):
190+
fixtures = ("_ColorScheme_Failure", "_ColorScheme_Success")
191+
173192
@skipIf(not has_colorschemeunit(), "ColorSchemeUnit is not installed")
174-
@prepare_package("_ColorScheme_Failure", color_scheme_test=True)
193+
@with_package("_ColorScheme_Failure", color_scheme_test=True)
175194
def test_fail_color_scheme(self, txt):
176195
self.assertRegexContains(txt, r'^There were 14 failures:$')
177196

178197
@skipIf(not has_colorschemeunit(), "ColorSchemeUnit is not installed")
179-
@prepare_package("_ColorScheme_Success", color_scheme_test=True)
198+
@with_package("_ColorScheme_Success", color_scheme_test=True)
180199
def test_success_color_scheme(self, txt):
181200
self.assertOk(txt)
182201

@@ -188,9 +207,10 @@ def tidy_path(path):
188207
class TestTempDirectoryTestCase(TempDirectoryTestCase):
189208

190209
def test_temp_dir(self):
191-
self.assertTrue(tidy_path(
192-
self._temp_dir),
193-
tidy_path(self.window.folders()[0]))
210+
self.assertTrue(
211+
tidy_path(self._temp_dir),
212+
tidy_path(self.window.folders()[0])
213+
)
194214

195215

196216
class TestViewTestCase(ViewTestCase):

0 commit comments

Comments
 (0)