[Zope-Checkins] CVS: ZODB3/ZEO - ClientStorage.py:1.73.2.4.2.1
Jeremy Hylton
jeremy@zope.com
Fri, 1 Nov 2002 14:17:25 -0500
Update of /cvs-repository/ZODB3/ZEO
In directory cvs.zope.org:/tmp/cvs-serv20135/ZEO
Modified Files:
Tag: ZODB3-deadlock-debug-branch
ClientStorage.py
Log Message:
Implement sortKey() for ClientStorage.
The sortKey is the address of the current storage server. Two
wrinkles of interest:
- A single client can connect to multiple servers, so its sortKey()
can change.
- If different clients use different names (e.g. one uses hostname and
one uses IP address), we could get different sortKeys. This config
should be avoided, but the code tries to get a canonical name just
in case.
=== ZODB3/ZEO/ClientStorage.py 1.73.2.4 => 1.73.2.4.2.1 ===
--- ZODB3/ZEO/ClientStorage.py:1.73.2.4 Wed Oct 16 17:44:05 2002
+++ ZODB3/ZEO/ClientStorage.py Fri Nov 1 14:17:24 2002
@@ -28,9 +28,11 @@
import cPickle
import os
+import socket
import tempfile
import threading
import time
+import types
from ZEO import ClientCache, ServerStub
from ZEO.TransactionBuffer import TransactionBuffer
@@ -204,6 +206,8 @@
self._storage = storage
self._read_only_fallback = read_only_fallback
self._connection = None
+ # _server_addr is used by sortKey()
+ self._server_addr = None
self._info = {'length': 0, 'size': 0, 'name': 'ZEO Client',
'supportsUndo':0, 'supportsVersions': 0,
@@ -339,6 +343,7 @@
log2(INFO, "Reconnected to storage")
else:
log2(INFO, "Connected to storage")
+ self.set_server_addr(conn.get_addr())
stub = self.StorageServerStubClass(conn)
self._oids = []
self._info.update(stub.get_info())
@@ -349,6 +354,33 @@
self._connection.close()
self._connection = conn
self._server = stub
+
+ def set_server_addr(self, addr):
+ # Normalize server address and convert to string
+ if isinstance(addr, types.StringType):
+ self._server_addr = addr
+ else:
+ assert isinstance(addr, types.TupleType)
+ # If the server is on a remote host, we need to guarantee
+ # that all clients used the same name for the server. If
+ # they don't, the sortKey() may be different for each client.
+ # The best solution seems to be the official name reported
+ # by gethostbyaddr().
+ host = addr[0]
+ try:
+ canonical, aliases, addrs = socket.gethostbyaddr(host)
+ except socket.error, err:
+ log2(INFO, "Error resoving host: %s (%s)" % (host, err))
+ canonical = host
+ self._server_addr = str((canonical, addr[1]))
+
+ def sortKey(self):
+ # If the client isn't connected to anything, it can't have a
+ # valid sortKey(). Raise an error to stop the transaction early.
+ if self._server_addr is None:
+ raise ClientDisconnected
+ else:
+ return self._server_addr
def verify_cache(self, server):
"""Internal routine called to verify the cache."""