keys.py 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206
  1. # -*- coding: utf-8 -*-
  2. # Copyright 2014-2016 OpenMarket Ltd
  3. #
  4. # Licensed under the Apache License, Version 2.0 (the "License");
  5. # you may not use this file except in compliance with the License.
  6. # You may obtain a copy of the License at
  7. #
  8. # http://www.apache.org/licenses/LICENSE-2.0
  9. #
  10. # Unless required by applicable law or agreed to in writing, software
  11. # distributed under the License is distributed on an "AS IS" BASIS,
  12. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. # See the License for the specific language governing permissions and
  14. # limitations under the License.
  15. from ._base import SQLBaseStore
  16. from synapse.util.caches.descriptors import cachedInlineCallbacks
  17. from twisted.internet import defer
  18. import OpenSSL
  19. from signedjson.key import decode_verify_key_bytes
  20. import hashlib
  21. class KeyStore(SQLBaseStore):
  22. """Persistence for signature verification keys and tls X.509 certificates
  23. """
  24. @defer.inlineCallbacks
  25. def get_server_certificate(self, server_name):
  26. """Retrieve the TLS X.509 certificate for the given server
  27. Args:
  28. server_name (bytes): The name of the server.
  29. Returns:
  30. (OpenSSL.crypto.X509): The tls certificate.
  31. """
  32. tls_certificate_bytes, = yield self._simple_select_one(
  33. table="server_tls_certificates",
  34. keyvalues={"server_name": server_name},
  35. retcols=("tls_certificate",),
  36. desc="get_server_certificate",
  37. )
  38. tls_certificate = OpenSSL.crypto.load_certificate(
  39. OpenSSL.crypto.FILETYPE_ASN1, tls_certificate_bytes,
  40. )
  41. defer.returnValue(tls_certificate)
  42. def store_server_certificate(self, server_name, from_server, time_now_ms,
  43. tls_certificate):
  44. """Stores the TLS X.509 certificate for the given server
  45. Args:
  46. server_name (str): The name of the server.
  47. from_server (str): Where the certificate was looked up
  48. time_now_ms (int): The time now in milliseconds
  49. tls_certificate (OpenSSL.crypto.X509): The X.509 certificate.
  50. """
  51. tls_certificate_bytes = OpenSSL.crypto.dump_certificate(
  52. OpenSSL.crypto.FILETYPE_ASN1, tls_certificate
  53. )
  54. fingerprint = hashlib.sha256(tls_certificate_bytes).hexdigest()
  55. return self._simple_upsert(
  56. table="server_tls_certificates",
  57. keyvalues={
  58. "server_name": server_name,
  59. "fingerprint": fingerprint,
  60. },
  61. values={
  62. "from_server": from_server,
  63. "ts_added_ms": time_now_ms,
  64. "tls_certificate": buffer(tls_certificate_bytes),
  65. },
  66. desc="store_server_certificate",
  67. )
  68. @cachedInlineCallbacks()
  69. def get_all_server_verify_keys(self, server_name):
  70. rows = yield self._simple_select_list(
  71. table="server_signature_keys",
  72. keyvalues={
  73. "server_name": server_name,
  74. },
  75. retcols=["key_id", "verify_key"],
  76. desc="get_all_server_verify_keys",
  77. )
  78. defer.returnValue({
  79. row["key_id"]: decode_verify_key_bytes(
  80. row["key_id"], str(row["verify_key"])
  81. )
  82. for row in rows
  83. })
  84. @defer.inlineCallbacks
  85. def get_server_verify_keys(self, server_name, key_ids):
  86. """Retrieve the NACL verification key for a given server for the given
  87. key_ids
  88. Args:
  89. server_name (str): The name of the server.
  90. key_ids (list of str): List of key_ids to try and look up.
  91. Returns:
  92. (list of VerifyKey): The verification keys.
  93. """
  94. keys = yield self.get_all_server_verify_keys(server_name)
  95. defer.returnValue({
  96. k: keys[k]
  97. for k in key_ids
  98. if k in keys and keys[k]
  99. })
  100. @defer.inlineCallbacks
  101. def store_server_verify_key(self, server_name, from_server, time_now_ms,
  102. verify_key):
  103. """Stores a NACL verification key for the given server.
  104. Args:
  105. server_name (str): The name of the server.
  106. key_id (str): The version of the key for the server.
  107. from_server (str): Where the verification key was looked up
  108. ts_now_ms (int): The time now in milliseconds
  109. verification_key (VerifyKey): The NACL verify key.
  110. """
  111. yield self._simple_upsert(
  112. table="server_signature_keys",
  113. keyvalues={
  114. "server_name": server_name,
  115. "key_id": "%s:%s" % (verify_key.alg, verify_key.version),
  116. },
  117. values={
  118. "from_server": from_server,
  119. "ts_added_ms": time_now_ms,
  120. "verify_key": buffer(verify_key.encode()),
  121. },
  122. desc="store_server_verify_key",
  123. )
  124. self.get_all_server_verify_keys.invalidate((server_name,))
  125. def store_server_keys_json(self, server_name, key_id, from_server,
  126. ts_now_ms, ts_expires_ms, key_json_bytes):
  127. """Stores the JSON bytes for a set of keys from a server
  128. The JSON should be signed by the originating server, the intermediate
  129. server, and by this server. Updates the value for the
  130. (server_name, key_id, from_server) triplet if one already existed.
  131. Args:
  132. server_name (str): The name of the server.
  133. key_id (str): The identifer of the key this JSON is for.
  134. from_server (str): The server this JSON was fetched from.
  135. ts_now_ms (int): The time now in milliseconds.
  136. ts_valid_until_ms (int): The time when this json stops being valid.
  137. key_json (bytes): The encoded JSON.
  138. """
  139. return self._simple_upsert(
  140. table="server_keys_json",
  141. keyvalues={
  142. "server_name": server_name,
  143. "key_id": key_id,
  144. "from_server": from_server,
  145. },
  146. values={
  147. "server_name": server_name,
  148. "key_id": key_id,
  149. "from_server": from_server,
  150. "ts_added_ms": ts_now_ms,
  151. "ts_valid_until_ms": ts_expires_ms,
  152. "key_json": buffer(key_json_bytes),
  153. },
  154. desc="store_server_keys_json",
  155. )
  156. def get_server_keys_json(self, server_keys):
  157. """Retrive the key json for a list of server_keys and key ids.
  158. If no keys are found for a given server, key_id and source then
  159. that server, key_id, and source triplet entry will be an empty list.
  160. The JSON is returned as a byte array so that it can be efficiently
  161. used in an HTTP response.
  162. Args:
  163. server_keys (list): List of (server_name, key_id, source) triplets.
  164. Returns:
  165. Dict mapping (server_name, key_id, source) triplets to dicts with
  166. "ts_valid_until_ms" and "key_json" keys.
  167. """
  168. def _get_server_keys_json_txn(txn):
  169. results = {}
  170. for server_name, key_id, from_server in server_keys:
  171. keyvalues = {"server_name": server_name}
  172. if key_id is not None:
  173. keyvalues["key_id"] = key_id
  174. if from_server is not None:
  175. keyvalues["from_server"] = from_server
  176. rows = self._simple_select_list_txn(
  177. txn,
  178. "server_keys_json",
  179. keyvalues=keyvalues,
  180. retcols=(
  181. "key_id",
  182. "from_server",
  183. "ts_added_ms",
  184. "ts_valid_until_ms",
  185. "key_json",
  186. ),
  187. )
  188. results[(server_name, key_id, from_server)] = rows
  189. return results
  190. return self.runInteraction(
  191. "get_server_keys_json", _get_server_keys_json_txn
  192. )