Commit bef7c0fe authored by Jeremy Hylton's avatar Jeremy Hylton

Add provisional monitor server that reports server statistics

Also, remove unused reuse_addr arg to ZEO.zrpc.server.  The server was
always calling set_reuse_addr().

No tests yet, that's the next step.  Simple functional tests work.
parent 873eb117
This diff is collapsed.
##############################################################################
#
# Copyright (c) 2003 Zope Corporation and Contributors.
# All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
# FOR A PARTICULAR PURPOSE
#
##############################################################################
"""Monitor behavior of ZEO server and record statistics.
$id:$
"""
import asyncore
import socket
import time
import types
import ZEO
class StorageStats:
"""Per-storage usage statistics."""
def __init__(self):
self.loads = 0
self.stores = 0
self.commits = 0
self.aborts = 0
self.active_txns = 0
self.clients = 0
self.verifying_clients = 0
self.lock_time = None
self.conflicts = 0
self.conflicts_resolved = 0
def dump(self, f):
print >> f, "Clients:", self.clients
print >> f, "Clients verifying:", self.verifying_clients
print >> f, "Active transactions:", self.active_txns
if self.lock_time:
howlong = time.time() - self.lock_time
print >> f, "Commit lock held for:", int(howlong)
print >> f, "Commits:", self.commits
print >> f, "Aborts:", self.aborts
print >> f, "Loads:", self.loads
print >> f, "Stores:", self.stores
print >> f, "Conflicts:", self.conflicts
print >> f, "Conflicts resolved:", self.conflicts_resolved
class StatsClient(asyncore.dispatcher):
def __init__(self, sock, addr):
asyncore.dispatcher.__init__(self, sock)
self.buf = []
self.closed = 0
def close(self):
self.closed = 1
# The socket is closed after all the data is written.
# See handle_write().
def write(self, s):
self.buf.append(s)
def writable(self):
return len(self.buf)
def readable(self):
# XXX what goes here?
return 0
def handle_write(self):
s = "".join(self.buf)
self.buf = []
n = self.socket.send(s)
if n < len(s):
self.buf.append(s[:n])
if self.closed and not self.buf:
asyncore.dispatcher.close(self)
class StatsServer(asyncore.dispatcher):
StatsConnectionClass = StatsClient
def __init__(self, addr, stats):
asyncore.dispatcher.__init__(self)
self.addr = addr
self.stats = stats
if type(self.addr) == types.TupleType:
self.create_socket(socket.AF_INET, socket.SOCK_STREAM)
else:
self.create_socket(socket.AF_UNIX, socket.SOCK_STREAM)
self.set_reuse_addr()
self.bind(self.addr)
self.listen(5)
def writable(self):
return 0
def readable(self):
return 1
def handle_accept(self):
try:
sock, addr = self.accept()
except socket.error:
return
f = self.StatsConnectionClass(sock, addr)
self.dump(f)
f.close()
def dump(self, f):
print >> f, "ZEO monitor server version %s" % ZEO.version
print >> f, time.ctime()
print >> f
L = self.stats.keys()
L.sort()
for k in L:
stats = self.stats[k]
print >> f, "Storage:", k
stats.dump(f)
print >> f
...@@ -22,6 +22,7 @@ Options: ...@@ -22,6 +22,7 @@ Options:
(a PATH must contain at least one "/") (a PATH must contain at least one "/")
-f/--filename FILENAME -- filename for FileStorage -f/--filename FILENAME -- filename for FileStorage
-h/--help -- print this usage message and exit -h/--help -- print this usage message and exit
-m/--monitor ADDRESS -- address of monitor server
Unless -C is specified, -a and -f are required. Unless -C is specified, -a and -f are required.
""" """
...@@ -147,43 +148,62 @@ class Options: ...@@ -147,43 +148,62 @@ class Options:
sys.stderr.write("For help, use %s -h\n" % self.progname) sys.stderr.write("For help, use %s -h\n" % self.progname)
sys.exit(2) sys.exit(2)
def parse_address(arg):
if "/" in arg:
family = socket.AF_UNIX
address = arg
else:
family = socket.AF_INET
if ":" in arg:
host, port = arg.split(":", 1)
else:
host = ""
port = arg
try:
port = int(port)
except: # int() can raise all sorts of errors
raise ValueError("invalid port number: %r" % port)
address = host, port
return family, address
class ZEOOptions(Options): class ZEOOptions(Options):
read_only = None read_only = None
transaction_timeout = None transaction_timeout = None
invalidation_queue_size = None invalidation_queue_size = None
monitor_address = None
family = None # set by -a; AF_UNIX or AF_INET family = None # set by -a; AF_UNIX or AF_INET
address = None # set by -a; string or (host, port) address = None # set by -a; string or (host, port)
storages = None # set by -f storages = None # set by -f
_short_options = "a:C:f:h" _short_options = "a:C:f:hm:"
_long_options = [ _long_options = [
"address=", "address=",
"configuration=", "configuration=",
"filename=", "filename=",
"help", "help",
"monitor=",
] ]
def handle_option(self, opt, arg): def handle_option(self, opt, arg):
# Alphabetical order please! # Alphabetical order please!
if opt in ("-a", "--address"): if opt in ("-a", "--address"):
if "/" in arg: try:
self.family = socket.AF_UNIX f, a = parse_address(arg)
self.address = arg except ValueError, err:
else: self.usage(str(err))
self.family = socket.AF_INET
if ":" in arg:
host, port = arg.split(":", 1)
else: else:
host = "" self.family = f
port = arg self.address = a
elif opt in ("-m", "--monitor"):
try: try:
port = int(port) f, a = parse_address(arg)
except: # int() can raise all sorts of errors except ValueError, err:
self.usage("invalid port number: %r" % port) self.usage(str(err))
self.address = (host, port) else:
self.monitor_family = f
self.monitor_address = a
elif opt in ("-f", "--filename"): elif opt in ("-f", "--filename"):
from ZODB.config import FileStorage from ZODB.config import FileStorage
class FSConfig: class FSConfig:
...@@ -238,7 +258,7 @@ class ZEOOptions(Options): ...@@ -238,7 +258,7 @@ class ZEOOptions(Options):
self.read_only = self.rootconf.read_only self.read_only = self.rootconf.read_only
self.transaction_timeout = self.rootconf.transaction_timeout self.transaction_timeout = self.rootconf.transaction_timeout
self.invalidation_queue_size = self.rootconf.invalidation_queue_size self.invalidation_queue_size = 100
def load_logconf(self): def load_logconf(self):
# Get logging options from conf, unless overridden by environment # Get logging options from conf, unless overridden by environment
...@@ -349,7 +369,8 @@ class ZEOServer: ...@@ -349,7 +369,8 @@ class ZEOServer:
self.storages, self.storages,
read_only=self.options.read_only, read_only=self.options.read_only,
invalidation_queue_size=self.options.invalidation_queue_size, invalidation_queue_size=self.options.invalidation_queue_size,
transaction_timeout=self.options.transaction_timeout) transaction_timeout=self.options.transaction_timeout,
monitor_address=self.options.monitor_address)
def loop_forever(self): def loop_forever(self):
import ThreadedAsync.LoopCallback import ThreadedAsync.LoopCallback
......
...@@ -58,6 +58,14 @@ ...@@ -58,6 +58,14 @@
</description> </description>
</key> </key>
<key name="monitor-address" datatype="socket-address" required="no">
<description>
The address at which the monitor server should listen. If
specified, a monitor server is started. The monitor server
provides server statistics in a simple text format.
</description>
</key>
<multisection name="+" type="storage" <multisection name="+" type="storage"
attribute="storages" attribute="storages"
required="yes"> required="yes">
......
...@@ -27,15 +27,11 @@ class Dispatcher(asyncore.dispatcher): ...@@ -27,15 +27,11 @@ class Dispatcher(asyncore.dispatcher):
"""A server that accepts incoming RPC connections""" """A server that accepts incoming RPC connections"""
__super_init = asyncore.dispatcher.__init__ __super_init = asyncore.dispatcher.__init__
reuse_addr = 1 def __init__(self, addr, factory=Connection):
def __init__(self, addr, factory=Connection, reuse_addr=None):
self.__super_init() self.__super_init()
self.addr = addr self.addr = addr
self.factory = factory self.factory = factory
self.clients = [] self.clients = []
if reuse_addr is not None:
self.reuse_addr = reuse_addr
self._open_socket() self._open_socket()
def _open_socket(self): def _open_socket(self):
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment