Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ponty/PyVirtualDisplay
Python wrapper for Xvfb, Xephyr and Xvnc
https://github.com/ponty/PyVirtualDisplay
Last synced: 2 months ago
JSON representation
Python wrapper for Xvfb, Xephyr and Xvnc
- Host: GitHub
- URL: https://github.com/ponty/PyVirtualDisplay
- Owner: ponty
- License: bsd-2-clause
- Created: 2011-02-11T10:41:09.000Z (almost 14 years ago)
- Default Branch: master
- Last Pushed: 2024-02-19T20:36:35.000Z (11 months ago)
- Last Synced: 2024-05-29T13:45:08.526Z (8 months ago)
- Language: Python
- Homepage:
- Size: 427 KB
- Stars: 684
- Watchers: 21
- Forks: 78
- Open Issues: 13
-
Metadata Files:
- Readme: README.md
- License: LICENSE.txt
Awesome Lists containing this project
- awesome - ponty/PyVirtualDisplay - Python wrapper for Xvfb, Xephyr and Xvnc (Python)
README
pyvirtualdisplay is a python wrapper for [Xvfb][1], [Xephyr][2] and [Xvnc][3] programs.
They all use the X Window System (not Windows, not macOS)
The selected program should be installed first so that it can be started without a path,
otherwise pyvirtualdisplay will not find the program.Links:
* home: https://github.com/ponty/pyvirtualdisplay
* PYPI: https://pypi.python.org/pypi/pyvirtualdisplay![workflow](https://github.com/ponty/pyvirtualdisplay/actions/workflows/main.yml/badge.svg)
Features:
- python wrapper
- supported python versions: 3.6, 3.7, 3.8, 3.9, 3.10, 3.11, 3.12
- back-ends: [Xvfb][1], [Xephyr][2] and [Xvnc][3]Possible applications:
* headless run
* GUI testing
* automatic GUI screenshotInstallation
============install the program:
```console
$ python3 -m pip install pyvirtualdisplay
```optional: [Pillow][pillow] should be installed for ``smartdisplay`` submodule:
```console
$ python3 -m pip install pillow
```optional: [EasyProcess][EasyProcess] should be installed for some examples:
```console
$ python3 -m pip install EasyProcess
```
optional: xmessage and gnumeric should be installed for some examples.On Ubuntu 22.04:
```console
$ sudo apt install x11-utils gnumeric
```If you get this error message on Linux then your Pillow version is old.
```
ImportError: ImageGrab is macOS and Windows only
```Install all dependencies and backends on Ubuntu 22.04:
```console
$ sudo apt-get install xvfb xserver-xephyr tigervnc-standalone-server x11-utils gnumeric
$ python3 -m pip install pyvirtualdisplay pillow EasyProcess
```Usage
=====Controlling the display with context manager:
```py
from pyvirtualdisplay import Display
with Display() as disp:
# display is active
print(disp.is_alive()) # True
# display is stopped
print(disp.is_alive()) # False
```Controlling the display with `start()` and `stop()` methods (not recommended):
```py
from pyvirtualdisplay import Display
disp = Display()
disp.start()
# display is active
disp.stop()
# display is stopped
```After Xvfb display is activated "DISPLAY" environment variable is set for Xvfb.
(e.g. `os.environ["DISPLAY"] = :1`)
After Xvfb display is stopped `start()` and `stop()` are not allowed to be called again, "DISPLAY" environment variable is restored to its original value.Selecting Xvfb backend:
```py
disp=Display()
# or
disp=Display(visible=False)
# or
disp=Display(backend="xvfb")
```Selecting Xephyr backend:
```py
disp=Display(visible=True)
# or
disp=Display(backend="xephyr")
```Selecting Xvnc backend:
```py
disp=Display(backend="xvnc")
```Setting display size:
```py
disp=Display(size=(100, 60))
```Setting display color depth:
```py
disp=Display(color_depth=24)
```Headless run
------------A messagebox is displayed on a hidden display.
```py
# pyvirtualdisplay/examples/headless.py"Start Xvfb server. Open xmessage window."
from easyprocess import EasyProcess
from pyvirtualdisplay import Display
with Display(visible=False, size=(100, 60)) as disp:
with EasyProcess(["xmessage", "hello"]) as proc:
proc.wait()```
Run it:
```console
$ python3 -m pyvirtualdisplay.examples.headless
```If `visible=True` then a nested Xephyr window opens and the GUI can be controlled.
vncserver
---------The same as headless example, but it can be controlled with a VNC client.
```py
# pyvirtualdisplay/examples/vncserver.py"Start virtual VNC server. Connect with: vncviewer localhost:5904"
from easyprocess import EasyProcess
from pyvirtualdisplay import Display
with Display(backend="xvnc", size=(100, 60), rfbport=5904) as disp:
with EasyProcess(["xmessage", "hello"]) as proc:
proc.wait()```
Run it:
```console
$ python3 -m pyvirtualdisplay.examples.vncserver
```Check it with vncviewer:
```console
$ vncviewer localhost:5904
```![](doc/gen/vncviewer_localhost:5904.png)
GUI Test
--------```py
# pyvirtualdisplay/examples/lowres.py"Testing gnumeric on low resolution."
from easyprocess import EasyProcessfrom pyvirtualdisplay import Display
# start Xephyr
with Display(visible=True, size=(320, 240)) as disp:
# start Gnumeric
with EasyProcess(["gnumeric"]) as proc:
proc.wait()```
Run it:
```console
$ python3 -m pyvirtualdisplay.examples.lowres
```Image:
![](doc/gen/python3_-m_pyvirtualdisplay.examples.lowres.png)
Screenshot
----------```py
# pyvirtualdisplay/examples/screenshot.py"Create screenshot of xmessage in background using 'smartdisplay' submodule"
from easyprocess import EasyProcessfrom pyvirtualdisplay.smartdisplay import SmartDisplay
# 'SmartDisplay' instead of 'Display'
# It has 'waitgrab()' method.
# It has more dependencies than Display.
with SmartDisplay() as disp:
with EasyProcess(["xmessage", "hello"]):
# wait until something is displayed on the virtual display (polling method)
# and then take a fullscreen screenshot
# and then crop it. Background is black.
img = disp.waitgrab()
img.save("xmessage.png")```
Run it:
```console
$ python3 -m pyvirtualdisplay.examples.screenshot
```Image:
![](doc/gen/xmessage.png)
Nested Xephyr
-------------```py
# pyvirtualdisplay/examples/nested.py"Nested Xephyr servers"
from easyprocess import EasyProcessfrom pyvirtualdisplay import Display
with Display(visible=True, size=(220, 180), bgcolor="black"):
with Display(visible=True, size=(200, 160), bgcolor="white"):
with Display(visible=True, size=(180, 140), bgcolor="black"):
with Display(visible=True, size=(160, 120), bgcolor="white"):
with Display(visible=True, size=(140, 100), bgcolor="black"):
with Display(visible=True, size=(120, 80), bgcolor="white"):
with Display(visible=True, size=(100, 60), bgcolor="black"):
with EasyProcess(["xmessage", "hello"]) as proc:
proc.wait()```
Run it:
```console
$ python3 -m pyvirtualdisplay.examples.nested
```Image:
![](doc/gen/python3_-m_pyvirtualdisplay.examples.nested.png)
xauth
=====Some programs require a functional Xauthority file. PyVirtualDisplay can
generate one and set the appropriate environment variables if you pass
``use_xauth=True`` to the ``Display`` constructor. Note however that this
feature needs ``xauth`` installed, otherwise a
``pyvirtualdisplay.xauth.NotFoundError`` is raised.Mouse cursor
============The cursor can be disabled in Xvfb using an extra argument which is passed directly to Xvfb:
```py
with Display(backend="xvfb", extra_args=["-nocursor"]):
...
```Based on Xvfb help:
```
...
-nocursor disable the cursor
...
```Concurrency
===========If more X servers are started at the same time then there is race for free display numbers.
_"Recent X servers as of version 1.13 (Xvfb, too) support the -displayfd command line option: It will make the X server choose the display itself"_
https://stackoverflow.com/questions/2520704/find-a-free-x11-display-number/Version 1.13 was released in 2012: https://www.x.org/releases/individual/xserver/
First help text is checked (e.g. `Xvfb -help`) to find if `-displayfd` flag is available.
If `-displayfd` flag is available then it is used to choose the display number.
If not then a free display number is generated and there are 10 retries by default
which should be enough for starting 10 X servers at the same time.`displayfd` usage is disabled on macOS because it doesn't work with XQuartz-2.7.11, always 0 is returned.
Thread safety
=============All previous examples are not thread-safe, because `pyvirtualdisplay` replaces `$DISPLAY` environment variable in global [`os.environ`][environ] in `start()` and sets back to original value in `stop()`.
To make it thread-safe you have to manage the `$DISPLAY` variable.
Set `manage_global_env` to `False` in constructor.```py
# pyvirtualdisplay/examples/threadsafe.py"Start Xvfb server and open xmessage window. Thread safe."
import threading
from easyprocess import EasyProcess
from pyvirtualdisplay.smartdisplay import SmartDisplay
def thread_function(index):
# manage_global_env=False is thread safe
with SmartDisplay(manage_global_env=False) as disp:
cmd = ["xmessage", str(index)]
# disp.new_display_var should be used for new processes
# disp.env() copies global os.environ and adds disp.new_display_var
with EasyProcess(cmd, env=disp.env()):
img = disp.waitgrab()
img.save("xmessage{}.png".format(index))t1 = threading.Thread(target=thread_function, args=(1,))
t2 = threading.Thread(target=thread_function, args=(2,))
t1.start()
t2.start()
t1.join()
t2.join()```
Run it:
```console
$ python3 -m pyvirtualdisplay.examples.threadsafe
```Images:
![](doc/gen/xmessage1.png)
![](doc/gen/xmessage2.png)Hierarchy
=========![Alt text](https://g.gravizo.com/source/svg?https%3A%2F%2Fraw.githubusercontent.com/ponty/pyvirtualdisplay/master/doc/hierarchy.dot)
[1]: http://en.wikipedia.org/wiki/Xvfb
[2]: http://en.wikipedia.org/wiki/Xephyr
[3]: https://tigervnc.org/
[pillow]: https://pillow.readthedocs.io
[environ]: https://docs.python.org/3/library/os.html#os.environ
[EasyProcess]: https://github.com/ponty/EasyProcess