Source code for schrodinger.application.desmond.xvfbwrapper
#!/usr/bin/env python
#
#   * Corey Goldberg, 2012, 2013
#
#   * inspired by:
#       PyVirtualDisplay: http://pypi.python.org/pypi/PyVirtualDisplay
"""wrapper for running display inside X virtual framebuffer (Xvfb)"""
import fnmatch
import os
import random
import subprocess
import time
[docs]class Xvfb:
[docs]    def __init__(self, width=800, height=680, colordepth=24, **kwargs):
        self.width = width
        self.height = height
        self.colordepth = colordepth
        self._set_xvfb_cmd()
        for key, value in kwargs.items():
            self.xvfb_cmd = self.xvfb_cmd + ['-%s' % key, value]
        self.proc = None
        if 'DISPLAY' in os.environ:
            self.old_display_num = os.environ['DISPLAY'].split(':')[1]
        else:
            self.old_display_num = None 
    def _set_xvfb_cmd(self):
        self.xvfb_cmd = [
            '-screen', '0',
            '%dx%dx%d' % (self.width, self.height, self.colordepth)
        ]
    def __enter__(self):
        self.start()
        return self
    def __exit__(self, exc_type, exc_val, exc_tb):
        self.stop()
[docs]    def start(self):
        if 'DISPLAY' in os.environ:
            print("DISPLAY already exists: %s " % os.environ['DISPLAY'])
            return
        self.vdisplay_num = self.search_for_free_display()
        self.xvfb_cmd = ['Xvfb', ':%d' % self.vdisplay_num] + self.xvfb_cmd
        self.proc = subprocess.Popen(self.xvfb_cmd,
                                     stdout=subprocess.DEVNULL,
                                     stderr=subprocess.DEVNULL,
                                     universal_newlines=True)
        print(self.xvfb_cmd)
        time.sleep(2)  # give Xvfb time to start
        ret_code = self.proc.poll()
        if ret_code is None:
            self._redirect_display(self.vdisplay_num)
        elif self.old_display_num:
            self._redirect_display(self.old_display_num)
            self.proc = None
            print('Error: Xvfb did not start') 
[docs]    def stop(self):
        if self.old_display_num:
            self._redirect_display(self.old_display_num)
        elif 'DISPLAY' in os.environ:
            del os.environ['DISPLAY']
        if self.proc is not None:
            self.proc.kill()
            self.proc.wait()
            self.proc = None
        self._set_xvfb_cmd() 
    def __del__(self):
        print("deleting X...")
        self.stop()
        del self
[docs]    def search_for_free_display(self):
        ls = [int(x.split('X')[1].split('-')[0]) for x in self._lock_files()]
        min_display_num = 1000
        if len(ls):
            display_num = max(min_display_num, max(ls) + 1)
        else:
            display_num = min_display_num
        random.seed()
        display_num += random.randint(0, 100)
        return display_num 
    def _lock_files(self):
        tmpdir = '/tmp'
        pattern = '.X*-lock'
        names = fnmatch.filter(os.listdir(tmpdir), pattern)
        ls = [os.path.join(tmpdir, child) for child in names]
        ls = [p for p in ls if os.path.isfile(p)]
        return ls
    def _redirect_display(self, display_num):
        if display_num:
            os.environ['DISPLAY'] = ':%s' % display_num