### py_interface -- A Python-implementation of an Erlang node
###
### $Id: erl_eventhandler.py,v 1.7 2006/07/16 23:38:59 tab Exp $
###
### Copyright (C) 2002  Tomas Abrahamsson
###
### Author: Tomas Abrahamsson <tab@lysator.liu.se>
### 
### This file is part of the Py-Interface library
###
### This library is free software; you can redistribute it and/or
### modify it under the terms of the GNU Library General Public
### License as published by the Free Software Foundation; either
### version 2 of the License, or (at your option) any later version.
### 
### This library is distributed in the hope that it will be useful,
### but WITHOUT ANY WARRANTY; without even the implied warranty of
### MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
### Library General Public License for more details.
### 
### You should have received a copy of the GNU Library General Public
### License along with this library; if not, write to the Free
### Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA

### erl_eventhandler.py -- A general event handler.
###                        It provides the possibility to register
###                        callbacks to be called at certain events:
###                        * when there is data to read on a file descriptor
###                        * when it's posssible to write to a file descriptor
###                        * at file descriptor exceptions
###                        * after timeout



import os
import sys
import string
import select
import time
from Tkinter import tkinter
import Tkinter

import erl_common

_evhandler = None

### ---------------------------------------------------
### API
###

def GetEventHandler():
    """Retrieve the eventhandler.
    No arguments.
    Returns: <instance of the EventHandler singleton>
    Throws:  nothing
    """
    global _evhandler

    if _evhandler == None:
        ## This is the first call to the event handler. It has not been
        ## created yet, so create it first.
        _evhandler = _EventHandler()
    return _evhandler

def SetEventHandlerStateTk(top):
    """Sets up the event handler to use Tkinter's mainloop.

    TOP = <instance of Tk>

    Returns: void
    Throws:  nothing
    
    Call this when you want to use the event handler in a program that uses
    Tkinter. The argument is expected to be the object that's returned by
    a call to Tkinter.Tk().
    """
    evhandler = GetEventHandler()
    evhandler._SetStateTk(top)

def SetEventHandlerStateStandalone():
    """Sets up the event handler to use its own mainloop. This is the default.

    No arguments
    Returns: void
    Throws:  nothing
    """
    evhandler = GetEventHandler()
    evhandler._SetStateStandAlone()

###
### End of API
### ---------------------------------------------------

class EVCallback:
    """This class is intended to be used by the _EventHandler class.
    It differs from erl_common.VCallback in that it does not
    tack on any extra arguments to the front. This is because tkinter
    sends some extra arguments to the callback, so the arguments to
    callbacks would have been depended on whether the event handler
    is in tkinter-state or not.
    """
    def __init__(self, callback, optArgs, namedArgs):
        self.callback = callback
        self.optArgs = optArgs
        self.namedArgs = namedArgs

    def __call__(self, *extraArgs):
        try:
            return apply(self.callback, self.optArgs, self.namedArgs)
        except KeyboardInterrupt:
            raise
        except:
            print "Error in EVCallback %s" % self.__repr__()
            raise

    def __repr__(self):
        return "<EVCallback to %s>" % `self.callback`

_tk = None
def _GetTk():
    global _tk
    if _tk == None:
        _tk = Tkinter.Frame().tk
    return _tk


_nextTimerId = 0
def GetNewTimerId():
    global _nextTimerId

    idToReturn = _nextTimerId
    _nextTimerId = _nextTimerId + 1
    return idToReturn

class _TimerEvent:
    def __init__(self, timeLeft, cb):
        self.addedWhen = time.time()
        self.id = GetNewTimerId()
        self.timeOutTime = self.addedWhen + timeLeft
        self.cb = cb

    def __cmp__(self, other):
        if self.timeOutTime == other.timeOutTime:
            return 0
        elif self.timeOutTime < other.timeOutTime:
            return -1
        else:
            return 1

class _EventHandler:
    STATE_STANDALONE = 1
    STATE_TK = 2

    READ = 1
    WRITE = 2
    EXCEPT = 4

    def __init__(self):
        # mappings of connection --> callback
        self.readEvents = {}
        self.writeEvents = {}
        self.exceptEvents = {}

        # A sorted list of timers: the timer with least time to run is first
        self.timerEvents = []

        # State:        STATE_STANDALONE     -- Tk is not involved
        #               STATE_TK             -- use the eventhandler in Tkinter
        self.state = self.STATE_STANDALONE
        self.tkTopLevel = None

    def _SetStateTk(self, topLevel):
        """State that we are using eventhandler in Tkinter.
Note: When using the Tkinter eventhandler, you cannot delete timer-events."""
        self.state = self.STATE_TK
        self.tkTopLevel = topLevel

    def SetStateStandAlone(self):
        """State that we are implementing our own eventhandler."""
        self.state = self.STATE_STANDALONE


    def PushReadEvent(self, connection, cbfunction, *optArgs, **namedArgs):
        """Register a callback to be called when there's data to read
        on a connection. Or really, push the the callback onto the
        stack of read-callbacks for the connection. Only the first callback
        on the stack is called.

        CONNECTION    = <any object that has a fileno() method>
        CB-FUNCTION   = <function(OPT-ARG ..., NAMED-ARG ...): void>
        OPT-ARG ...   = <any arguments>
        NAMED-ARG ... = <any named arguments>

        Returns: void
        Throws:  nothing
        """
        cb = EVCallback(cbfunction, optArgs, namedArgs)
        if self.readEvents.has_key(connection):
            handlers = self.readEvents[connection]
        else:
            handlers = []
        newHandlers = self._PushHandler(cb, handlers)
        self.readEvents[connection] = newHandlers
        if self.state == self.STATE_TK:
            self._TkPushFileHandler(self.READ, connection)
            

    def PopReadEvent(self, connection):
        """Unregister a read callback for a connection.
        Or really, pop it from the stack of callbacks.

        CONNECTION    = <any object that has a fileno() method>
        
        Returns: void
        Throws:  nothing
        """
        handlers = self.readEvents[connection]
        newHandlers = self._PopHandler(handlers)
        if len(newHandlers) == 0:
            del self.readEvents[connection]
        else:
            self.readEvents[connection] = newHandlers
        if self.state == self.STATE_TK:
            self._TkPopFileHandler(self.READ, connection)


    def PushWriteEvent(self, connection, cbfunction, *optArgs, **namedArgs):
        """Register a callback to be called when there's data to write
        to a connection. Or really, push the the callback onto the
        stack of write-callbacks for the connection. Only the first callback
        on the stack is called.

        CONNECTION    = <any object that has a fileno() method>
        CB-FUNCTION   = <function(OPT-ARG ..., NAMED-ARG ...): void>
        OPT-ARG ...   = <any arguments>
        NAMED-ARG ... = <any named arguments>

        Returns: void
        Throws:  nothing
        """
        cb = EVCallback(cbfunction, optArgs, namedArgs)
        if self.writeEvents.has_key(connection):
            handlers = self.writeEvents[connection]
        else:
            handlers = []
        newHandlers = self._PushHandler(cb, handlers)
        self.writeEvents[connection] = newHandlers
        if self.state == self.STATE_TK:
            self._TkPushFileHandler(self.WRITE, connection)

    def PopWriteEvent(self, connection):
        """Unregister a write callback for a connection.
        Or really, pop it from the stack of callbacks.

        CONNECTION    = <any object that has a fileno() method>
        
        Returns: void
        Throws:  nothing
        """
        handlers = self.writeEvents[connection]
        newHandlers = self._PopHandler(handlers)
        if len(newHandlers) == 0:
            del self.writeEvents[connection]
        else:
            self.writeEvents[connection] = newHandlers
        if self.state == self.STATE_TK:
            self._TkPopFileHandler(self.WRITE, connection)


    def PushExceptEvent(self, connection, cbfunction, *optArgs, **namedArgs):
        """Register a callback to be called when there's an exception
        on a connection. Or really, push the the callback onto the
        stack of exception-callbacks for the connection.
        Only the first callback on the stack is called.

        CONNECTION    = <any object that has a fileno() method>
        CB-FUNCTION   = <function(OPT-ARG ..., NAMED-ARG ...): void>
        OPT-ARG ...   = <any arguments>
        NAMED-ARG ... = <any named arguments>

        Returns: void
        Throws:  nothing
        """
        cb = EVCallback(cbfunction, optArgs, namedArgs)
        if self.exceptEvents.has_key(connection):
            handlers = self.exceptEvents[connection]
        else:
            handlers = []
        newHandlers = self._PushHandler(cb, handlers)
        self.exceptEvents[connection] = newHandlers
        if self.state == self.STATE_TK:
            self._TkPushFileHandler(self.EXCEPT, connection)

    def PopExceptEvent(self, connection):
        """Unregister an exception callback for a connection.
        Or really, pop it from the stack of callbacks.

        CONNECTION    = <any object that has a fileno() method>
        
        Returns: void
        Throws:  nothing
        """
        handlers = self.exceptEvents[connection]
        newHandlers = self._PopHandler(handlers)
        if len(newHandlers) == 0:
            del self.exceptEvents[connection]
        else:
            self.exceptEvents[connection] = newHandlers
        if self.state == self.STATE_TK:
            self._TkPopFileHandler(self.EXCEPT, connection)


    def AddTimerEvent(self, timeLeft, cbfunction, *optArgs, **namedArgs):
        """Register a callback to be called after a certain amount of time.

        TIME-LEFT     = integer | float
                      Timeout (in seconds) for the timer
        CB-FUNCTION   = <function(OPT-ARG ..., NAMED-ARG ...): void>
        OPT-ARG ...   = <any arguments>
        NAMED-ARG ... = <any named arguments>

        Returns: timer-id
        Throws:  nothing
        """
        cb = EVCallback(cbfunction, optArgs, namedArgs)
        if self.state == self.STATE_STANDALONE:
            newTimerEvent = _TimerEvent(timeLeft, cb)
            self.timerEvents.append(newTimerEvent)
            self.timerEvents.sort()
            return newTimerEvent.id
        elif self.state == self.STATE_TK:
            return _GetTk().createtimerhandler(int(round(timeLeft * 1000)), cb)

    def DelTimerEvent(self, id):
        """Unregister a timer callback.

        TIMER-ID      = timer-id

        Returns: nothing
        Throws:  "Cannot delete timer events when using tk"

        Note that it is not possible to unregister timer events when using Tk.
        """
        if self.state == self.STATE_TK:
            raise "Cannot delete timer events when using tk"

        indexForId = None
        it = map(None, range(len(self.timerEvents)), self.timerEvents)
        for i, ev in it:
            if ev.id == id:
                indexForId = i
                break
        if indexForId != None:
            del self.timerEvents[indexForId]

    def Loop(self):
        """Start the event handler.

        No arguments:
        Returns: void
        Throws:  nothing
        """
        if self.state == self.STATE_TK:
            self.__LoopTk()
        elif self.state == self.STATE_STANDALONE:
            self.__LoopStandalone()


    def __LoopTk(self):
        self.tkTopLevel.mainloop()

    def __LoopStandalone(self):
        self.continueLooping = 1
        while self.continueLooping:
            rList = self.readEvents.keys()
            wList = self.writeEvents.keys()
            eList = self.exceptEvents.keys()

            timeout = None
            if len(self.timerEvents) > 0:
                firstTimerEv = self.timerEvents[0]
                now = time.time()
                timeout = firstTimerEv.timeOutTime - now

            if timeout == None:
                # No timer to wait for
                try:
                    reads, writes, excepts = select.select(rList, wList, eList)
                except select.error, info:
                    (errno, errText) = info
                    if errno == 4:
                        # 'Interrupted system call'
                        # ignore this one.
                        # loop again in the while loop
                        continue
                    else:
                        # Other error: serious. Raise again.
                        raise

            elif timeout < 0:
                # Signal timeout
                (reads, writes, excepts) = ([], [], [])
            else:
                # Select and wait for timer
                reads, writes, excepts = select.select(rList, wList, eList,
                                                       timeout)

            # Check for handles that are clear for reading
            for readable in reads:
                cb = self.readEvents[readable][0]
                cb()

            # Check for handles that are clear for writing
            for writable in writes:
                cb = self.writeEvents[writable][0]
                cb()

            # Check for handles that are in exception situation
            for exceptable in excepts:
                cb = self.exceptEvents[exceptable][0]
                cb()

            # Check for timers that have timed out
            while 1:
                expiredTimers = filter(lambda ev, now=time.time():
                                       ev.timeOutTime <= now,
                                       self.timerEvents)
                if len(expiredTimers) == 0:
                    break               # skip the while loop
                
                for expiredTimer in expiredTimers:
                    expiredTimer.cb()
                    self.DelTimerEvent(expiredTimer.id)


    def StopLooping(self):
        """Start the event handler.

        No arguments:
        Returns: void
        Throws:  nothing

        This can use in e.g. callbacks to stop the event handler loop."""
        self.continueLooping = 0

    def _PushHandler(self, cb, handlers):
        return [cb] + handlers

    def _PopHandler(self, handlers):
        return handlers[1:]

    def _TkPushFileHandler(self, eventType, connection):
        fileNum = connection.fileno()
        if eventType == self.READ:
            cb = self.readEvents[connection][0]
            _GetTk().createfilehandler(fileNum, tkinter.READABLE, cb)
        elif eventType == self.WRITE:
            cb = self.writeEvents[connection][0]
            _GetTk().createfilehandler(fileNum, tkinter.WRITABLE, cb)
        elif eventType == self.EXCEPT:
            cb = self.exceptEvents[connection][0]
            _GetTk().createfilehandler(fileNum, tkinter.EXCEPTION,cb)

    def _TkPopFileHandler(self, eventType, connection):
        fileNum = connection.fileno()
        ## In tkinter, all handlers (readable as well as writable/exception)
        ## are deleted when we delete a handler.
        ## The net result is that the code is all the same no matter
        ## what type of handler we delete.
        _GetTk().deletefilehandler(fileNum)
        if self.readEvents.has_key(connection):
            cb = self.readEvents[connection][0]
            _GetTk().createfilehandler(fileNum, tkinter.READABLE, cb)
        if self.writeEvents.has_key(connection):
            cb = self.writeEvents[connection][0]
            _GetTk().createfilehandler(fileNum, tkinter.WRITABLE, cb)
        if self.exceptEvents.has_key(connection):
            cb = self.exceptEvents[connection][0]
            _GetTk().createfilehandler(fileNum, tkinter.EXCEPTION,cb)


syntax highlighted by Code2HTML, v. 0.9.1