move virtualenv handling to pytestplugin

This commit is contained in:
holger krekel
2025-11-24 01:26:12 +01:00
parent 1d163fe0e0
commit 7ea50ca8e0
2 changed files with 94 additions and 71 deletions

View File

@@ -197,3 +197,85 @@ def log():
print(" " + msg)
return Printer()
#
# support for testing against different deltachat-rpc-server/clients
# installed into a temporary virtualenv and connected via 'execnet' channels
#
@pytest.fixture
def alice_and_remote_bob(tmp_path, acfactory):
"""return local Alice account, and a remote 'eval' function.
The 'eval' function allows to remote-execute arbitrary expressions
that can use the `bob` online account, and the `alice_contact`.
"""
import subprocess
import sys
import execnet
def factory(v):
alice = acfactory.get_online_account()
venv = tmp_path.joinpath("venv1")
python = sys.executable
subprocess.check_call([python, "-m", "venv", venv])
pip = venv.joinpath("bin", "pip")
subprocess.check_call([pip, "install", "pytest", f"deltachat-rpc-server=={v}", f"deltachat-rpc-client=={v}"])
python = venv.joinpath("bin", "python")
gw = execnet.makegateway(f"popen//python={python}")
accounts_dir = str(tmp_path.joinpath("account1_venv1"))
channel = gw.remote_exec(v220_loop)
cm = os.environ.get("CHATMAIL_DOMAIN")
channel.send((accounts_dir, str(venv.joinpath("bin", "deltachat-rpc-server")), cm))
sysinfo = channel.receive()
assert sysinfo == f"v{v}"
channel.send(alice.self_contact.make_vcard())
bob_vcard = channel.receive()
[alice_contact_bob] = alice.import_vcard(bob_vcard)
def eval(eval_str):
channel.send(eval_str)
return channel.receive()
return alice, alice_contact_bob, eval
return factory
def v220_loop(channel):
import os
import pathlib
from deltachat_rpc_client import DeltaChat, Rpc
from deltachat_rpc_client.pytestplugin import ACFactory
accounts_dir, rpc_server_path, chatmail_domain = channel.receive()
os.environ["CHATMAIL_DOMAIN"] = chatmail_domain
bin_path = str(pathlib.Path(rpc_server_path).parent)
os.environ["PATH"] = bin_path + ":" + os.environ["PATH"]
rpc = Rpc(accounts_dir=accounts_dir)
with rpc:
dc = DeltaChat(rpc)
channel.send(dc.rpc.get_system_info()["deltachat_core_version"])
acfactory = ACFactory(dc)
bob = acfactory.get_online_account()
alice_vcard = channel.receive()
[alice_contact] = bob.import_vcard(alice_vcard)
ns = {"bob": bob, "bob_contact_alice": alice_contact}
channel.send(bob.self_contact.make_vcard())
while 1:
eval_str = channel.receive()
res = eval(eval_str, ns)
try:
channel.send(res)
except Exception:
# some unserializable result
channel.send(None)

View File

@@ -1,83 +1,24 @@
import os
import subprocess
import sys
import execnet
def get_cross_v220(tmp_path, alice):
venv = tmp_path.joinpath("venv1")
python = sys.executable
subprocess.check_call([python, "-m", "venv", venv])
v = "2.20.0"
pip = venv.joinpath("bin", "pip")
subprocess.check_call([pip, "install", "pytest", f"deltachat-rpc-server=={v}", f"deltachat-rpc-client=={v}"])
python = venv.joinpath("bin", "python")
gw = execnet.makegateway(f"popen//python={python}")
accounts_dir = str(tmp_path.joinpath("account1_venv1"))
channel = gw.remote_exec(v220_loop)
channel.send((accounts_dir, str(venv.joinpath("bin", "deltachat-rpc-server")), os.environ.get("CHATMAIL_DOMAIN")))
sysinfo = channel.receive()
assert sysinfo == "v2.20.0"
channel.send(alice.self_contact.make_vcard())
def eval(eval_str):
channel.send(eval_str)
return channel.receive()
return eval
def v220_loop(channel):
import os
import pathlib
from deltachat_rpc_client import DeltaChat, Rpc
from deltachat_rpc_client.pytestplugin import ACFactory
accounts_dir, rpc_server_path, chatmail_domain = channel.receive()
os.environ["CHATMAIL_DOMAIN"] = chatmail_domain
bin_path = str(pathlib.Path(rpc_server_path).parent)
os.environ["PATH"] = bin_path + ":" + os.environ["PATH"]
rpc = Rpc(accounts_dir=accounts_dir)
with rpc:
dc = DeltaChat(rpc)
channel.send(dc.rpc.get_system_info()["deltachat_core_version"])
acfactory = ACFactory(dc)
bob = acfactory.get_online_account()
alice_vcard = channel.receive()
[alice_contact] = bob.import_vcard(alice_vcard)
ns = {"bob": bob, "alice_contact": alice_contact}
while 1:
eval_str = channel.receive()
res = eval(eval_str, ns)
try:
channel.send(res)
except Exception:
# some unserializable result
channel.send(None)
def test_qr_setup_contact(acfactory, tmp_path) -> None:
(alice,) = acfactory.get_online_accounts(1)
remote_eval = get_cross_v220(tmp_path, alice)
def test_qr_setup_contact(alice_and_remote_bob) -> None:
alice, alice_contact_bob, remote_eval = alice_and_remote_bob("2.20.0")
qr_code = alice.get_qr_code()
remote_eval(f"bob.secure_join({qr_code!r})")
alice.wait_for_securejoin_inviter_success()
# Test that Alice verified Bob's profile.
vcard = remote_eval("bob.self_contact.make_vcard()")
[alice_contact_bob] = alice.import_vcard(vcard)
alice_contact_bob_snapshot = alice_contact_bob.get_snapshot()
assert alice_contact_bob_snapshot.is_verified
remote_eval("bob.wait_for_securejoin_joiner_success()")
# Test that Bob verified Alice's profile.
assert remote_eval("alice_contact.get_snapshot().is_verified")
assert remote_eval("bob_contact_alice.get_snapshot().is_verified")
def test_send_and_receive_message(alice_and_remote_bob) -> None:
alice, alice_contact_bob, remote_eval = alice_and_remote_bob("2.20.0")
remote_eval("bob_contact_alice.create_chat().send_text('hello')")
msg = alice.wait_for_incoming_msg()
assert msg.get_snapshot().text == "hello"