i3pystatus/statushandler.py
2013-02-11 21:37:07 +01:00

79 lines
2.0 KiB
Python

#!/usr/bin/env python
import sys
import json
import urllib.request, urllib.error, urllib.parse
from threading import Thread
class Module(object):
output = None
async = False
def registered(self, status_handler):
"""Called when this module is registered with a status handler"""
def tick(self):
"""Only called if self.async == False. Called once per tick"""
class I3statusHandler(object):
modules = []
def __init__(self):
pass
def register_module(self, module):
"""Register a new module."""
self.modules.append(module)
def print_line(self, message):
"""Unbuffered printing to stdout."""
sys.stdout.write(message + '\n')
sys.stdout.flush()
def read_line(self):
"""Interrupted respecting reader for stdin."""
# try reading a line, removing any extra whitespace
try:
line = sys.stdin.readline().strip()
# i3status sends EOF, or an empty line
if not line:
sys.exit(3)
return line
# exit on ctrl-c
except KeyboardInterrupt:
sys.exit()
def run(self):
self.print_line(self.read_line())
self.print_line(self.read_line())
for module in self.modules:
if module.async:
module.thread = Thread(target=module.mainloop)
module.thread.daemon = True
module.thread.start()
while True:
line, prefix = self.read_line(), ''
# ignore comma at start of lines
if line.startswith(','):
line, prefix = line[1:], ','
j = json.loads(line)
for module in self.modules:
if not module.async:
module.tick()
output = module.output
if output:
j.insert(0, output)
# and echo back new encoded json
self.print_line(prefix+json.dumps(j))