[Author Prev][Author Next][Thread Prev][Thread Next][Author Index][Thread Index]
[or-cvs] r19716: {arm} Decently big batch of feature additions and bug fixes. added (arm/trunk/interface)
Author: atagar
Date: 2009-06-15 02:42:19 -0400 (Mon, 15 Jun 2009)
New Revision: 19716
Added:
arm/trunk/interface/confPanel.py
arm/trunk/interface/headerPanel.py
Removed:
arm/trunk/interface/staticPanel.py
Modified:
arm/trunk/interface/bandwidthPanel.py
arm/trunk/interface/controller.py
arm/trunk/interface/util.py
Log:
Decently big batch of feature additions and bug fixes.
added: second page that presents torrc with syntax highlighing, optional comment stripping, etc
added: ps sampling (cpu/memory usage, pid, and uptime)
added: help popup with page controls
bug fix: corrected issue that caused periodic refreshing to fail
bug fix: accounting reset time takes into account DST
bug fix: make accounting input and header pausable
Modified: arm/trunk/interface/bandwidthPanel.py
===================================================================
--- arm/trunk/interface/bandwidthPanel.py 2009-06-14 02:39:33 UTC (rev 19715)
+++ arm/trunk/interface/bandwidthPanel.py 2009-06-15 06:42:19 UTC (rev 19716)
@@ -34,6 +34,7 @@
self.lastUploadRate = 0
self.maxDownloadRate = 1 # max rates seen, used to determine graph bounds
self.maxUploadRate = 1
+ self.accountingInfo = None # accounting data (set by _updateAccountingInfo method)
self.isPaused = False
self.pauseBuffer = None # mirror instance used to track updates when paused
@@ -112,35 +113,22 @@
self.addstr(7 - row, col + 40, " ", curses.A_STANDOUT | ulColor)
if self.isAccounting:
- try:
- accountingParams = self.conn.get_info(["accounting/hibernating", "accounting/bytes", "accounting/bytes-left", "accounting/interval-end"])
-
- hibernateStr = accountingParams["accounting/hibernating"]
+ if not self.isPaused: self._updateAccountingInfo()
+
+ if self.accountingInfo:
+ status = self.accountingInfo["status"]
hibernateColor = "green"
- if hibernateStr == "soft": hibernateColor = "yellow"
- elif hibernateStr == "hard": hibernateColor = "red"
+ if status == "soft": hibernateColor = "yellow"
+ elif status == "hard": hibernateColor = "red"
self.addstr(9, 0, "Accounting (", curses.A_BOLD)
- self.addstr(9, 12, hibernateStr, curses.A_BOLD | util.getColor(hibernateColor))
- self.addstr(9, 12 + len(hibernateStr), "):", curses.A_BOLD)
+ self.addstr(9, 12, status, curses.A_BOLD | util.getColor(hibernateColor))
+ self.addstr(9, 12 + len(status), "):", curses.A_BOLD)
- # timezone subtraction converts from gmt to local
- sec = time.mktime(time.strptime(accountingParams["accounting/interval-end"], "%Y-%m-%d %H:%M:%S")) - time.time() - time.timezone
- resetHours = sec / 3600
- sec %= 3600
- resetMin = sec / 60
- sec %= 60
-
- self.addstr(9, 35, "Time to reset: %i:%02i:%02i" % (resetHours, resetMin, sec))
-
- read = util.getSizeLabel(int(accountingParams["accounting/bytes"].split(" ")[0]))
- written = util.getSizeLabel(int(accountingParams["accounting/bytes"].split(" ")[1]))
- limit = util.getSizeLabel(int(accountingParams["accounting/bytes"].split(" ")[0]) + int(accountingParams["accounting/bytes-left"].split(" ")[0]))
-
- self.addstr(10, 2, "%s / %s" % (read, limit), dlColor)
- self.addstr(10, 37, "%s / %s" % (written, limit), ulColor)
-
- except TorCtl.TorCtlClosed:
+ self.addstr(9, 35, "Time to reset: %s" % self.accountingInfo["resetTime"])
+ self.addstr(10, 2, "%s / %s" % (self.accountingInfo["read"], self.accountingInfo["readLimit"]), dlColor)
+ self.addstr(10, 37, "%s / %s" % (self.accountingInfo["written"], self.accountingInfo["writtenLimit"]), ulColor)
+ else:
self.addstr(9, 0, "Accounting:", curses.A_BOLD)
self.addstr(9, 12, "Shutting Down...")
@@ -175,4 +163,38 @@
self.downloadRates = self.pauseBuffer.downloadRates
self.uploadRates = self.pauseBuffer.uploadRates
self.redraw()
+
+ def _updateAccountingInfo(self):
+ """
+ Updates mapping used for accounting info. This includes the following keys:
+ status, resetTime, read, written, readLimit, writtenLimit
+
+ Sets mapping to None if the Tor connection is closed.
+ """
+
+ try:
+ self.accountingInfo = {}
+
+ accountingParams = self.conn.get_info(["accounting/hibernating", "accounting/bytes", "accounting/bytes-left", "accounting/interval-end"])
+ self.accountingInfo["status"] = accountingParams["accounting/hibernating"]
+
+ # altzone subtraction converts from gmt to local with respect to DST
+ sec = time.mktime(time.strptime(accountingParams["accounting/interval-end"], "%Y-%m-%d %H:%M:%S")) - time.time() - time.altzone
+ resetHours = sec / 3600
+ sec %= 3600
+ resetMin = sec / 60
+ sec %= 60
+ self.accountingInfo["resetTime"] = "%i:%02i:%02i" % (resetHours, resetMin, sec)
+
+ read = int(accountingParams["accounting/bytes"].split(" ")[0])
+ written = int(accountingParams["accounting/bytes"].split(" ")[1])
+ readLeft = int(accountingParams["accounting/bytes-left"].split(" ")[0])
+ writtenLeft = int(accountingParams["accounting/bytes-left"].split(" ")[1])
+
+ self.accountingInfo["read"] = util.getSizeLabel(read)
+ self.accountingInfo["written"] = util.getSizeLabel(written)
+ self.accountingInfo["readLimit"] = util.getSizeLabel(read + readLeft)
+ self.accountingInfo["writtenLimit"] = util.getSizeLabel(written + writtenLeft)
+ except TorCtl.TorCtlClosed:
+ self.accountingInfo = None
Added: arm/trunk/interface/confPanel.py
===================================================================
--- arm/trunk/interface/confPanel.py (rev 0)
+++ arm/trunk/interface/confPanel.py 2009-06-15 06:42:19 UTC (rev 19716)
@@ -0,0 +1,88 @@
+#!/usr/bin/env python
+# confPanel.py -- Presents torrc with syntax highlighting.
+# Released under the GPL v3 (http://www.gnu.org/licenses/gpl.html)
+
+import math
+import curses
+from TorCtl import TorCtl
+
+import util
+
+class ConfPanel(util.Panel):
+ """
+ Presents torrc with syntax highlighting in a scroll-able area.
+ """
+
+ def __init__(self, lock, confLocation):
+ util.Panel.__init__(self, lock, -1)
+ self.confLocation = confLocation
+ self.showLineNum = True
+ self.stripComments = False
+ self.reset()
+
+ def reset(self):
+ """
+ Reloads torrc contents and resets scroll height.
+ """
+ confFile = open(self.confLocation, "r")
+ self.confContents = confFile.readlines()
+ confFile.close()
+ self.scroll = 0
+
+ def handleKey(self, key):
+ self._resetBounds()
+ pageHeight = self.maxY - 1
+ if key == curses.KEY_UP: self.scroll = max(self.scroll - 1, 0)
+ elif key == curses.KEY_DOWN: self.scroll = min(self.scroll + 1, len(self.confContents) - pageHeight)
+ elif key == curses.KEY_PPAGE: self.scroll = max(self.scroll - pageHeight, 0)
+ elif key == curses.KEY_NPAGE: self.scroll = min(self.scroll + pageHeight, len(self.confContents) - pageHeight)
+ elif key == ord('n') or key == ord('N'): self.showLineNum = not self.showLineNum
+ elif key == ord('r') or key == ord('R'): self.reset()
+ elif key == ord('s') or key == ord('S'):
+ self.stripComments = not self.stripComments
+ self.scroll = 0
+ self.redraw()
+
+ def redraw(self):
+ if self.win:
+ if not self.lock.acquire(False): return
+ try:
+ self.clear()
+ self.addstr(0, 0, "Tor Config (%s):" % self.confLocation, util.LABEL_ATTR)
+
+ if self.stripComments:
+ displayText = []
+
+ for line in self.confContents:
+ commentStart = line.find("#")
+ if commentStart != -1: line = line[:commentStart]
+
+ line = line.strip()
+ if line: displayText.append(line)
+ else: displayText = self.confContents
+
+ pageHeight = self.maxY - 1
+ numFieldWidth = int(math.log10(len(displayText))) + 1
+ for i in range(self.scroll, min(len(displayText), self.scroll + pageHeight)):
+ lineText = displayText[i].strip()
+ endBreak = 0
+
+ if self.showLineNum:
+ self.addstr(i - self.scroll + 1, 0, ("%%%ii" % numFieldWidth) % (i + 1), curses.A_BOLD | util.getColor("yellow"))
+ numOffset = numFieldWidth + 1
+ else: numOffset = 0
+
+ if not lineText: continue
+ elif not lineText[0] == "#":
+ ctlBreak = lineText.find(" ")
+ endBreak = lineText.find("#")
+ if endBreak == -1: endBreak = len(lineText)
+
+ self.addstr(i - self.scroll + 1, numOffset, lineText[:ctlBreak], curses.A_BOLD | util.getColor("green"))
+ self.addstr(i - self.scroll + 1, numOffset + ctlBreak, lineText[ctlBreak:endBreak], curses.A_BOLD | util.getColor("cyan"))
+ self.addstr(i - self.scroll + 1, numOffset + endBreak, lineText[endBreak:], util.getColor("white"))
+
+ self.refresh()
+ finally:
+ self.lock.release()
+
Modified: arm/trunk/interface/controller.py
===================================================================
--- arm/trunk/interface/controller.py 2009-06-14 02:39:33 UTC (rev 19715)
+++ arm/trunk/interface/controller.py 2009-06-15 06:42:19 UTC (rev 19716)
@@ -1,5 +1,5 @@
#!/usr/bin/env python
-# controller.py -- arm interface (curses monitor for relay status).
+# controller.py -- arm interface (curses monitor for relay status)
# Released under the GPL v3 (http://www.gnu.org/licenses/gpl.html)
"""
@@ -12,7 +12,8 @@
from TorCtl import TorCtl
import util
-import staticPanel
+import headerPanel
+import confPanel
import bandwidthPanel
import logPanel
@@ -21,10 +22,15 @@
# concurrency bugs produce especially sinister glitches)
# enums for message in control label
-CTL_HELP, CTL_PAUSED, CTL_EVENT_INPUT, CTL_EVENT_ERR = range(4)
+CTL_HELP, CTL_PAUSED = range(2)
# panel order per page
-PAGE_1 = ["summary", "control", "bandwidth", "log"]
+PAGE_S = ["header", "control", "popup"] # sticky (ie, always available) page
+PAGES = [
+ ["bandwidth", "log"],
+ ["torrc"]]
+PAUSEABLE = ["header", "bandwidth", "log"]
+PAGE_COUNT = 2 # all page numbering is internally represented as 0-indexed
# TODO: page 2: configuration information
# TODO: page 3: current connections
@@ -33,24 +39,34 @@
def __init__(self, lock):
util.Panel.__init__(self, lock, 1)
- self.msgType = CTL_HELP
- self.arg = ""
+ self.msgText = CTL_HELP # message text to be displyed
+ self.msgAttr = curses.A_NORMAL # formatting attributes
+ self.page = 1 # page number currently being displayed
- def setMsg(self, msgType, arg=""):
- self.msgType = msgType
- self.arg = arg
+ def setMsg(self, msgText, msgAttr=curses.A_NORMAL):
+ """
+ Sets the message and display attributes. If msgType matches CTL_HELP or
+ CTL_PAUSED then uses the default message for those statuses.
+ """
+
+ self.msgText = msgText
+ self.msgAttr = msgAttr
def redraw(self):
if self.win:
self.clear()
- if self.msgType == CTL_HELP: self.addstr(0, 0, "q: quit, e: change events, p: pause")
- elif self.msgType == CTL_PAUSED: self.addstr(0, 0, "Paused", curses.A_STANDOUT)
- elif self.msgType == CTL_EVENT_INPUT: self.addstr(0, 0, "Events to log: ")
- elif self.msgType == CTL_EVENT_ERR: self.addstr(0, 0, self.arg, curses.A_STANDOUT)
- else:
- assert False, "Unrecognized event type for control label: " + str(self.msgType)
+ msgText = self.msgText
+ msgAttr = self.msgAttr
+ if msgText == CTL_HELP:
+ msgText = "page %i / %i - q: quit, p: pause, h: help" % (self.page, PAGE_COUNT)
+ msgAttr = curses.A_NORMAL
+ elif msgText == CTL_PAUSED:
+ msgText = "Paused"
+ msgAttr = curses.A_STANDOUT
+
+ self.addstr(0, 0, msgText, msgAttr)
self.refresh()
def setEventListening(loggedEvents, conn, logListener):
@@ -108,12 +124,13 @@
try: curses.curs_set(0)
except curses.error: pass
- staticInfo = staticPanel.getStaticInfo(conn)
panels = {
- "summary": staticPanel.SummaryPanel(cursesLock, staticInfo),
+ "header": headerPanel.HeaderPanel(cursesLock, conn),
"control": ControlPanel(cursesLock),
+ "popup": util.Panel(cursesLock, 8),
"bandwidth": bandwidthPanel.BandwidthMonitor(cursesLock, conn),
- "log": logPanel.LogMonitor(cursesLock, loggedEvents)}
+ "log": logPanel.LogMonitor(cursesLock, loggedEvents),
+ "torrc": confPanel.ConfPanel(cursesLock, conn.get_info("config-file")["config-file"])}
# listeners that update bandwidth and log panels with Tor status
conn.add_event_listener(panels["log"])
@@ -126,6 +143,7 @@
oldY, oldX = -1, -1
isUnresponsive = False # true if it's been over five seconds since the last BW event (probably due to Tor closing)
isPaused = False # if true updates are frozen
+ page = 0
while True:
# tried only refreshing when the screen was resized but it caused a
@@ -135,12 +153,21 @@
cursesLock.acquire()
try:
y, x = stdscr.getmaxyx()
- if y > oldY:
+ if x > oldX or y > oldY:
# gives panels a chance to take advantage of the maximum bounds
startY = 0
- for panelKey in PAGE_1:
+ for panelKey in PAGE_S[:2]:
panels[panelKey].recreate(stdscr, startY)
startY += panels[panelKey].height
+
+ isChanged = panels["popup"].recreate(stdscr, startY, 80)
+
+ for panelSet in PAGES:
+ tmpStartY = startY
+
+ for panelKey in panelSet:
+ panels[panelKey].recreate(stdscr, tmpStartY)
+ tmpStartY += panels[panelKey].height
# if it's been at least five seconds since the last BW event Tor's probably done
if not isUnresponsive and panels["log"].getHeartbeat() >= 5:
@@ -152,7 +179,7 @@
panels["log"].monitor_event("WARN", "Relay resumed")
# I haven't the foggiest why, but doesn't work if redrawn out of order...
- for panelKey in PAGE_1: panels[panelKey].redraw()
+ for panelKey in (PAGE_S + PAGES[page]): panels[panelKey].redraw()
oldY, oldX = y, x
stdscr.refresh()
finally:
@@ -160,20 +187,73 @@
key = stdscr.getch()
if key == 27 or key == ord('q') or key == ord('Q'): break # quits (also on esc)
- elif key == ord('e') or key == ord('E'):
- # allow user to enter new types of events to log - unchanged if left blank
+ elif key == curses.KEY_LEFT or key == curses.KEY_RIGHT:
+ # switch page
+ if key == curses.KEY_LEFT: page = (page - 1) % PAGE_COUNT
+ else: page = (page + 1) % PAGE_COUNT
+ # pauses panels that aren't visible to prevent events from accumilating
+ # (otherwise they'll wait on the curses lock which might get demanding)
+ for key in PAUSEABLE: panels[key].setPaused(isPaused or key not in PAGES[page])
+
+ panels["control"].page = page + 1
+ panels["control"].refresh()
+ elif key == ord('p') or key == ord('P'):
+ # toggles update freezing
cursesLock.acquire()
try:
- # pauses listeners so events can still be handed (otherwise they wait
- # on curses lock which might get demanding if the user takes their time)
- isBwPaused = panels["bandwidth"].isPaused
- isLogPaused = panels["log"].isPaused
- panels["bandwidth"].setPaused(True)
- panels["log"].setPaused(True)
+ isPaused = not isPaused
+ for key in PAUSEABLE: panels[key].setPaused(isPaused or key not in PAGES[page])
+ msgType = CTL_PAUSED if isPaused else CTL_HELP
+ panels["control"].setMsg(msgType)
+ finally:
+ cursesLock.release()
+ elif key == ord('h') or key == ord('H'):
+ # displays popup for current page's controls
+ cursesLock.acquire()
+ try:
+ for key in PAUSEABLE: panels[key].setPaused(True)
+ panels["control"].setMsg("Press any key...")
+ panels["control"].redraw()
+
+ # lists commands
+ popup = panels["popup"]
+ popup.clear()
+ popup.win.box()
+ popup.addstr(0, 0, "Page %i Commands:" % (page + 1), util.LABEL_ATTR)
+
+ if page == 0:
+ popup.addstr(1, 2, "e: change logged events")
+ elif page == 1:
+ popup.addstr(1, 2, "up arrow: scroll up a line")
+ popup.addstr(1, 35, "down arrow: scroll down a line")
+ popup.addstr(2, 2, "page up: scroll up a page")
+ popup.addstr(2, 35, "page down: scroll down a page")
+ popup.addstr(3, 2, "s: toggle comment stripping")
+ popup.addstr(3, 35, "n: toggle line numbering")
+ popup.addstr(4, 2, "r: reload torrc")
+
+ popup.refresh()
+
+ curses.cbreak()
+ stdscr.getch()
+ curses.halfdelay(REFRESH_RATE * 10)
+
+ msgType = CTL_PAUSED if isPaused else CTL_HELP
+ panels["control"].setMsg(msgType)
+
+ for key in PAUSEABLE: panels[key].setPaused(isPaused or key not in PAGES[page])
+ finally:
+ cursesLock.release()
+ elif page == 0 and (key == ord('e') or key == ord('E')):
+ # allow user to enter new types of events to log - unchanged if left blank
+ cursesLock.acquire()
+ try:
+ for key in PAUSEABLE: panels[key].setPaused(True)
+
# provides prompt
- panels["control"].setMsg(CTL_EVENT_INPUT)
+ panels["control"].setMsg("Events to log: ")
panels["control"].redraw()
# makes cursor and typing visible
@@ -181,16 +261,16 @@
except curses.error: pass
curses.echo()
- # switches bandwidth area to list event types
- bwPanel = panels["bandwidth"]
- bwPanel.clear()
- bwPanel.addstr(0, 0, "Event Types:", util.LABEL_ATTR)
+ # lists event types
+ popup = panels["popup"]
+ popup.clear()
+ popup.addstr(0, 0, "Event Types:", util.LABEL_ATTR)
lineNum = 1
for line in logPanel.EVENT_LISTING.split("\n"):
line = line.strip()
- bwPanel.addstr(lineNum, 0, line[:x - 1])
+ popup.addstr(lineNum, 0, line[:x - 1])
lineNum += 1
- bwPanel.refresh()
+ popup.refresh()
# gets user input (this blocks monitor updates)
eventsInput = panels["control"].win.getstr(0, 15)
@@ -200,6 +280,7 @@
try: curses.curs_set(0)
except curses.error: pass
curses.noecho()
+ curses.halfdelay(REFRESH_RATE * 10) # evidenlty previous tweaks reset this...
# TODO: it would be nice to quit on esc, but looks like this might not be possible...
if eventsInput != "":
@@ -208,29 +289,18 @@
loggedEvents = setEventListening(expandedEvents, conn, panels["log"])
panels["log"].loggedEvents = loggedEvents
except ValueError, exc:
- panels["control"].setMsg(CTL_EVENT_ERR, "Invalid flags: %s" % str(exc))
+ panels["control"].setMsg("Invalid flags: %s" % str(exc), curses.A_STANDOUT)
panels["control"].redraw()
time.sleep(2)
msgType = CTL_PAUSED if isPaused else CTL_HELP
panels["control"].setMsg(msgType)
- # returns listeners to previous pause status
- panels["bandwidth"].setPaused(isBwPaused)
- panels["log"].setPaused(isLogPaused)
+ for key in PAUSEABLE: panels[key].setPaused(isPaused or key not in PAGES[page])
finally:
cursesLock.release()
- elif key == ord('p') or key == ord('P'):
- # toggles update freezing
- cursesLock.acquire()
- try:
- isPaused = not isPaused
- panels["log"].setPaused(isPaused)
- panels["bandwidth"].setPaused(isPaused)
- msgType = CTL_PAUSED if isPaused else CTL_HELP
- panels["control"].setMsg(msgType)
- finally:
- cursesLock.release()
+ elif page == 1:
+ panels["torrc"].handleKey(key)
def startTorMonitor(conn, loggedEvents):
curses.wrapper(drawTorMonitor, conn, loggedEvents)
Added: arm/trunk/interface/headerPanel.py
===================================================================
--- arm/trunk/interface/headerPanel.py (rev 0)
+++ arm/trunk/interface/headerPanel.py 2009-06-15 06:42:19 UTC (rev 19716)
@@ -0,0 +1,154 @@
+#!/usr/bin/env python
+# summaryPanel.py -- Static system and Tor related information.
+# Released under the GPL v3 (http://www.gnu.org/licenses/gpl.html)
+
+import os
+import curses
+from TorCtl import TorCtl
+
+import util
+
+class HeaderPanel(util.Panel):
+ """
+ Draws top area containing static information.
+
+ arm - <System Name> (<OS> <Version>) Tor <Tor Version>
+ <Relay Nickname> - <IP Addr>:<ORPort>, [Dir Port: <DirPort>, ]Control Port (<open, password, cookie>): <ControlPort>
+ cpu: <cpu%> mem: <mem> (<mem%>) uid: <uid> uptime: <upmin>:<upsec>
+ fingerprint: <Fingerprint>
+
+ Example:
+ arm - odin (Linux 2.6.24-24-generic) Tor 0.2.1.15-rc
+ odin - 76.104.132.98:9001, Dir Port: 9030, Control Port (cookie): 9051
+ cpu: 14.6% mem: 42 MB (4.2%) pid: 20060 uptime: 48:27
+ fingerprint: BDAD31F6F318E0413833E8EBDA956F76E4D66788
+ """
+
+ def __init__(self, lock, conn):
+ util.Panel.__init__(self, lock, 5)
+ self.vals = [] # mapping of information to be presented
+ self.conn = conn # Tor control port connection
+ self.isPaused = False
+ self._updateParams()
+
+ def redraw(self):
+ if self.win:
+ if not self.isPaused: self._updateParams()
+
+ # extra erase/refresh is needed to avoid internal caching screwing up and
+ # refusing to redisplay content in the case of graphical glitches - probably
+ # an obscure curses bug...
+ self.win.erase()
+ self.win.refresh()
+
+ self.clear()
+
+ # Line 1
+ self.addstr(0, 0, "arm - %s (%s %s)" % (self.vals["sys-name"], self.vals["sys-os"], self.vals["sys-version"]))
+ self.addstr(0, 45, "Tor %s" % self.vals["version"])
+
+ # Line 2 (authentication label red if open, green if credentials required)
+ dirPortLabel = "Dir Port: %s, " % self.vals["DirPort"] if not self.vals["DirPort"] == None else ""
+
+ # TODO: if both cookie and password are set then which takes priority?
+ if self.vals["IsPasswordAuthSet"]: controlPortAuthLabel = "password"
+ elif self.vals["IsCookieAuthSet"]: controlPortAuthLabel = "cookie"
+ else: controlPortAuthLabel = "open"
+ controlPortAuthColor = "red" if controlPortAuthLabel == "open" else "green"
+
+ labelStart = "%s - %s:%s, %sControl Port (" % (self.vals["Nickname"], self.vals["address"], self.vals["ORPort"], dirPortLabel)
+ self.addstr(1, 0, labelStart)
+ xLoc = len(labelStart)
+ self.addstr(1, xLoc, controlPortAuthLabel, util.getColor(controlPortAuthColor))
+ xLoc += len(controlPortAuthLabel)
+ self.addstr(1, xLoc, "): %s" % self.vals["ControlPort"])
+
+ # Line 3 (system usage info)
+ self.addstr(2, 0, "cpu: %s%%" % self.vals["%cpu"])
+ self.addstr(2, 13, "mem: %s (%s%%)" % (util.getSizeLabel(int(self.vals["rss"]) * 1024), self.vals["%mem"]))
+ self.addstr(2, 34, "pid: %s" % self.vals["pid"])
+ self.addstr(2, 47, "uptime: %s" % self.vals["etime"])
+
+ # Line 4 (fingerprint)
+ self.addstr(3, 0, "fingerprint: %s" % self.vals["fingerprint"])
+
+
+ # Lines 3-5
+ #self.addstr(3, 0, "Config: %s" % self.vals["config-file"])
+ #exitPolicy = self.vals["ExitPolicy"]
+
+ # adds note when default exit policy is appended
+ #if exitPolicy == None: exitPolicy = "<default>"
+ #elif not exitPolicy.endswith("accept *:*") and not exitPolicy.endswith("reject *:*"):
+ # exitPolicy += ", <default>"
+ #self.addstr(4, 0, "Exit Policy: %s" % exitPolicy)
+
+ self.refresh()
+
+ def setPaused(self, isPause):
+ """
+ If true, prevents updates from being presented.
+ """
+
+ self.isPaused = isPause
+
+ def _updateParams(self):
+ """
+ Updates mapping of static Tor settings and system information to their
+ corresponding string values. Keys include:
+ info - version, config-file, *address, *fingerprint
+ sys - sys-name, sys-os, sys-version
+ ps - *%cpu, *rss, *%mem, *pid, *etime
+ config - Nickname, ORPort, DirPort, ControlPort, ExitPolicy
+ config booleans - IsPasswordAuthSet, IsCookieAuthSet, IsAccountingEnabled
+
+ * volatile parameter that'll be reset (otherwise won't be checked if
+ already set)
+ """
+
+ if not self.vals:
+ # retrieves static params
+ self.vals = self.conn.get_info(["version", "config-file"])
+
+ # populates with some basic system information
+ unameVals = os.uname()
+ self.vals["sys-name"] = unameVals[1]
+ self.vals["sys-os"] = unameVals[0]
+ self.vals["sys-version"] = unameVals[2]
+
+ # parameters from the user's torrc
+ configFields = ["Nickname", "ORPort", "DirPort", "ControlPort", "ExitPolicy"]
+ self.vals.update(dict([(key, self.conn.get_option(key)[0][1]) for key in configFields]))
+
+ # simply keeps booleans for if authentication info is set
+ self.vals["IsPasswordAuthSet"] = not self.conn.get_option("HashedControlPassword")[0][1] == None
+ self.vals["IsCookieAuthSet"] = self.conn.get_option("CookieAuthentication")[0][1] == "1"
+
+ self.vals["IsAccountingEnabled"] = self.conn.get_info('accounting/enabled')['accounting/enabled'] == "1"
+
+ # gets parameters that throw errors if unavailable
+ for param in ["address", "fingerprint"]:
+ try: self.vals.update(self.conn.get_info(param))
+ except TorCtl.ErrorReply: self.vals[param] = "Unknown"
+ except TorCtl.TorCtlClosed:
+ # Tor shut down - keep last known values
+ if not self.vals[param]: self.vals[param] = "Unknown"
+
+ # ps call provides header followed by params for tor
+ psParams = ["%cpu", "rss", "%mem", "pid", "etime"]
+ psCall = os.popen('ps -C %s -o %s' % ("tor", ",".join(psParams)))
+
+ try: sampling = psCall.read().strip().split()[len(psParams):]
+ except IOError: sampling = [] # ps call failed
+ psCall.close()
+
+ if len(sampling) < 5:
+ # either ps failed or returned no tor instance
+ sampling = [""] * len(psParams)
+
+ # %cpu, rss, and %mem are better zeroed out
+ for i in range(3): sampling[i] = "0"
+
+ for i in range(len(psParams)):
+ self.vals[psParams[i]] = sampling[i]
+
Deleted: arm/trunk/interface/staticPanel.py
===================================================================
--- arm/trunk/interface/staticPanel.py 2009-06-14 02:39:33 UTC (rev 19715)
+++ arm/trunk/interface/staticPanel.py 2009-06-15 06:42:19 UTC (rev 19716)
@@ -1,114 +0,0 @@
-#!/usr/bin/env python
-# summaryPanel.py -- Static system and Tor related information.
-# Released under the GPL v3 (http://www.gnu.org/licenses/gpl.html)
-
-import os
-import curses
-from TorCtl import TorCtl
-
-import util
-
-def getStaticInfo(conn):
- """
- Provides mapping of static Tor settings and system information to their
- corresponding string values. Keys include:
- info - version, config-file, address, fingerprint
- sys - sys-name, sys-os, sys-version
- config - Nickname, ORPort, DirPort, ControlPort, ExitPolicy
- config booleans - IsPasswordAuthSet, IsCookieAuthSet, IsAccountingEnabled
- """
-
- vals = conn.get_info(["version", "config-file"])
-
- # gets parameters that throw errors if unavailable
- for param in ["address", "fingerprint"]:
- try:
- vals.update(conn.get_info(param))
- except TorCtl.ErrorReply:
- vals[param] = "Unknown"
-
- # populates with some basic system information
- unameVals = os.uname()
- vals["sys-name"] = unameVals[1]
- vals["sys-os"] = unameVals[0]
- vals["sys-version"] = unameVals[2]
-
- # parameters from the user's torrc
- configFields = ["Nickname", "ORPort", "DirPort", "ControlPort", "ExitPolicy"]
- vals.update(dict([(key, conn.get_option(key)[0][1]) for key in configFields]))
-
- # simply keeps booleans for if authentication info is set
- vals["IsPasswordAuthSet"] = not conn.get_option("HashedControlPassword")[0][1] == None
- vals["IsCookieAuthSet"] = conn.get_option("CookieAuthentication")[0][1] == "1"
-
- vals["IsAccountingEnabled"] = conn.get_info('accounting/enabled')['accounting/enabled'] == "1"
-
- return vals
-
-class SummaryPanel(util.Panel):
- """
- Draws top area containing static information.
-
- arm - <System Name> (<OS> <Version>) Tor <Tor Version>
- <Relay Nickname> - <IP Addr>:<ORPort>, [Dir Port: <DirPort>, ]Control Port (<open, password, cookie>): <ControlPort>
- Fingerprint: <Fingerprint>
- Config: <Config>
- Exit Policy: <ExitPolicy>
-
- Example:
- arm - odin (Linux 2.6.24-24-generic) Tor 0.2.0.34 (r18423)
- odin - 76.104.132.98:9001, Dir Port: 9030, Control Port (cookie): 9051
- Fingerprint: BDAD31F6F318E0413833E8EBDA956F76E4D66788
- Config: /home/atagar/.vidalia/torrc
- Exit Policy: reject *:*
- """
-
- def __init__(self, lock, vals):
- util.Panel.__init__(self, lock, 6)
- self.vals = vals # mapping of information to be presented
-
- def redraw(self):
- i = 1
-
- if self.win:
- # extra erase/refresh is needed to avoid internal caching screwing up and
- # refusing to redisplay content in the case of graphical glitches - probably
- # an obscure curses bug...
- self.win.erase()
- self.win.refresh()
-
- self.clear()
-
- # Line 1
- self.addstr(0, 0, "arm - %s (%s %s)" % (self.vals["sys-name"], self.vals["sys-os"], self.vals["sys-version"]))
- self.addstr(0, 45, "Tor %s" % self.vals["version"])
-
- # Line 2 (authentication label red if open, green if credentials required)
- dirPortLabel = "Dir Port: %s, " % self.vals["DirPort"] if not self.vals["DirPort"] == None else ""
-
- # TODO: if both cookie and password are set then which takes priority?
- if self.vals["IsPasswordAuthSet"]: controlPortAuthLabel = "password"
- elif self.vals["IsCookieAuthSet"]: controlPortAuthLabel = "cookie"
- else: controlPortAuthLabel = "open"
- controlPortAuthColor = "red" if controlPortAuthLabel == "open" else "green"
-
- labelStart = "%s - %s:%s, %sControl Port (" % (self.vals["Nickname"], self.vals["address"], self.vals["ORPort"], dirPortLabel)
- self.addstr(1, 0, labelStart)
- xLoc = len(labelStart)
- self.addstr(1, xLoc, controlPortAuthLabel, util.getColor(controlPortAuthColor))
- xLoc += len(controlPortAuthLabel)
- self.addstr(1, xLoc, "): %s" % self.vals["ControlPort"])
-
- # Lines 3-5
- self.addstr(2, 0, "Fingerprint: %s" % self.vals["fingerprint"])
- self.addstr(3, 0, "Config: %s" % self.vals["config-file"])
- exitPolicy = self.vals["ExitPolicy"]
-
- # adds note when default exit policy is appended
- if exitPolicy == None: exitPolicy = "<default>"
- elif not exitPolicy.endswith("accept *:*") and not exitPolicy.endswith("reject *:*"):
- exitPolicy += ", <default>"
- self.addstr(4, 0, "Exit Policy: %s" % exitPolicy)
-
- self.refresh()
-
Modified: arm/trunk/interface/util.py
===================================================================
--- arm/trunk/interface/util.py 2009-06-14 02:39:33 UTC (rev 19715)
+++ arm/trunk/interface/util.py 2009-06-15 06:42:19 UTC (rev 19716)
@@ -79,7 +79,7 @@
def redraw(self):
pass # overwritten by implementations
- def recreate(self, stdscr, startY):
+ def recreate(self, stdscr, startY, maxX=-1):
"""
Creates a new subwindow for the panel if:
- panel currently doesn't have a subwindow
@@ -105,11 +105,12 @@
newHeight = y - startY
if self.height != -1: newHeight = min(newHeight, self.height)
- if self.startY != startY or newHeight > self.maxY or self.isDisplaced:
+ if self.startY != startY or newHeight > self.maxY or self.isDisplaced or (self.maxX > maxX and maxX != -1):
# window growing or moving - recreate
self.startY = startY
startY = min(startY, y - 1) # better create a displaced window than leave it as None
-
+ if maxX != -1: x = min(x, maxX)
+
self.win = stdscr.subwin(newHeight, x, startY, 0)
return True
else: return False