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
|
import os
import threading
import unittest
from datetime import timedelta
from time import sleep
import pytest
from reactivex.internal.basic import default_now
from reactivex.scheduler.mainloop import GtkScheduler
gi = pytest.importorskip("gi")
from gi.repository import GLib, Gtk # isort: skip
gi.require_version("Gtk", "3.0")
# Removing GNOME_DESKTOP_SESSION_ID from environment
# prevents QtScheduler test from failing with message
# Gtk-ERROR **: GTK+ 2.x symbols detected.
# Using GTK+ 2.x and GTK+ 3 in the same process is not supported
if "GNOME_DESKTOP_SESSION_ID" in os.environ:
del os.environ["GNOME_DESKTOP_SESSION_ID"]
class TestGtkScheduler(unittest.TestCase):
def test_gtk_schedule_now(self):
scheduler = GtkScheduler(GLib)
diff = scheduler.now - default_now()
assert abs(diff) < timedelta(milliseconds=1)
def test_gtk_schedule_now_units(self):
scheduler = GtkScheduler(GLib)
diff = scheduler.now
sleep(0.1)
diff = scheduler.now - diff
assert timedelta(milliseconds=80) < diff < timedelta(milliseconds=180)
def test_gtk_schedule_action(self):
scheduler = GtkScheduler(GLib)
gate = threading.Semaphore(0)
ran = False
def action(scheduler, state):
nonlocal ran
ran = True
scheduler.schedule(action)
def done(data):
Gtk.main_quit()
gate.release()
return False
GLib.timeout_add(50, done, None)
Gtk.main()
gate.acquire()
assert ran is True
def test_gtk_schedule_action_relative(self):
scheduler = GtkScheduler(GLib)
gate = threading.Semaphore(0)
starttime = default_now()
endtime = None
def action(scheduler, state):
nonlocal endtime
endtime = default_now()
scheduler.schedule_relative(0.1, action)
def done(data):
Gtk.main_quit()
gate.release()
return False
GLib.timeout_add(200, done, None)
Gtk.main()
gate.acquire()
assert endtime is not None
diff = endtime - starttime
assert diff > timedelta(milliseconds=80)
def test_gtk_schedule_action_absolute(self):
scheduler = GtkScheduler(GLib)
gate = threading.Semaphore(0)
starttime = default_now()
endtime = None
def action(scheduler, state):
nonlocal endtime
endtime = default_now()
due = scheduler.now + timedelta(milliseconds=100)
scheduler.schedule_absolute(due, action)
def done(data):
Gtk.main_quit()
gate.release()
return False
GLib.timeout_add(200, done, None)
Gtk.main()
gate.acquire()
assert endtime is not None
diff = endtime - starttime
assert diff > timedelta(milliseconds=80)
def test_gtk_schedule_action_cancel(self):
ran = False
scheduler = GtkScheduler(GLib)
gate = threading.Semaphore(0)
def action(scheduler, state):
nonlocal ran
ran = True
d = scheduler.schedule_relative(0.1, action)
d.dispose()
def done(data):
Gtk.main_quit()
gate.release()
return False
GLib.timeout_add(200, done, None)
Gtk.main()
gate.acquire()
assert ran is False
def test_gtk_schedule_action_periodic(self):
scheduler = GtkScheduler(GLib)
gate = threading.Semaphore(0)
period = 0.05
counter = 3
def action(state):
nonlocal counter
if state:
counter -= 1
return state - 1
scheduler.schedule_periodic(period, action, counter)
def done(data):
Gtk.main_quit()
gate.release()
return False
GLib.timeout_add(300, done, None)
Gtk.main()
gate.acquire()
assert counter == 0
|