proper error handling

This commit is contained in:
Brendan Taylor 2007-07-10 05:41:43 +00:00
parent ccac0b4299
commit d32e8352d5
1 changed files with 37 additions and 21 deletions

View File

@ -263,12 +263,11 @@ class EncryptedStanzaSession(StanzaSession):
return self.encrypter.encrypt(encryptable) return self.encrypter.encrypt(encryptable)
# FIXME: get a real PRNG # FIXME: use a real PRNG
def random_bytes(self, bytes): def random_bytes(self, bytes):
return os.urandom(bytes) return os.urandom(bytes)
def generate_nonce(self): def generate_nonce(self):
# FIXME: this isn't a very good PRNG
return self.random_bytes(8) return self.random_bytes(8)
def decrypt_stanza(self, stanza): def decrypt_stanza(self, stanza):
@ -581,19 +580,19 @@ class EncryptedStanzaSession(StanzaSession):
x = xmpp.DataForm(typ='result') x = xmpp.DataForm(typ='result')
for field in ('nonce', 'dhkeys', 'rshashes', 'identity', 'mac'): for field in ('nonce', 'dhkeys', 'rshashes', 'identity', 'mac'):
assert field in form.asDict(), "your acceptance form didn't have a %s field" % repr(field) assert field in form.asDict(), "alice's form didn't have a %s field" % field
# 4.5.1 generating provisory session keys # 4.5.1 generating provisory session keys
e = self.decode_mpi(base64.b64decode(form['dhkeys'])) e = self.decode_mpi(base64.b64decode(form['dhkeys']))
p = dh.primes[self.modp] p = dh.primes[self.modp]
if not self.sha256(self.encode_mpi(e)) == self.He: if (not self.sha256(self.encode_mpi(e)) == self.He): or \
# XXX return <feature-not-implemented/> (not e > 1 and e < (p - 1)):
pass err = xmpp.Error(response, xmpp.ERR_FEATURE_NOT_IMPLEMENTED)
err.T.error.T.text.setData("invalid DH value 'e'")
if not e > 1 and e < (p - 1): self.send(err)
# XXX return <feature-not-implemented/> self.status = None
pass return
k = self.sha256(self.encode_mpi(self.powmod(e, self.y, p))) k = self.sha256(self.encode_mpi(self.powmod(e, self.y, p)))
@ -606,8 +605,11 @@ class EncryptedStanzaSession(StanzaSession):
m_a_calculated = self.hmac(self.km_o, self.encode_mpi(self.c_o) + id_a) m_a_calculated = self.hmac(self.km_o, self.encode_mpi(self.c_o) + id_a)
if m_a_calculated != m_a: if m_a_calculated != m_a:
# XXX return <feature-not-implemented/> err = xmpp.Error(response, xmpp.ERR_FEATURE_NOT_IMPLEMENTED)
pass err.T.error.T.text.setData('calculated m_a differs from received m_a')
self.send(err)
self.status = None
return
mac_a = self.decrypt(id_a) mac_a = self.decrypt(id_a)
@ -617,10 +619,11 @@ class EncryptedStanzaSession(StanzaSession):
mac_a_calculated = self.hmac(self.ks_o, self.n_s + self.n_o + self.encode_mpi(e) + self.form_a + form_a2) mac_a_calculated = self.hmac(self.ks_o, self.n_s + self.n_o + self.encode_mpi(e) + self.form_a + form_a2)
if mac_a_calculated != mac_a: if mac_a_calculated != mac_a:
# XXX return <feature-not-implemented/> err = xmpp.Error(response, xmpp.ERR_FEATURE_NOT_IMPLEMENTED)
pass err.T.error.T.text.setData('calculated mac_a differs from received mac_a')
self.send(err)
# TODO: 4.5.3 self.status = None
return
# 4.5.4 generating bob's final session keys # 4.5.4 generating bob's final session keys
self.srs = '' self.srs = ''
@ -697,18 +700,32 @@ class EncryptedStanzaSession(StanzaSession):
#4.6.2 Verifying Bob's Identity #4.6.2 Verifying Bob's Identity
m_b = base64.b64decode(form['mac'])
id_b = base64.b64decode(form['identity']) id_b = base64.b64decode(form['identity'])
m_b = self.hmac(self.encode_mpi(self.c_o) + id_b, self.km_o) m_b_calculated = self.hmac(self.encode_mpi(self.c_o) + id_b, self.km_o)
if m_b_calculated != m_b:
err = xmpp.Error(response, xmpp.ERR_FEATURE_NOT_IMPLEMENTED)
err.T.error.T.text.setData('calculated m_b differs from received m_b')
self.send(err)
self.status = None
return
mac_b = self.decrypt(id_b) mac_b = self.decrypt(id_b)
form_b2 = ''.join(map(lambda el: xmpp.c14n.c14n(el), form.getChildren())) form_b2 = ''.join(map(lambda el: xmpp.c14n.c14n(el), form.getChildren()))
mac_b = self.hmac(self.n_s + self.n_o + self.encode_mpi(self.d) + self.form_b + form_b2, self.ks_o) mac_b_calculated = self.hmac(self.n_s + self.n_o + self.encode_mpi(self.d) + self.form_b + form_b2, self.ks_o)
if mac_b_calculated != mac_b:
err = xmpp.Error(response, xmpp.ERR_FEATURE_NOT_IMPLEMENTED)
err.T.error.T.text.setData('calculated mac_b differs from received mac_b')
self.send(err)
self.status = None
return
# Note: If Alice discovers an error then she SHOULD ignore any encrypted content she received in the stanza. # Note: If Alice discovers an error then she SHOULD ignore any encrypted content she received in the stanza.
# XXX check for MAC equality?
if self.negotiated['logging'] == 'mustnot': if self.negotiated['logging'] == 'mustnot':
self.loggable = False self.loggable = False
@ -721,9 +738,8 @@ class EncryptedStanzaSession(StanzaSession):
# minimum number of bytes needed to represent that range # minimum number of bytes needed to represent that range
bytes = int(math.ceil(math.log(top - bottom, 256))) bytes = int(math.ceil(math.log(top - bottom, 256)))
# FIXME: use a real PRNG
# in retrospect, this is horribly inadequate. # in retrospect, this is horribly inadequate.
return (self.decode_mpi(os.urandom(bytes)) % (top - bottom)) + bottom return (self.decode_mpi(self.random_bytes(bytes)) % (top - bottom)) + bottom
def make_dhhash(self, modp): def make_dhhash(self, modp):
p = dh.primes[modp] p = dh.primes[modp]