aboutsummaryrefslogtreecommitdiffstats
path: root/src/py/imc/nonblock.py
blob: f68673d9f415b907d3be9b46591600669cc2577f (plain) (blame)
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
import types
from greenlet import greenlet

gr_waitmap = {}

gr_main = greenlet.getcurrent()

def current():
    return greenlet.getcurrent()

def switchtop():
    global gr_main

    return gr_main.switch(None)

def callee(f):
    def wrapper(*args,**kwargs):
        kwargs['_grid'] = str(id(greenlet.getcurrent()))
        return f(*args,**kwargs)

    return wrapper

def caller(f):
    def wrapper(*args,**kwargs):
        global gr_waitmap

        try:
            gr = greenlet(lambda *args,**kwargs : (str(id(greenlet.getcurrent())),f(*args,**kwargs)))
            grid = str(id(gr))
            gr_waitmap[grid] = gr

            result = gr.switch(*args,**kwargs)
            if result == None:
                return (False,None)

            ret_grid,ret = result
            del gr_waitmap[grid]

            if greenlet.getcurrent() == gr_main:
                return (False,None)

            else:
                while ret_grid != grid:
                    ret_grid,ret = gr_main.switch()

                return (True,ret)

        except Exception:
            return (False,'Einternal')

    return wrapper

def retcall(grid,value):
    global gr_waitmap

    try:
        gr = gr_waitmap[grid]
        gr.switch(value)

    except Exception:
        pass