456 lines
No EOL
14 KiB
Python
456 lines
No EOL
14 KiB
Python
import re, time, cgi, os
|
|
from Plugin import PluginManager
|
|
from Config import config
|
|
|
|
@PluginManager.registerTo("UiRequest")
|
|
class UiRequestPlugin(object):
|
|
def formatTableRow(self, row):
|
|
back = []
|
|
for format, val in row:
|
|
if val == None:
|
|
formatted = "n/a"
|
|
elif format == "since":
|
|
if val:
|
|
formatted = "%.0f" % (time.time()-val)
|
|
else:
|
|
formatted = "n/a"
|
|
else:
|
|
formatted = format % val
|
|
back.append("<td>%s</td>" % formatted)
|
|
return "<tr>%s</tr>" % "".join(back)
|
|
|
|
|
|
def getObjSize(self, obj, hpy = None):
|
|
if hpy:
|
|
return float(hpy.iso(obj).domisize)/1024
|
|
else:
|
|
return 0
|
|
|
|
|
|
# /Stats entry point
|
|
def actionStats(self):
|
|
import gc, sys
|
|
from Ui import UiRequest
|
|
|
|
hpy = None
|
|
if self.get.get("size") == "1": # Calc obj size
|
|
try:
|
|
import guppy
|
|
hpy = guppy.hpy()
|
|
except:
|
|
pass
|
|
self.sendHeader()
|
|
s = time.time()
|
|
main = sys.modules["main"]
|
|
|
|
# Style
|
|
yield """
|
|
<style>
|
|
* { font-family: monospace }
|
|
table * { text-align: right; padding: 0px 10px }
|
|
</style>
|
|
"""
|
|
|
|
# Memory
|
|
try:
|
|
yield "rev%s | " % config.rev
|
|
yield "IP external: %s | " % config.ip_external
|
|
yield "Opened: %s | " % main.file_server.port_opened
|
|
yield "Recv: %.2fMB, Sent: %.2fMB | " % (float(main.file_server.bytes_recv)/1024/1024, float(main.file_server.bytes_sent)/1024/1024)
|
|
yield "Peerid: %s | " % main.file_server.peer_id
|
|
import psutil
|
|
process = psutil.Process(os.getpid())
|
|
mem = process.get_memory_info()[0] / float(2 ** 20)
|
|
yield "Mem: %.2fMB | " % mem
|
|
yield "Threads: %s | " % len(process.threads())
|
|
yield "CPU: usr %.2fs sys %.2fs | " % process.cpu_times()
|
|
yield "Open files: %s | " % len(process.open_files())
|
|
yield "Sockets: %s | " % len(process.connections())
|
|
yield "Calc size <a href='?size=1'>on</a> <a href='?size=0'>off</a>"
|
|
except Exception, err:
|
|
pass
|
|
yield "<br>"
|
|
|
|
# Connections
|
|
yield "<b>Connections</b> (%s, total made: %s):<br>" % (len(main.file_server.connections), main.file_server.last_connection_id)
|
|
yield "<table><tr> <th>id</th> <th>protocol</th> <th>type</th> <th>ip</th> <th>open</th> <th>ping</th> <th>buff</th>"
|
|
yield "<th>idle</th> <th>open</th> <th>delay</th> <th>sent</th> <th>received</th> <th>last sent</th> <th>waiting</th> <th>version</th> <th>peerid</th> </tr>"
|
|
for connection in main.file_server.connections:
|
|
yield self.formatTableRow([
|
|
("%3d", connection.id),
|
|
("%s", connection.protocol),
|
|
("%s", connection.type),
|
|
("%s:%s", (connection.ip, connection.port)),
|
|
("%s", connection.handshake.get("port_opened")),
|
|
("%6.3f", connection.last_ping_delay),
|
|
("%s", connection.incomplete_buff_recv),
|
|
("since", max(connection.last_send_time, connection.last_recv_time)),
|
|
("since", connection.start_time),
|
|
("%.3f", connection.last_sent_time-connection.last_send_time),
|
|
("%.0fkB", connection.bytes_sent/1024),
|
|
("%.0fkB", connection.bytes_recv/1024),
|
|
("%s", connection.last_cmd),
|
|
("%s", connection.waiting_requests.keys()),
|
|
("%s r%s", (connection.handshake.get("version"), connection.handshake.get("rev", "?")) ),
|
|
("%s", connection.handshake.get("peer_id")),
|
|
])
|
|
yield "</table>"
|
|
|
|
|
|
# Sites
|
|
yield "<br><br><b>Sites</b>:"
|
|
yield "<table>"
|
|
yield "<tr><th>address</th> <th>connected</th> <th>peers</th> <th>content.json</th> </tr>"
|
|
for site in self.server.sites.values():
|
|
yield self.formatTableRow([
|
|
("<a href='#ShowPeers' onclick='document.getElementById(\"peers_%s\").style.display=\"initial\"; return false'>%s</a>", (site.address, site.address)),
|
|
("%s", [peer.connection.id for peer in site.peers.values() if peer.connection and peer.connection.connected]),
|
|
("%s/%s/%s", (
|
|
len([peer for peer in site.peers.values() if peer.connection and peer.connection.connected]),
|
|
len(site.getConnectablePeers(100)),
|
|
len(site.peers)
|
|
) ),
|
|
("%s", len(site.content_manager.contents)),
|
|
])
|
|
yield "<tr><td id='peers_%s' style='display: none; white-space: pre'>" % site.address
|
|
for key, peer in site.peers.items():
|
|
yield "(%s, err: %s) %22s -<br>" % (peer.connection, peer.connection_error, key)
|
|
yield "<br></td></tr>"
|
|
yield "</table>"
|
|
|
|
|
|
# Objects
|
|
|
|
obj_count = {}
|
|
for obj in gc.get_objects():
|
|
obj_type = str(type(obj))
|
|
if not obj_type in obj_count:
|
|
obj_count[obj_type] = [0, 0]
|
|
obj_count[obj_type][0] += 1 # Count
|
|
obj_count[obj_type][1] += float(sys.getsizeof(obj))/1024 # Size
|
|
|
|
yield "<br><br><b>Objects in memory (types: %s, total: %s, %.2fkb):</b><br>" % (len(obj_count), sum([stat[0] for stat in obj_count.values()]), sum([stat[1] for stat in obj_count.values()]))
|
|
|
|
for obj, stat in sorted(obj_count.items(), key=lambda x: x[1][0], reverse=True): # Sorted by count
|
|
yield " - %.1fkb = %s x <a href=\"/Listobj?type=%s\">%s</a><br>" % (stat[1], stat[0], obj, cgi.escape(obj))
|
|
|
|
|
|
from greenlet import greenlet
|
|
objs = [obj for obj in gc.get_objects() if isinstance(obj, greenlet)]
|
|
yield "<br>Greenlets (%s):<br>" % len(objs)
|
|
for obj in objs:
|
|
yield " - %.1fkb: %s<br>" % (self.getObjSize(obj, hpy), cgi.escape(repr(obj)))
|
|
|
|
|
|
from Worker import Worker
|
|
objs = [obj for obj in gc.get_objects() if isinstance(obj, Worker)]
|
|
yield "<br>Workers (%s):<br>" % len(objs)
|
|
for obj in objs:
|
|
yield " - %.1fkb: %s<br>" % (self.getObjSize(obj, hpy), cgi.escape(repr(obj)))
|
|
|
|
|
|
from Connection import Connection
|
|
objs = [obj for obj in gc.get_objects() if isinstance(obj, Connection)]
|
|
yield "<br>Connections (%s):<br>" % len(objs)
|
|
for obj in objs:
|
|
yield " - %.1fkb: %s<br>" % (self.getObjSize(obj, hpy), cgi.escape(repr(obj)))
|
|
|
|
from socket import socket
|
|
objs = [obj for obj in gc.get_objects() if isinstance(obj, socket)]
|
|
yield "<br>Sockets (%s):<br>" % len(objs)
|
|
for obj in objs:
|
|
yield " - %.1fkb: %s<br>" % (self.getObjSize(obj, hpy), cgi.escape(repr(obj)))
|
|
|
|
from msgpack import Unpacker
|
|
objs = [obj for obj in gc.get_objects() if isinstance(obj, Unpacker)]
|
|
yield "<br>Msgpack unpacker (%s):<br>" % len(objs)
|
|
for obj in objs:
|
|
yield " - %.1fkb: %s<br>" % (self.getObjSize(obj, hpy), cgi.escape(repr(obj)))
|
|
|
|
|
|
from Site import Site
|
|
objs = [obj for obj in gc.get_objects() if isinstance(obj, Site)]
|
|
yield "<br>Sites (%s):<br>" % len(objs)
|
|
for obj in objs:
|
|
yield " - %.1fkb: %s<br>" % (self.getObjSize(obj, hpy), cgi.escape(repr(obj)))
|
|
|
|
|
|
objs = [obj for obj in gc.get_objects() if isinstance(obj, self.server.log.__class__)]
|
|
yield "<br>Loggers (%s):<br>" % len(objs)
|
|
for obj in objs:
|
|
yield " - %.1fkb: %s<br>" % (self.getObjSize(obj, hpy), cgi.escape(repr(obj.name)))
|
|
|
|
|
|
objs = [obj for obj in gc.get_objects() if isinstance(obj, UiRequest)]
|
|
yield "<br>UiRequests (%s):<br>" % len(objs)
|
|
for obj in objs:
|
|
yield " - %.1fkb: %s<br>" % (self.getObjSize(obj, hpy), cgi.escape(repr(obj)))
|
|
|
|
|
|
from Peer import Peer
|
|
objs = [obj for obj in gc.get_objects() if isinstance(obj, Peer)]
|
|
yield "<br>Peers (%s):<br>" % len(objs)
|
|
for obj in objs:
|
|
yield " - %.1fkb: %s<br>" % (self.getObjSize(obj, hpy), cgi.escape(repr(obj)))
|
|
|
|
|
|
objs = [(key, val) for key, val in sys.modules.iteritems() if val is not None]
|
|
objs.sort()
|
|
yield "<br>Modules (%s):<br>" % len(objs)
|
|
for module_name, module in objs:
|
|
yield " - %.3fkb: %s %s<br>" % (self.getObjSize(module, hpy), module_name, cgi.escape(repr(module)))
|
|
|
|
yield "Done in %.1f" % (time.time()-s)
|
|
|
|
|
|
def actionListobj(self):
|
|
import gc, sys
|
|
|
|
self.sendHeader()
|
|
type_filter = self.get.get("type")
|
|
|
|
yield """
|
|
<style>
|
|
* { font-family: monospace; white-space: pre }
|
|
table * { text-align: right; padding: 0px 10px }
|
|
</style>
|
|
"""
|
|
|
|
yield "Listing all %s objects in memory...<br>" % cgi.escape(type_filter)
|
|
|
|
ref_count = {}
|
|
objs = gc.get_objects()
|
|
for obj in objs:
|
|
obj_type = str(type(obj))
|
|
if obj_type != type_filter: continue
|
|
refs = [ref for ref in gc.get_referrers(obj) if hasattr(ref, "__class__") and ref.__class__.__name__ not in ["list", "dict", "function", "type", "frame", "WeakSet", "tuple"]]
|
|
if not refs: continue
|
|
yield "%.1fkb <span title=\"%s\">%s</span>... " % (float(sys.getsizeof(obj))/1024, cgi.escape(str(obj)), cgi.escape(str(obj)[0:100].ljust(100)) )
|
|
for ref in refs:
|
|
yield " ["
|
|
if "object at" in str(ref) or len(str(ref)) > 100:
|
|
yield str(ref.__class__.__name__)
|
|
else:
|
|
yield str(ref.__class__.__name__)+":"+cgi.escape(str(ref))
|
|
yield "] "
|
|
ref_type = ref.__class__.__name__
|
|
if ref_type not in ref_count:
|
|
ref_count[ref_type] = [0,0]
|
|
ref_count[ref_type][0] += 1 # Count
|
|
ref_count[ref_type][1] += float(sys.getsizeof(obj))/1024 # Size
|
|
yield "<br>"
|
|
|
|
yield "<br>Object referrer (total: %s, %.2fkb):<br>" % (len(ref_count), sum([stat[1] for stat in ref_count.values()]))
|
|
|
|
for obj, stat in sorted(ref_count.items(), key=lambda x: x[1][0], reverse=True)[0:30]: # Sorted by count
|
|
yield " - %.1fkb = %s x %s<br>" % (stat[1], stat[0], cgi.escape(str(obj)) )
|
|
|
|
|
|
def actionBenchmark(self):
|
|
import sys
|
|
from contextlib import contextmanager
|
|
|
|
output = self.sendHeader()
|
|
|
|
@contextmanager
|
|
def benchmark(name, standard):
|
|
s = time.time()
|
|
output("- %s" % name)
|
|
try:
|
|
yield 1
|
|
except Exception, err:
|
|
output("<br><b>! Error: %s</b><br>" % err)
|
|
taken = time.time()-s
|
|
multipler = standard/taken
|
|
if multipler < 0.3: speed = "Sloooow"
|
|
elif multipler < 0.5: speed = "Ehh"
|
|
elif multipler < 0.8: speed = "Goodish"
|
|
elif multipler < 1.2: speed = "OK"
|
|
elif multipler < 1.7: speed = "Fine"
|
|
elif multipler < 2.5: speed = "Fast"
|
|
elif multipler < 3.5: speed = "WOW"
|
|
else: speed = "Insane!!"
|
|
output("%.3fs [x%.2f: %s]<br>" % (taken, multipler, speed))
|
|
time.sleep(0.01)
|
|
|
|
|
|
yield """
|
|
<style>
|
|
* { font-family: monospace }
|
|
table * { text-align: right; padding: 0px 10px }
|
|
</style>
|
|
"""
|
|
|
|
yield "Benchmarking ZeroNet %s (rev%s) Python %s, platform: %s...<br>" % (config.version, config.rev, sys.version, sys.platform)
|
|
|
|
t = time.time()
|
|
|
|
yield "<br>CryptBitcoin:<br>"
|
|
from Crypt import CryptBitcoin
|
|
|
|
# seed = CryptBitcoin.newSeed()
|
|
# yield "- Seed: %s<br>" % seed
|
|
seed = "e180efa477c63b0f2757eac7b1cce781877177fe0966be62754ffd4c8592ce38"
|
|
|
|
with benchmark("hdPrivatekey x 10", 0.7):
|
|
for i in range(10):
|
|
privatekey = CryptBitcoin.hdPrivatekey(seed, i*10)
|
|
yield "."
|
|
valid = "5JsunC55XGVqFQj5kPGK4MWgTL26jKbnPhjnmchSNPo75XXCwtk"
|
|
assert privatekey == valid, "%s != %s" % (privatekey, valid)
|
|
|
|
|
|
data = "Hello"*1024 #5k
|
|
with benchmark("sign x 10", 0.35):
|
|
for i in range(10):
|
|
yield "."
|
|
sign = CryptBitcoin.sign(data, privatekey)
|
|
valid = "HFGXaDauZ8vX/N9Jn+MRiGm9h+I94zUhDnNYFaqMGuOi+4+BbWHjuwmx0EaKNV1G+kP0tQDxWu0YApxwxZbSmZU="
|
|
assert sign == valid, "%s != %s" % (sign, valid)
|
|
|
|
|
|
address = CryptBitcoin.privatekeyToAddress(privatekey)
|
|
if CryptBitcoin.opensslVerify: # Openssl avalible
|
|
with benchmark("openssl verify x 100", 0.37):
|
|
for i in range(100):
|
|
if i%10==0: yield "."
|
|
ok = CryptBitcoin.verify(data, address, sign)
|
|
assert ok, "does not verify from %s" % address
|
|
else:
|
|
yield " - openssl verify x 100...not avalible :(<br>"
|
|
|
|
opensslVerify_bk = CryptBitcoin.opensslVerify # Emulate openssl not found in any way
|
|
CryptBitcoin.opensslVerify = None
|
|
with benchmark("pure-python verify x 10", 1.6):
|
|
for i in range(10):
|
|
yield "."
|
|
ok = CryptBitcoin.verify(data, address, sign)
|
|
assert ok, "does not verify from %s" % address
|
|
CryptBitcoin.opensslVerify = opensslVerify_bk
|
|
|
|
|
|
yield "<br>CryptHash:<br>"
|
|
from Crypt import CryptHash
|
|
from cStringIO import StringIO
|
|
|
|
data = StringIO("Hello"*1024*1024) #5m
|
|
with benchmark("sha512 x 10 000", 1):
|
|
for i in range(10):
|
|
for y in range(10000):
|
|
hash = CryptHash.sha512sum(data)
|
|
yield "."
|
|
valid = "cf83e1357eefb8bdf1542850d66d8007d620e4050b5715dc83f4a921d36ce9ce"
|
|
assert hash == valid, "%s != %s" % (hash, valid)
|
|
|
|
|
|
yield "<br>Db:<br>"
|
|
from Db import Db
|
|
|
|
schema = {
|
|
"db_name": "TestDb",
|
|
"db_file": "data/benchmark.db",
|
|
"maps": {
|
|
".*": {
|
|
"to_table": {
|
|
"test": "test"
|
|
}
|
|
}
|
|
},
|
|
"tables": {
|
|
"test": {
|
|
"cols": [
|
|
["test_id", "INTEGER"],
|
|
["title", "TEXT"],
|
|
["json_id", "INTEGER REFERENCES json (json_id)"]
|
|
],
|
|
"indexes": ["CREATE UNIQUE INDEX test_key ON test(test_id, json_id)"],
|
|
"schema_changed": 1426195822
|
|
}
|
|
}
|
|
}
|
|
|
|
if os.path.isfile("data/benchmark.db"): os.unlink("data/benchmark.db")
|
|
|
|
with benchmark("Open x 10", 0.13):
|
|
for i in range(10):
|
|
db = Db(schema, "data/benchmark.db")
|
|
db.checkTables()
|
|
db.close()
|
|
yield "."
|
|
|
|
|
|
db = Db(schema, "data/benchmark.db")
|
|
db.checkTables()
|
|
import json
|
|
|
|
with benchmark("Insert x 10 x 1000", 1.0):
|
|
for u in range(10): # 10 user
|
|
data = {"test": []}
|
|
for i in range(1000): # 1000 line of data
|
|
data["test"].append({"test_id": i, "title": "Testdata for %s message %s" % (u, i)})
|
|
json.dump(data, open("data/test_%s.json" % u, "w"))
|
|
db.loadJson("data/test_%s.json" % u)
|
|
os.unlink("data/test_%s.json" % u)
|
|
yield "."
|
|
|
|
|
|
with benchmark("Buffered insert x 100 x 100", 1.3):
|
|
cur = db.getCursor()
|
|
cur.execute("BEGIN")
|
|
cur.logging = False
|
|
for u in range(100, 200): # 100 user
|
|
data = {"test": []}
|
|
for i in range(100): # 1000 line of data
|
|
data["test"].append({"test_id": i, "title": "Testdata for %s message %s" % (u, i)})
|
|
json.dump(data, open("data/test_%s.json" % u, "w"))
|
|
db.loadJson("data/test_%s.json" % u, cur=cur)
|
|
os.unlink("data/test_%s.json" % u)
|
|
if u%10 == 0: yield "."
|
|
cur.execute("COMMIT")
|
|
|
|
yield " - Total rows in db: %s<br>" % db.execute("SELECT COUNT(*) AS num FROM test").fetchone()[0]
|
|
|
|
with benchmark("Indexed query x 1000", 0.25):
|
|
found = 0
|
|
cur = db.getCursor()
|
|
cur.logging = False
|
|
for i in range(1000): # 1000x by test_id
|
|
res = cur.execute("SELECT * FROM test WHERE test_id = %s" % i)
|
|
for row in res:
|
|
found += 1
|
|
if i%100 == 0: yield "."
|
|
|
|
assert found == 20000, "Found: %s != 20000" % found
|
|
|
|
|
|
with benchmark("Not indexed query x 100", 0.6):
|
|
found = 0
|
|
cur = db.getCursor()
|
|
cur.logging = False
|
|
for i in range(100): # 1000x by test_id
|
|
res = cur.execute("SELECT * FROM test WHERE json_id = %s" % i)
|
|
for row in res:
|
|
found += 1
|
|
if i%10 == 0: yield "."
|
|
|
|
assert found == 18900, "Found: %s != 18900" % found
|
|
|
|
|
|
with benchmark("Like query x 100", 1.8):
|
|
found = 0
|
|
cur = db.getCursor()
|
|
cur.logging = False
|
|
for i in range(100): # 1000x by test_id
|
|
res = cur.execute("SELECT * FROM test WHERE title LIKE '%%message %s%%'" % i)
|
|
for row in res:
|
|
found += 1
|
|
if i%10 == 0: yield "."
|
|
|
|
assert found == 38900, "Found: %s != 11000" % found
|
|
|
|
|
|
db.close()
|
|
if os.path.isfile("data/benchmark.db"): os.unlink("data/benchmark.db")
|
|
|
|
|
|
yield "<br>Done. Total: %.2fs" % (time.time()-t) |