uievent.py 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. # ex:ts=4:sw=4:sts=4:et
  2. # -*- tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*-
  3. #
  4. # Copyright (C) 2006 - 2007 Michael 'Mickey' Lauer
  5. # Copyright (C) 2006 - 2007 Richard Purdie
  6. #
  7. # This program is free software; you can redistribute it and/or modify
  8. # it under the terms of the GNU General Public License version 2 as
  9. # published by the Free Software Foundation.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # You should have received a copy of the GNU General Public License along
  17. # with this program; if not, write to the Free Software Foundation, Inc.,
  18. # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  19. """
  20. Use this class to fork off a thread to recieve event callbacks from the bitbake
  21. server and queue them for the UI to process. This process must be used to avoid
  22. client/server deadlocks.
  23. """
  24. import socket, threading, pickle
  25. from SimpleXMLRPCServer import SimpleXMLRPCServer, SimpleXMLRPCRequestHandler
  26. class BBUIEventQueue:
  27. def __init__(self, BBServer, clientinfo=("localhost, 0")):
  28. self.eventQueue = []
  29. self.eventQueueLock = threading.Lock()
  30. self.eventQueueNotify = threading.Event()
  31. self.BBServer = BBServer
  32. self.clientinfo = clientinfo
  33. server = UIXMLRPCServer(self.clientinfo)
  34. self.host, self.port = server.socket.getsockname()
  35. server.register_function( self.system_quit, "event.quit" )
  36. server.register_function( self.send_event, "event.sendpickle" )
  37. server.socket.settimeout(1)
  38. self.EventHandle = None
  39. # the event handler registration may fail here due to cooker being in invalid state
  40. # this is a transient situation, and we should retry a couple of times before
  41. # giving up
  42. for count_tries in range(5):
  43. self.EventHandle, error = self.BBServer.registerEventHandler(self.host, self.port)
  44. if self.EventHandle != None:
  45. break
  46. errmsg = "Could not register UI event handler. Error: %s, host %s, "\
  47. "port %d" % (error, self.host, self.port)
  48. bb.warn("%s, retry" % errmsg)
  49. import time
  50. time.sleep(1)
  51. else:
  52. raise Exception(errmsg)
  53. self.server = server
  54. self.t = threading.Thread()
  55. self.t.setDaemon(True)
  56. self.t.run = self.startCallbackHandler
  57. self.t.start()
  58. def getEvent(self):
  59. self.eventQueueLock.acquire()
  60. if len(self.eventQueue) == 0:
  61. self.eventQueueLock.release()
  62. return None
  63. item = self.eventQueue.pop(0)
  64. if len(self.eventQueue) == 0:
  65. self.eventQueueNotify.clear()
  66. self.eventQueueLock.release()
  67. return item
  68. def waitEvent(self, delay):
  69. self.eventQueueNotify.wait(delay)
  70. return self.getEvent()
  71. def queue_event(self, event):
  72. self.eventQueueLock.acquire()
  73. self.eventQueue.append(event)
  74. self.eventQueueNotify.set()
  75. self.eventQueueLock.release()
  76. def send_event(self, event):
  77. self.queue_event(pickle.loads(event))
  78. def startCallbackHandler(self):
  79. self.server.timeout = 1
  80. bb.utils.set_process_name("UIEventQueue")
  81. while not self.server.quit:
  82. try:
  83. self.server.handle_request()
  84. except Exception as e:
  85. import traceback
  86. logger.error("BBUIEventQueue.startCallbackHandler: Exception while trying to handle request: %s\n%s" % (e, traceback.format_exc(e)))
  87. self.server.server_close()
  88. def system_quit( self ):
  89. """
  90. Shut down the callback thread
  91. """
  92. try:
  93. self.BBServer.unregisterEventHandler(self.EventHandle)
  94. except:
  95. pass
  96. self.server.quit = True
  97. class UIXMLRPCServer (SimpleXMLRPCServer):
  98. def __init__( self, interface ):
  99. self.quit = False
  100. SimpleXMLRPCServer.__init__( self,
  101. interface,
  102. requestHandler=SimpleXMLRPCRequestHandler,
  103. logRequests=False, allow_none=True)
  104. def get_request(self):
  105. while not self.quit:
  106. try:
  107. sock, addr = self.socket.accept()
  108. sock.settimeout(1)
  109. return (sock, addr)
  110. except socket.timeout:
  111. pass
  112. return (None, None)
  113. def close_request(self, request):
  114. if request is None:
  115. return
  116. SimpleXMLRPCServer.close_request(self, request)
  117. def process_request(self, request, client_address):
  118. if request is None:
  119. return
  120. SimpleXMLRPCServer.process_request(self, request, client_address)