-
Notifications
You must be signed in to change notification settings - Fork 67
/
Copy pathtest_all.py
169 lines (140 loc) · 4.32 KB
/
test_all.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
import pytest
def test_import():
import juliacall
juliacall
def test_newmodule():
import juliacall
jl = juliacall.Main
m = juliacall.newmodule("TestModule")
assert isinstance(m, juliacall.ModuleValue)
assert jl.isa(m, jl.Module)
assert str(jl.nameof(m)) == "TestModule"
def test_convert():
import juliacall
jl = juliacall.Main
for x, t in [(None, jl.Nothing), (True, jl.Bool), ([1, 2, 3], jl.Vector)]:
y = juliacall.convert(t, x)
assert isinstance(y, juliacall.AnyValue)
assert jl.isa(y, t)
def test_interactive():
import juliacall
juliacall.interactive(True)
juliacall.interactive(False)
def test_JuliaError():
import juliacall
jl = juliacall.Main
assert isinstance(juliacall.JuliaError, type)
assert issubclass(juliacall.JuliaError, Exception)
try:
juliacall.Base.error("test error")
err = None
except juliacall.JuliaError as e:
err = e
assert err is not None
assert isinstance(err, juliacall.JuliaError)
exc = err.exception
assert jl.isa(exc, jl.ErrorException)
assert str(exc.msg) == "test error"
bt = err.backtrace
assert bt is not None
def test_issue_394():
"https://github.com/JuliaPy/PythonCall.jl/issues/394"
from juliacall import Main as jl
x = 3
f = lambda x: x + 1
y = 5
jl.x = x
assert jl.x is x
jl.f = f
assert jl.f is f
jl.y = y
assert jl.y is y
assert jl.x is x
assert jl.f is f
assert jl.y is y
assert jl.seval("f(x)") == 4
def test_issue_433():
"https://github.com/JuliaPy/PythonCall.jl/issues/433"
from juliacall import Main as jl
# Smoke test
jl.seval("x=1\nx=1")
assert jl.x == 1
# Do multiple things
out = jl.seval(
"""
function _issue_433_g(x)
return x^2
end
_issue_433_g(5)
"""
)
assert out == 25
def test_julia_gc():
from juliacall import Main as jl
# We make a bunch of python objects with no reference to them,
# then call GC to try to finalize them.
# We want to make sure we don't segfault.
# We also programmatically check things are working by verifying the queue is empty.
# Debugging note: if you get segfaults, then run the tests with
# `PYTHON_JULIACALL_HANDLE_SIGNALS=yes python3 -X faulthandler -m pytest -p no:faulthandler -s --nbval --cov=pysrc ./pytest/`
# in order to recover a bit more information from the segfault.
jl.seval(
"""
using PythonCall, Test
let
pyobjs = map(pylist, 1:100)
Threads.@threads for obj in pyobjs
finalize(obj)
end
end
GC.gc()
@test isempty(PythonCall.GC.QUEUE.items)
"""
)
@pytest.mark.parametrize(
["yld", "raw"], [(yld, raw) for yld in [False, True] for raw in [False, True]]
)
def test_call_nogil(yld, raw):
"""Tests that we can execute Julia code in parallel by releasing the GIL."""
from concurrent.futures import ThreadPoolExecutor, wait
from time import time
from juliacall import Main as jl
# julia implementation of sleep which unlocks the GIL
if yld:
# use sleep, which yields
jsleep = jl.sleep
else:
# use Libc.systemsleep which does not yield
jsleep = jl.Libc.systemsleep
if raw:
# test RawValue instead of AnyValue
jsleep = jsleep._jl_raw()
jsleep = jsleep._jl_call_nogil
jyield = getattr(jl, "yield")
# precompile
jsleep(0.01)
jyield()
# use two threads
pool = ThreadPoolExecutor(2)
# run jsleep(1) twice concurrently
t0 = time()
fs = [pool.submit(jsleep, 1) for _ in range(2)]
# submitting tasks should be very fast
t1 = time() - t0
assert t1 < 0.1
# wait for the tasks to finish
if yld:
# we need to explicitly yield back to give the Julia scheduler a chance to
# finish the sleep calls, so we yield every 0.1 seconds
status = wait(fs, timeout=0.1)
t2 = time() - t0
while status.not_done:
jyield()
status = wait(fs, timeout=0.1)
t2 = time() - t0
assert t2 < 2.0
else:
wait(fs)
t2 = time() - t0
# executing the tasks should take about 1 second because they happen in parallel
assert 0.9 < t2 < 1.5