From 611b0fbd9c0e55b0649340563a63434cf51c5adf Mon Sep 17 00:00:00 2001 From: Mattias Andrée Date: Mon, 21 Jul 2014 23:18:56 +0200 Subject: ropty, does not work with the standard programs, but you can echo into it MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Signed-off-by: Mattias Andrée --- DEPENDENCIES | 1 + Makefile | 4 +- TODO | 1 - examples/plugins/ropty | 30 +++++++++++++ src/plugins/ropty.py | 113 +++++++++++++++++++++++++++++++++++++++++++++++++ 5 files changed, 146 insertions(+), 3 deletions(-) create mode 100644 examples/plugins/ropty create mode 100644 src/plugins/ropty.py diff --git a/DEPENDENCIES b/DEPENDENCIES index 7b43bbb..31835c2 100644 --- a/DEPENDENCIES +++ b/DEPENDENCIES @@ -13,6 +13,7 @@ OPTIONAL RUNTIME DEPENDENCIES: python-pyalsaaudio: for ALSA volume control hdparm: for hdparm support python-pytz: for timezone support + python-sysv-ipc: for ropty example BUILD DEPENDENCIES: diff --git a/Makefile b/Makefile index 2422e4f..7cf4f12 100644 --- a/Makefile +++ b/Makefile @@ -32,12 +32,12 @@ PLUGINS = chase clock cpuinfo cpuonline cpu df discstats ipaddress \ kmsg leapsec linereader loadavg lunar mem moc network \ pacman snmp snmp6 softirqs solar uname uptime users \ vmstat weather xdisplay xkb alsa dentrystate inodestate \ - files hdparm tzclock + files hdparm tzclock ropty PLUGIN_EXAMPLES = chase clock cpu cpuinfo cpuonline df discstats \ ipaddress kmsg loadavg lunar mem moc network \ pacman uname uptime users xdisplay xkb alsa \ - dentrystate inodestate files tzclock + dentrystate inodestate files tzclock ropty EXAMPLES = mixed moderate test xmonad diff --git a/TODO b/TODO index fcf72c9..f3cdf55 100644 --- a/TODO +++ b/TODO @@ -20,7 +20,6 @@ List of plugins to implement: Keyboard layout EWMH Meteor showers - pty for write/talk messages Lunar and solar eclipses e-mail Identify active and visible windows diff --git a/examples/plugins/ropty b/examples/plugins/ropty new file mode 100644 index 0000000..5e41351 --- /dev/null +++ b/examples/plugins/ropty @@ -0,0 +1,30 @@ +# -*- python -*- + +# A xpybar configuration example testing the features of plugins.kmsg + +from plugins.ropty import ROPTY + + +OUTPUT, HEIGHT, YPOS, TOP = 0, 12, 24, True + +text = '' +pty_ = None + +start_ = start +def start(): + global pty_ + start_() + def refresh(): + if pty_.size() == 0: + text = '' + else: + line = pty_.next + text = '\033[37;41m%i\033[00m%s' + text %= (pty_.size(), line) + bar.invalidate() + pty_ = ROPTY(refresh) + +def redraw(): + bar.clear() + bar.draw_coloured_text(0, 10, 0, 2, text) + diff --git a/src/plugins/ropty.py b/src/plugins/ropty.py new file mode 100644 index 0000000..09e4a23 --- /dev/null +++ b/src/plugins/ropty.py @@ -0,0 +1,113 @@ +# -*- python -*- +''' +xpybar – xmobar replacement written in python +Copyright © 2014 Mattias Andrée (maandree@member.fsf.org) + +This program is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +This program 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 General Public License for more details. + +You should have received a copy of the GNU General Public License +along with this program. If not, see . +''' +import os +import threading + +from plugins.linereader import LineReader + + +class ROPTY(LineReader): + ''' + Read-only PTY for viewing of wall, write and talk messages + + @variable on_update:()→void Called when a new line is available + @variable master:int The file descriptor of the PTY master + @variable slave:int The file descriptor of the PTY slave + ''' + + + def __init__(self, on_update = None): + ''' + Constructor + + @param on_update:()→void Called when a new line is available + ''' + def noop(): + pass + self.on_update = noop if on_update is None else on_update + (self.master, self.slave) = os.openpty() + self.__reader = LineReader(self.master) + self.__condition = threading.Condition() + self.__queue = [] + def background(): + try: + while True: + got = self.__reader.next() + if got is None: + return + self.__condition.acquire() + try: + self.__queue.append(got) + finally: + self.__condition.release() + self.on_update() + except: + return + self.__thread = threading.Thread(target = background) + self.__thread.setDaemon(True) + self.__thread.start() + + + def close(self): + ''' + Close the PTY + ''' + os.close(self.slave) + os.close(self.master) + + + def size(self): + ''' + Return the number of available lines + + @return :int The number of available lines + ''' + self.__condition.acquire() + try: + return len(self.__queue) + finally: + self.__condition.release() + + + def __len__(self): + ''' + Return the number of available lines + + @return :int The number of available lines + ''' + return self.size() + + + def next(self): + ''' + Return and unqueue the next available line + + @return :str The next available line, `None` if there + are no more lines currently available + ''' + self.__condition.acquire() + try: + if len(self.__queue) == 0: + return None + rc = self.__queue[0] + self.__queue[:] = self.__queue[1:] + finally: + self.__condition.release() + return rc + -- cgit v1.2.3-70-g09d2