Try not to hang when user views large request & response bodies

Two different strategies here:

    - Use a simple heuristic to detect if we're looking at XML data when indent
    mode is used. On non-XML data we can hang even on small documents.

    - Only view partial data for large bodies. At the moment the cutoff is
    100k. I might finetune this later.
This commit is contained in:
Aldo Cortesi 2011-06-27 15:59:17 +12:00
parent 2ae7808ca9
commit f004326855
3 changed files with 47 additions and 5 deletions

View File

@ -20,6 +20,8 @@ import urwid.raw_display
import urwid
import controller, utils, filt, proxy, flow
VIEW_CUTOFF = 1024*100
class Stop(Exception): pass
@ -879,14 +881,27 @@ class ConsoleMaster(flow.FlowMaster):
self.spawn_external_viewer(serr, None)
self.refresh_connection(f)
def _trailer(self, content, txt):
rem = len(content) - VIEW_CUTOFF
if rem > 0:
txt.append(urwid.Text(""))
txt.append(
urwid.Text(
[
("highlight", "... %s of data not shown"%utils.pretty_size(rem))
]
)
)
def _view_conn_normal(self, content, txt):
for i in content.splitlines():
for i in content[:VIEW_CUTOFF].splitlines():
txt.append(
urwid.Text(("text", i))
)
self._trailer(content, txt)
def _view_conn_binary(self, content, txt):
for offset, hex, s in utils.hexdump(content):
for offset, hex, s in utils.hexdump(content[:VIEW_CUTOFF]):
txt.append(urwid.Text([
("offset", offset),
" ",
@ -894,12 +909,14 @@ class ConsoleMaster(flow.FlowMaster):
" ",
("text", s),
]))
self._trailer(content, txt)
def _view_conn_pretty(self, content, txt):
for i in utils.pretty_xmlish(content):
for i in utils.pretty_xmlish(content[:VIEW_CUTOFF]):
txt.append(
urwid.Text(("text", i)),
)
self._trailer(content, txt)
@utils.LRUCache(20)
def _cached_conn_text(self, content, hdrItems, viewmode):
@ -918,12 +935,18 @@ class ConsoleMaster(flow.FlowMaster):
if viewmode == VIEW_BODY_BINARY:
self._view_conn_binary(content, txt)
elif viewmode == VIEW_BODY_INDENT:
if utils.isXML(content):
self._view_conn_pretty(content, txt)
else:
if utils.isBin(content):
self._view_conn_binary(content, txt)
else:
self._view_conn_normal(content, txt)
else:
if utils.isBin(content):
self._view_conn_binary(content, txt)
else:
self._view_conn_normal(content, txt)
return urwid.ListBox(txt)
def _readflow(self, path):

View File

@ -44,6 +44,16 @@ def isBin(s):
return False
def isXML(s):
for i in s:
if i in "\n \t":
continue
elif i == "<":
return True
else:
return False
def cleanBin(s):
parts = []
for i in s:
@ -51,6 +61,7 @@ def cleanBin(s):
if o > 31 and o < 127:
parts.append(i)
else:
if i not in "\n\r\t":
parts.append(".")
return "".join(parts)

View File

@ -18,6 +18,13 @@ class uisBin(libpry.AutoTree):
assert utils.isBin("testing\x7f")
class uisXML(libpry.AutoTree):
def test_simple(self):
assert not utils.isXML("foo")
assert utils.isXML("<foo")
assert utils.isXML(" \n<foo")
class uhexdump(libpry.AutoTree):
def test_simple(self):
assert utils.hexdump("one\0"*10)
@ -352,6 +359,7 @@ class uLRUCache(libpry.AutoTree):
tests = [
uformat_timestamp(),
uisBin(),
uisXML(),
uhexdump(),
upretty_size(),
uisStringLike(),