2008-06-18 12:24:16 -07:00
|
|
|
|
|
|
|
import time, os.path
|
2009-06-22 19:10:47 -07:00
|
|
|
from base64 import b32decode
|
2008-06-18 12:24:16 -07:00
|
|
|
from zope.interface import implements
|
|
|
|
from twisted.application import service
|
2009-06-22 19:10:47 -07:00
|
|
|
from foolscap.api import Referenceable, SturdyRef
|
2008-11-21 17:43:52 -07:00
|
|
|
import allmydata
|
2008-06-18 12:24:16 -07:00
|
|
|
from allmydata import node
|
2009-06-22 19:10:47 -07:00
|
|
|
from allmydata.util import log, rrefutil
|
2008-06-18 17:04:41 -07:00
|
|
|
from allmydata.introducer.interfaces import \
|
|
|
|
RIIntroducerPublisherAndSubscriberService
|
2008-06-18 12:24:16 -07:00
|
|
|
|
|
|
|
class IntroducerNode(node.Node):
|
|
|
|
PORTNUMFILE = "introducer.port"
|
|
|
|
NODETYPE = "introducer"
|
2011-08-02 18:32:12 -07:00
|
|
|
GENERATED_FILES = ['introducer.furl']
|
2008-06-18 12:24:16 -07:00
|
|
|
|
|
|
|
def __init__(self, basedir="."):
|
|
|
|
node.Node.__init__(self, basedir)
|
2008-09-30 16:21:49 -07:00
|
|
|
self.read_config()
|
2008-06-18 12:24:16 -07:00
|
|
|
self.init_introducer()
|
2008-09-30 16:21:49 -07:00
|
|
|
webport = self.get_config("node", "web.port", None)
|
2008-06-18 12:24:16 -07:00
|
|
|
if webport:
|
|
|
|
self.init_web(webport) # strports string
|
|
|
|
|
|
|
|
def init_introducer(self):
|
|
|
|
introducerservice = IntroducerService(self.basedir)
|
|
|
|
self.add_service(introducerservice)
|
|
|
|
|
|
|
|
d = self.when_tub_ready()
|
|
|
|
def _publish(res):
|
|
|
|
self.introducer_url = self.tub.registerReference(introducerservice,
|
|
|
|
"introducer")
|
|
|
|
self.log(" introducer is at %s" % self.introducer_url)
|
|
|
|
self.write_config("introducer.furl", self.introducer_url + "\n")
|
|
|
|
d.addCallback(_publish)
|
2008-08-25 18:57:59 -07:00
|
|
|
d.addErrback(log.err, facility="tahoe.init",
|
|
|
|
level=log.BAD, umid="UaNs9A")
|
2008-06-18 12:24:16 -07:00
|
|
|
|
|
|
|
def init_web(self, webport):
|
|
|
|
self.log("init_web(webport=%s)", args=(webport,))
|
|
|
|
|
|
|
|
from allmydata.webish import IntroducerWebishServer
|
|
|
|
nodeurl_path = os.path.join(self.basedir, "node.url")
|
2009-02-20 12:15:54 -07:00
|
|
|
ws = IntroducerWebishServer(self, webport, nodeurl_path)
|
2008-06-18 12:24:16 -07:00
|
|
|
self.add_service(ws)
|
|
|
|
|
|
|
|
class IntroducerService(service.MultiService, Referenceable):
|
|
|
|
implements(RIIntroducerPublisherAndSubscriberService)
|
|
|
|
name = "introducer"
|
2008-11-21 17:43:52 -07:00
|
|
|
VERSION = { "http://allmydata.org/tahoe/protocols/introducer/v1":
|
|
|
|
{ },
|
versioning: include an "appname" in the application version string in the versioning protocol, and make that appname be controlled by setup.py
It is currently hardcoded in setup.py to be 'allmydata-tahoe'. Ticket #556 is to make it configurable by a runtime command-line argument to setup.py: "--appname=foo", but I suddenly wondered if we really wanted that and at the same time realized that we don't need that for tahoe-1.3.0 release, so this patch just hardcodes it in setup.py.
setup.py inspects a file named 'src/allmydata/_appname.py' and assert that it contains the string "__appname__ = 'allmydata-tahoe'", and creates it if it isn't already present. src/allmydata/__init__.py import _appname and reads __appname__ from it. The rest of the Python code imports allmydata and inspects "allmydata.__appname__", although actually every use it uses "allmydata.__full_version__" instead, where "allmydata.__full_version__" is created in src/allmydata/__init__.py to be:
__full_version__ = __appname + '-' + str(__version__).
All the code that emits an "application version string" when describing what version of a protocol it supports (introducer server, storage server, upload helper), or when describing itself in general (introducer client), usese allmydata.__full_version__.
This fixes ticket #556 at least well enough for tahoe-1.3.0 release.
2009-02-11 17:18:16 -07:00
|
|
|
"application-version": str(allmydata.__full_version__),
|
2008-11-21 17:43:52 -07:00
|
|
|
}
|
2008-06-18 12:24:16 -07:00
|
|
|
|
|
|
|
def __init__(self, basedir="."):
|
|
|
|
service.MultiService.__init__(self)
|
|
|
|
self.introducer_url = None
|
2009-06-22 19:10:47 -07:00
|
|
|
# 'index' is (service_name, tubid)
|
2008-06-18 12:24:16 -07:00
|
|
|
self._announcements = {} # dict of index -> (announcement, timestamp)
|
|
|
|
self._subscribers = {} # dict of (rref->timestamp) dicts
|
2009-06-22 19:10:47 -07:00
|
|
|
self._debug_counts = {"inbound_message": 0,
|
|
|
|
"inbound_duplicate": 0,
|
|
|
|
"inbound_update": 0,
|
|
|
|
"outbound_message": 0,
|
|
|
|
"outbound_announcements": 0,
|
|
|
|
"inbound_subscribe": 0}
|
2008-06-18 12:24:16 -07:00
|
|
|
|
|
|
|
def log(self, *args, **kwargs):
|
|
|
|
if "facility" not in kwargs:
|
|
|
|
kwargs["facility"] = "tahoe.introducer"
|
|
|
|
return log.msg(*args, **kwargs)
|
|
|
|
|
|
|
|
def get_announcements(self):
|
|
|
|
return self._announcements
|
|
|
|
def get_subscribers(self):
|
|
|
|
return self._subscribers
|
|
|
|
|
2008-11-21 17:43:52 -07:00
|
|
|
def remote_get_version(self):
|
|
|
|
return self.VERSION
|
|
|
|
|
2008-06-18 12:24:16 -07:00
|
|
|
def remote_publish(self, announcement):
|
2009-06-22 19:10:47 -07:00
|
|
|
try:
|
|
|
|
self._publish(announcement)
|
|
|
|
except:
|
|
|
|
log.err(format="Introducer.remote_publish failed on %(ann)s",
|
|
|
|
ann=announcement, level=log.UNUSUAL, umid="620rWA")
|
|
|
|
raise
|
|
|
|
|
|
|
|
def _publish(self, announcement):
|
|
|
|
self._debug_counts["inbound_message"] += 1
|
2008-06-18 12:24:16 -07:00
|
|
|
self.log("introducer: announcement published: %s" % (announcement,) )
|
2009-06-22 19:10:47 -07:00
|
|
|
(furl, service_name, ri_name, nickname_utf8, ver, oldest) = announcement
|
|
|
|
|
|
|
|
nodeid = b32decode(SturdyRef(furl).tubID.upper())
|
|
|
|
index = (service_name, nodeid)
|
|
|
|
|
2008-06-18 12:24:16 -07:00
|
|
|
if index in self._announcements:
|
|
|
|
(old_announcement, timestamp) = self._announcements[index]
|
|
|
|
if old_announcement == announcement:
|
|
|
|
self.log("but we already knew it, ignoring", level=log.NOISY)
|
2009-06-22 19:10:47 -07:00
|
|
|
self._debug_counts["inbound_duplicate"] += 1
|
2008-06-18 12:24:16 -07:00
|
|
|
return
|
|
|
|
else:
|
|
|
|
self.log("old announcement being updated", level=log.NOISY)
|
2009-06-22 19:10:47 -07:00
|
|
|
self._debug_counts["inbound_update"] += 1
|
2008-06-18 12:24:16 -07:00
|
|
|
self._announcements[index] = (announcement, time.time())
|
2009-06-22 19:10:47 -07:00
|
|
|
|
2008-06-18 12:24:16 -07:00
|
|
|
for s in self._subscribers.get(service_name, []):
|
2009-06-22 19:10:47 -07:00
|
|
|
self._debug_counts["outbound_message"] += 1
|
|
|
|
self._debug_counts["outbound_announcements"] += 1
|
|
|
|
d = s.callRemote("announce", set([announcement]))
|
|
|
|
d.addErrback(rrefutil.trap_deadref)
|
|
|
|
d.addErrback(log.err,
|
|
|
|
format="subscriber errored on announcement %(ann)s",
|
|
|
|
ann=announcement, facility="tahoe.introducer",
|
|
|
|
level=log.UNUSUAL, umid="jfGMXQ")
|
2008-06-18 12:24:16 -07:00
|
|
|
|
|
|
|
def remote_subscribe(self, subscriber, service_name):
|
|
|
|
self.log("introducer: subscription[%s] request at %s" % (service_name,
|
|
|
|
subscriber))
|
2009-06-22 19:10:47 -07:00
|
|
|
self._debug_counts["inbound_subscribe"] += 1
|
2008-06-18 12:24:16 -07:00
|
|
|
if service_name not in self._subscribers:
|
|
|
|
self._subscribers[service_name] = {}
|
|
|
|
subscribers = self._subscribers[service_name]
|
|
|
|
if subscriber in subscribers:
|
|
|
|
self.log("but they're already subscribed, ignoring",
|
|
|
|
level=log.UNUSUAL)
|
|
|
|
return
|
|
|
|
subscribers[subscriber] = time.time()
|
|
|
|
def _remove():
|
|
|
|
self.log("introducer: unsubscribing[%s] %s" % (service_name,
|
|
|
|
subscriber))
|
|
|
|
subscribers.pop(subscriber, None)
|
|
|
|
subscriber.notifyOnDisconnect(_remove)
|
|
|
|
|
2009-06-22 19:10:47 -07:00
|
|
|
announcements = set(
|
|
|
|
[ ann
|
|
|
|
for (sn2,nodeid),(ann,when) in self._announcements.items()
|
|
|
|
if sn2 == service_name] )
|
2008-06-18 12:24:16 -07:00
|
|
|
|
2009-06-22 19:10:47 -07:00
|
|
|
self._debug_counts["outbound_message"] += 1
|
|
|
|
self._debug_counts["outbound_announcements"] += len(announcements)
|
|
|
|
d = subscriber.callRemote("announce", announcements)
|
|
|
|
d.addErrback(rrefutil.trap_deadref)
|
|
|
|
d.addErrback(log.err,
|
|
|
|
format="subscriber errored during subscribe %(anns)s",
|
|
|
|
anns=announcements, facility="tahoe.introducer",
|
|
|
|
level=log.UNUSUAL, umid="mtZepQ")
|