From 44fc7ddd0ce6bb692ff26577c8f11143aedc5f55 Mon Sep 17 00:00:00 2001
From: Yann Leboulanger <asterix@lagaule.org>
Date: Wed, 14 May 2008 18:30:39 +0000
Subject: [PATCH] fix caps hash computaion according to XEP-0115. fixes #3925

---
 src/common/caps.py    |  2 +-
 src/common/helpers.py | 33 ++++++++++++++++++++++++++++++++-
 2 files changed, 33 insertions(+), 2 deletions(-)

diff --git a/src/common/caps.py b/src/common/caps.py
index 6c4a3444e7..b80a7b42f3 100644
--- a/src/common/caps.py
+++ b/src/common/caps.py
@@ -236,7 +236,7 @@ class ConnectionCaps(object):
 			return
 		node, hash = node.split('#', 1)
 		computed_hash = helpers.compute_caps_hash(identities, features,
-			contact.caps_hash_method)
+			dataforms=dataforms, hash_method=contact.caps_hash_method)
 		if computed_hash != hash:
 			# wrong hash, forget it
 			contact.caps_node = ''
diff --git a/src/common/helpers.py b/src/common/helpers.py
index 8e9666e309..91cb833277 100644
--- a/src/common/helpers.py
+++ b/src/common/helpers.py
@@ -1245,7 +1245,18 @@ def sort_identities_func(i1, i2):
 		return 1
 	return 0
 
-def compute_caps_hash(identities, features, hash_method='sha-1'):
+def sort_dataforms_func(d1, d2):
+	f1 = d1.getField('FORM_TYPE')
+	f2 = d2.getField('FORM_TYPE')
+	if f1 and f2 and (f1.getValue() < f2.getValue()):
+		return -1
+	return 1
+
+def compute_caps_hash(identities, features, dataforms=[], hash_method='sha-1'):
+	'''Compute caps hash according to XEP-0115, V1.5
+	
+	dataforms are xmpp.DataForms objects as common.dataforms don't allow several
+	values without a field type list-multi'''
 	S = ''
 	identities.sort(cmp=sort_identities_func)
 	for i in identities:
@@ -1266,6 +1277,26 @@ def compute_caps_hash(identities, features, hash_method='sha-1'):
 	features.sort()
 	for f in features:
 		S += '%s<' % f
+	dataforms.sort(cmp=sort_dataforms_func)
+	for dataform in dataforms:
+		# fields indexed by var
+		fields = {}
+		for f in dataform.getChildren():
+			fields[f.getVar()] = f
+		form_type = fields.get('FORM_TYPE')
+		if form_type:
+			S += form_type.getValue() + '<'
+			del fields['FORM_TYPE']
+		vars = fields.keys()
+		vars.sort()
+		for var in vars:
+			S += '%s<' % var
+			values = fields[var].getValues()
+			values.sort()
+			for value in values:
+				S += '%s<' % value
+
+	print S
 	if hash_method == 'sha-1':
 		hash = hash_sha1(S)
 	elif hash_method == 'md5':
-- 
GitLab