Skip to content

Commit e5eaa8a

Browse files
author
deathaxe
authored
Fix plugin reloading before test start (#258)
This commit ... 1. refactors reloader's dummy plugin install/removal by using set_timeout() to check for dummy being loaded in order to not block the calling thread. 2. uses an `on_done` callback to continue operation after dummy has been removed. 3. always loads a dummy plugin as it is required to teach ST about new command and event instances. It does not work without that step. Before this commit the whole UI thread was blocked, causing dummy plugin not being reloaded until timeout was hit. Running tests on worker thread had other negative impacts. Reloading now behaves like "DefferableTextTestRunner" using set_timeout to schedule coroutines.
1 parent da72e65 commit e5eaa8a

File tree

5 files changed

+27
-80
lines changed

5 files changed

+27
-80
lines changed

README.md

-1
Original file line numberDiff line numberDiff line change
@@ -140,7 +140,6 @@ UnitTesting could be configured by providing the following settings in `unittest
140140
| verbosity | verbosity level | 2 |
141141
| capture_console | capture stdout and stderr in the test output | false |
142142
| reload_package_on_testing | reloading package will increase coverage rate | true |
143-
| show_reload_progress | print a detailed list of reloaded modules to console | false |
144143
| coverage | track test case coverage | false |
145144
| coverage_on_worker_thread | (experimental) | false |
146145
| generate_html_report | generate HTML report for coverage | false |

sublime-package.json

-9
Original file line numberDiff line numberDiff line change
@@ -35,10 +35,6 @@
3535
"type": ["string", "null"],
3636
"markdownDescription": "Name of the test output instead of showing in the panel.",
3737
},
38-
"show_reload_progress": {
39-
"type": "boolean",
40-
"default": true,
41-
},
4238
"reload_package_on_testing": {
4339
"type": "boolean",
4440
"default": true,
@@ -49,11 +45,6 @@
4945
"default": false,
5046
"markdownDescription": "Create coverage report.",
5147
},
52-
"start_coverage_after_reload": {
53-
"type": "boolean",
54-
"default": false,
55-
"markdownDescription": "Irrelevent if `reload_package_on_testing` is false.",
56-
},
5748
"coverage_on_worker_thread": {
5849
"type": "boolean",
5950
"default": false,

unittesting/base.py

-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@
2424
"capture_console": False,
2525
# reloader
2626
"reload_package_on_testing": True,
27-
"show_reload_progress": False,
2827
# coverage
2928
"coverage": False,
3029
"coverage_on_worker_thread": False, # experimental

unittesting/unit.py

+11-10
Original file line numberDiff line numberDiff line change
@@ -94,17 +94,20 @@ def run(self, package=None, **kwargs):
9494

9595
stream = self.load_stream(package, settings)
9696

97-
if settings["async"]:
98-
threading.Thread(
99-
target=self.run_coverage, args=(package, stream, settings)
100-
).start()
101-
else:
102-
self.run_coverage(package, stream, settings)
97+
def run_tests():
98+
if settings["async"]:
99+
threading.Thread(
100+
target=self.run_coverage, args=(package, stream, settings)
101+
).start()
102+
else:
103+
self.run_coverage(package, stream, settings)
103104

104-
def run_coverage(self, package, stream, settings):
105105
if settings["reload_package_on_testing"]:
106-
reload_package(package)
106+
reload_package(package, on_done=run_tests)
107+
else:
108+
run_tests()
107109

110+
def run_coverage(self, package, stream, settings):
108111
if not coverage or not settings["coverage"]:
109112
if settings["coverage"]:
110113
stream.write("Warning: coverage cannot be loaded.\n\n")
@@ -139,8 +142,6 @@ def run_coverage(self, package, stream, settings):
139142
cov.start()
140143

141144
if settings["coverage_on_worker_thread"]:
142-
import threading
143-
144145
original_set_timeout_async = sublime.set_timeout_async
145146

146147
def set_timeout_async(callback, *args, **kwargs):

unittesting/utils/reloader.py

+16-59
Original file line numberDiff line numberDiff line change
@@ -5,16 +5,6 @@
55
import sublime
66
import sublime_plugin
77
import sys
8-
import threading
9-
10-
11-
def dprint(*args, fill=None, fill_width=60, **kwargs):
12-
if fill is not None:
13-
sep = str(kwargs.get('sep', ' '))
14-
caption = sep.join(args)
15-
args = "{0:{fill}<{width}}".format(caption and caption + sep,
16-
fill=fill, width=fill_width),
17-
print("UnitTesting:", *args, **kwargs)
188

199

2010
def path_contains(a, b):
@@ -61,8 +51,10 @@ def package_plugins(pkg_name):
6151
]
6252

6353

64-
def reload_package(pkg_name, dummy=True, verbose=True):
54+
def reload_package(pkg_name, on_done=None):
6555
if pkg_name not in sys.modules:
56+
if on_done:
57+
sublime.set_timeout(on_done)
6658
return
6759

6860
all_modules = {
@@ -89,21 +81,8 @@ def reload_package(pkg_name, dummy=True, verbose=True):
8981
for plugin in plugins:
9082
sublime_plugin.reload_plugin(plugin)
9183

92-
# Install and uninstall a dummy package so ST updates
93-
# command and event listener bindings
94-
if dummy:
95-
load_dummy(verbose)
96-
97-
98-
def load_dummy(verbose):
99-
"""
100-
Hack to trigger automatic "reloading plugins".
101-
102-
This is needed to ensure TextCommand's and WindowCommand's are ready.
103-
"""
104-
if verbose:
105-
dprint("installing dummy package")
106-
84+
# Hack to trigger automatic "reloading plugins".
85+
# This is needed to ensure TextCommand's and WindowCommand's are ready.
10786
if sys.version_info >= (3, 8):
10887
# in ST 4, User package is always loaded in python 3.8
10988
dummy_name = "User._dummy"
@@ -113,37 +92,15 @@ def load_dummy(verbose):
11392
dummy_name = "_dummy"
11493
dummy_py = os.path.join(sublime.packages_path(), "_dummy.py")
11594

116-
with open(dummy_py, "w"):
117-
pass
118-
119-
def remove_dummy(trial=0):
120-
if dummy_name in sys.modules:
121-
if verbose:
122-
dprint("removing dummy package")
123-
try:
124-
os.unlink(dummy_py)
125-
except FileNotFoundError:
126-
pass
127-
after_remove_dummy()
128-
elif trial < 300:
129-
threading.Timer(0.1, lambda: remove_dummy(trial + 1)).start()
130-
else:
131-
try:
132-
os.unlink(dummy_py)
133-
except FileNotFoundError:
134-
pass
135-
136-
condition = threading.Condition()
137-
138-
def after_remove_dummy(trial=0):
95+
open(dummy_py, "a").close()
96+
97+
def check_loaded():
13998
if dummy_name not in sys.modules:
140-
condition.acquire()
141-
condition.notify()
142-
condition.release()
143-
elif trial < 300:
144-
threading.Timer(0.1, lambda: after_remove_dummy(trial + 1)).start()
145-
146-
threading.Timer(0.1, remove_dummy).start()
147-
condition.acquire()
148-
condition.wait(30) # 30 seconds should be enough for all regular usages
149-
condition.release()
99+
sublime.set_timeout(check_loaded, 100)
100+
return
101+
102+
os.remove(dummy_py)
103+
if on_done:
104+
sublime.set_timeout(on_done, 200)
105+
106+
sublime.set_timeout(check_loaded, 100)

0 commit comments

Comments
 (0)