2008-04-01 18:45:13 -07:00
|
|
|
|
|
|
|
import os
|
|
|
|
import time
|
|
|
|
|
2009-05-21 17:38:23 -07:00
|
|
|
from foolscap.api import Referenceable, Tub
|
2008-04-01 18:45:13 -07:00
|
|
|
from zope.interface import implements
|
|
|
|
from twisted.internet import reactor
|
|
|
|
from twisted.application import service
|
2008-09-20 10:35:45 -07:00
|
|
|
from allmydata.util import log
|
2008-04-01 18:45:13 -07:00
|
|
|
|
|
|
|
from pycryptopp.publickey import rsa
|
|
|
|
from allmydata.interfaces import RIKeyGenerator
|
|
|
|
|
2009-05-21 17:38:23 -07:00
|
|
|
class KeyGenerator(service.MultiService, Referenceable):
|
2008-04-01 18:45:13 -07:00
|
|
|
implements(RIKeyGenerator)
|
|
|
|
|
|
|
|
pool_size = 16 # no. keys to keep on hand in the pool
|
|
|
|
pool_refresh_delay = 6 # no. sec to wait after a fetch before generating new keys
|
|
|
|
verbose = False
|
|
|
|
|
2008-04-22 12:28:18 -07:00
|
|
|
def __init__(self, default_key_size=2048):
|
2008-04-03 15:57:07 -07:00
|
|
|
service.MultiService.__init__(self)
|
2008-04-01 18:45:13 -07:00
|
|
|
self.keypool = []
|
|
|
|
self.last_fetch = 0
|
2008-04-22 12:28:18 -07:00
|
|
|
self.default_key_size = default_key_size
|
2008-04-03 15:57:07 -07:00
|
|
|
|
|
|
|
def startService(self):
|
|
|
|
self.timer = reactor.callLater(0, self.maybe_refill_pool)
|
|
|
|
return service.MultiService.startService(self)
|
|
|
|
|
|
|
|
def stopService(self):
|
|
|
|
if self.timer.active():
|
|
|
|
self.timer.cancel()
|
|
|
|
return service.MultiService.stopService(self)
|
2008-04-01 18:45:13 -07:00
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
return '<KeyGenerator[%s]>' % (len(self.keypool),)
|
|
|
|
|
|
|
|
def vlog(self, msg):
|
|
|
|
if self.verbose:
|
|
|
|
log.msg(msg)
|
|
|
|
|
|
|
|
def reset_timer(self):
|
|
|
|
self.last_fetch = time.time()
|
2008-04-03 15:57:07 -07:00
|
|
|
if self.timer.active():
|
|
|
|
self.timer.reset(self.pool_refresh_delay)
|
|
|
|
else:
|
|
|
|
self.timer = reactor.callLater(self.pool_refresh_delay, self.maybe_refill_pool)
|
2008-04-01 18:45:13 -07:00
|
|
|
|
|
|
|
def maybe_refill_pool(self):
|
|
|
|
now = time.time()
|
|
|
|
if self.last_fetch + self.pool_refresh_delay < now:
|
|
|
|
self.vlog('%s refilling pool' % (self,))
|
|
|
|
while len(self.keypool) < self.pool_size:
|
2008-04-22 12:28:18 -07:00
|
|
|
self.keypool.append(self.gen_key(self.default_key_size))
|
2008-04-01 18:45:13 -07:00
|
|
|
else:
|
|
|
|
self.vlog('%s not refilling pool' % (self,))
|
2008-04-10 14:27:35 -07:00
|
|
|
reactor.callLater(1, self.maybe_refill_pool)
|
2008-04-01 18:45:13 -07:00
|
|
|
|
|
|
|
def gen_key(self, key_size):
|
|
|
|
self.vlog('%s generating key size %s' % (self, key_size, ))
|
|
|
|
signer = rsa.generate(key_size)
|
|
|
|
verifier = signer.get_verifying_key()
|
|
|
|
return verifier.serialize(), signer.serialize()
|
|
|
|
|
|
|
|
def remote_get_rsa_key_pair(self, key_size):
|
|
|
|
self.vlog('%s remote_get_key' % (self,))
|
2008-04-22 12:28:18 -07:00
|
|
|
if key_size != self.default_key_size or not self.keypool:
|
2008-04-03 18:43:46 -07:00
|
|
|
key = self.gen_key(key_size)
|
2008-04-03 13:01:43 -07:00
|
|
|
self.reset_timer()
|
2008-04-03 18:43:46 -07:00
|
|
|
return key
|
2008-04-01 18:45:13 -07:00
|
|
|
else:
|
|
|
|
self.reset_timer()
|
|
|
|
return self.keypool.pop()
|
|
|
|
|
|
|
|
class KeyGeneratorService(service.MultiService):
|
|
|
|
furl_file = 'key_generator.furl'
|
|
|
|
|
2008-04-22 12:28:18 -07:00
|
|
|
def __init__(self, basedir='.', display_furl=True, default_key_size=2048):
|
2008-04-01 18:45:13 -07:00
|
|
|
service.MultiService.__init__(self)
|
2008-04-03 15:57:07 -07:00
|
|
|
self.basedir = basedir
|
2009-05-21 17:38:23 -07:00
|
|
|
self.tub = Tub(certFile=os.path.join(self.basedir, 'key_generator.pem'))
|
2009-05-21 17:46:32 -07:00
|
|
|
self.tub.setOption("expose-remote-exception-types", False)
|
2008-04-01 18:45:13 -07:00
|
|
|
self.tub.setServiceParent(self)
|
2008-04-22 12:28:18 -07:00
|
|
|
self.key_generator = KeyGenerator(default_key_size=default_key_size)
|
2008-04-03 15:57:07 -07:00
|
|
|
self.key_generator.setServiceParent(self)
|
2008-04-01 18:45:13 -07:00
|
|
|
|
|
|
|
portnum = self.get_portnum()
|
|
|
|
self.listener = self.tub.listenOn(portnum or 'tcp:0')
|
|
|
|
d = self.tub.setLocationAutomatically()
|
|
|
|
if portnum is None:
|
|
|
|
d.addCallback(self.save_portnum)
|
2008-04-03 13:01:43 -07:00
|
|
|
d.addCallback(self.tub_ready, display_furl)
|
2008-04-01 18:45:13 -07:00
|
|
|
d.addErrback(log.err)
|
|
|
|
|
|
|
|
def get_portnum(self):
|
2008-04-03 15:57:07 -07:00
|
|
|
portnumfile = os.path.join(self.basedir, 'portnum')
|
|
|
|
if os.path.exists(portnumfile):
|
|
|
|
return file(portnumfile, 'rb').read().strip()
|
2008-04-01 18:45:13 -07:00
|
|
|
|
|
|
|
def save_portnum(self, junk):
|
|
|
|
portnum = self.listener.getPortnum()
|
2008-04-03 15:57:07 -07:00
|
|
|
portnumfile = os.path.join(self.basedir, 'portnum')
|
|
|
|
file(portnumfile, 'wb').write('%d\n' % (portnum,))
|
2008-04-01 18:45:13 -07:00
|
|
|
|
2008-04-03 13:01:43 -07:00
|
|
|
def tub_ready(self, junk, display_furl):
|
2008-04-03 15:57:07 -07:00
|
|
|
kgf = os.path.join(self.basedir, self.furl_file)
|
|
|
|
self.keygen_furl = self.tub.registerReference(self.key_generator, furlFile=kgf)
|
2008-04-03 13:01:43 -07:00
|
|
|
if display_furl:
|
2008-04-03 15:57:07 -07:00
|
|
|
print 'key generator at:', self.keygen_furl
|