types.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. # Copyright (c) 2020 Tulir Asokan
  2. #
  3. # This Source Code Form is subject to the terms of the Mozilla Public
  4. # License, v. 2.0. If a copy of the MPL was not distributed with this
  5. # file, You can obtain one at http://mozilla.org/MPL/2.0/.
  6. from typing import Optional, Dict, Any, List, NewType
  7. from uuid import UUID
  8. from attr import dataclass
  9. import attr
  10. from mautrix.types import SerializableAttrs, SerializableEnum
  11. GroupID = NewType('GroupID', str)
  12. @dataclass
  13. class Account(SerializableAttrs['Account']):
  14. device_id: int = attr.ib(metadata={"json": "deviceId"})
  15. username: str
  16. filename: str
  17. registered: bool
  18. has_keys: bool
  19. subscribed: bool
  20. uuid: Optional[UUID] = None
  21. @dataclass(frozen=True, eq=False)
  22. class Address(SerializableAttrs['Address']):
  23. number: Optional[str] = None
  24. uuid: Optional[UUID] = None
  25. @property
  26. def is_valid(self) -> bool:
  27. return bool(self.number) or bool(self.uuid)
  28. @property
  29. def best_identifier(self) -> str:
  30. return str(self.uuid) if self.uuid else self.number
  31. def __eq__(self, other: 'Address') -> bool:
  32. if not isinstance(other, Address):
  33. return False
  34. if self.uuid and other.uuid:
  35. return self.uuid == other.uuid
  36. elif self.number and other.number:
  37. return self.number == other.number
  38. return False
  39. def __hash__(self) -> int:
  40. if self.uuid:
  41. return hash(self.uuid)
  42. return hash(self.number)
  43. @classmethod
  44. def parse(cls, value: str) -> 'Address':
  45. return Address(number=value) if value.startswith("+") else Address(uuid=UUID(value))
  46. @dataclass
  47. class TrustLevel(SerializableEnum):
  48. TRUSTED_UNVERIFIED = "TRUSTED_UNVERIFIED"
  49. TRUSTED_VERIFIED = "TRUSTED_VERIFIED"
  50. UNTRUSTED = "UNTRUSTED"
  51. @dataclass
  52. class Identity(SerializableAttrs['Identity']):
  53. trust_level: TrustLevel
  54. added: int
  55. fingerprint: str
  56. safety_number: str
  57. qr_code_data: str
  58. address: Address
  59. @dataclass
  60. class GetIdentitiesResponse(SerializableAttrs['GetIdentitiesResponse']):
  61. identities: List[Identity]
  62. @dataclass
  63. class Contact(SerializableAttrs['Contact']):
  64. address: Address
  65. name: Optional[str] = None
  66. color: Optional[str] = None
  67. profile_key: Optional[str] = attr.ib(default=None, metadata={"json": "profileKey"})
  68. message_expiration_time: int = attr.ib(default=0, metadata={"json": "messageExpirationTime"})
  69. @dataclass
  70. class Capabilities(SerializableAttrs['Capabilities']):
  71. gv2: bool = False
  72. storage: bool = False
  73. gv1_migration: bool = attr.ib(default=False, metadata={"json": "gv1-migration"})
  74. @dataclass
  75. class Profile(SerializableAttrs['Profile']):
  76. name: str = ""
  77. profile_name: str = ""
  78. avatar: str = ""
  79. identity_key: str = ""
  80. unidentified_access: str = ""
  81. unrestricted_unidentified_access: bool = False
  82. address: Optional[Address] = None
  83. expiration_time: int = 0
  84. capabilities: Optional[Capabilities] = None
  85. @dataclass
  86. class Group(SerializableAttrs['Group']):
  87. group_id: GroupID = attr.ib(metadata={"json": "groupId"})
  88. name: str = "Unknown group"
  89. # Sometimes "UPDATE"
  90. type: Optional[str] = None
  91. # Not always present
  92. members: List[Address] = attr.ib(factory=lambda: [])
  93. avatar_id: int = attr.ib(default=0, metadata={"json": "avatarId"})
  94. @dataclass(kw_only=True)
  95. class GroupV2ID(SerializableAttrs['GroupV2ID']):
  96. id: GroupID
  97. revision: Optional[int] = None
  98. class AccessControlMode(SerializableEnum):
  99. UNKNOWN = "UNKNOWN"
  100. ANY = "ANY"
  101. MEMBER = "MEMBER"
  102. ADMINISTRATOR = "ADMINISTRATOR"
  103. UNSATISFIABLE = "UNSATISFIABLE"
  104. UNRECOGNIZED = "UNRECOGNIZED"
  105. @dataclass
  106. class GroupAccessControl(SerializableAttrs['GroupAccessControl']):
  107. attributes: AccessControlMode = AccessControlMode.UNKNOWN
  108. link: AccessControlMode = AccessControlMode.UNKNOWN
  109. members: AccessControlMode = AccessControlMode.UNKNOWN
  110. class GroupMemberRole(SerializableEnum):
  111. UNKNOWN = "UNKNOWN"
  112. DEFAULT = "DEFAULT"
  113. ADMINISTRATOR = "ADMINISTRATOR"
  114. UNRECOGNIZED = "UNRECOGNIZED"
  115. @dataclass
  116. class GroupMember(SerializableAttrs['GroupMember']):
  117. uuid: UUID
  118. joined_revision: int = 0
  119. role: GroupMemberRole = GroupMemberRole.UNKNOWN
  120. @dataclass(kw_only=True)
  121. class GroupV2(GroupV2ID, SerializableAttrs['GroupV2']):
  122. title: str
  123. avatar: Optional[str] = None
  124. timer: Optional[int] = None
  125. master_key: Optional[str] = attr.ib(default=None, metadata={"json": "masterKey"})
  126. invite_link: Optional[str] = attr.ib(default=None, metadata={"json": "inviteLink"})
  127. access_control: GroupAccessControl = attr.ib(factory=lambda: GroupAccessControl(),
  128. metadata={"json": "accessControl"})
  129. members: List[Address]
  130. member_detail: List[GroupMember] = attr.ib(factory=lambda: [],
  131. metadata={"json": "memberDetail"})
  132. pending_members: List[Address] = attr.ib(factory=lambda: [],
  133. metadata={"json": "pendingMembers"})
  134. pending_member_detail: List[GroupMember] = attr.ib(factory=lambda: [],
  135. metadata={"json": "pendingMemberDetail"})
  136. requesting_members: List[Address] = attr.ib(factory=lambda: [],
  137. metadata={"json": "requestingMembers"})
  138. @dataclass
  139. class Attachment(SerializableAttrs['Attachment']):
  140. width: int = 0
  141. height: int = 0
  142. caption: Optional[str] = None
  143. preview: Optional[str] = None
  144. blurhash: Optional[str] = None
  145. voice_note: bool = attr.ib(default=False, metadata={"json": "voiceNote"})
  146. content_type: Optional[str] = attr.ib(default=None, metadata={"json": "contentType"})
  147. custom_filename: Optional[str] = attr.ib(default=None, metadata={"json": "customFilename"})
  148. # Only for incoming
  149. id: Optional[str] = None
  150. incoming_filename: Optional[str] = attr.ib(default=None, metadata={"json": "storedFilename"})
  151. digest: Optional[str] = None
  152. # Only for outgoing
  153. outgoing_filename: Optional[str] = attr.ib(default=None, metadata={"json": "filename"})
  154. @dataclass
  155. class Quote(SerializableAttrs['Quote']):
  156. id: int
  157. author: Address
  158. text: Optional[str] = None
  159. # TODO: attachments, mentions
  160. @dataclass(kw_only=True)
  161. class Reaction(SerializableAttrs['Reaction']):
  162. emoji: str
  163. remove: bool = False
  164. target_author: Address = attr.ib(metadata={"json": "targetAuthor"})
  165. target_sent_timestamp: int = attr.ib(metadata={"json": "targetSentTimestamp"})
  166. @dataclass
  167. class Sticker(SerializableAttrs['Sticker']):
  168. attachment: Attachment
  169. pack_id: str = attr.ib(metadata={"json": "packID"})
  170. pack_key: str = attr.ib(metadata={"json": "packKey"})
  171. sticker_id: int = attr.ib(metadata={"json": "stickerID"})
  172. @dataclass
  173. class RemoteDelete(SerializableAttrs['RemoteDelete']):
  174. target_sent_timestamp: int = attr.ib(metadata={"json": "targetSentTimestamp"})
  175. @dataclass
  176. class Mention(SerializableAttrs['Mention']):
  177. uuid: UUID
  178. length: int
  179. start: int = 0
  180. @dataclass
  181. class MessageData(SerializableAttrs['MessageData']):
  182. timestamp: int
  183. body: Optional[str] = None
  184. quote: Optional[Quote] = None
  185. reaction: Optional[Reaction] = None
  186. attachments: List[Attachment] = attr.ib(factory=lambda: [])
  187. sticker: Optional[Sticker] = None
  188. mentions: List[Mention] = attr.ib(factory=lambda: [])
  189. group: Optional[Group] = None
  190. group_v2: Optional[GroupV2ID] = attr.ib(default=None, metadata={"json": "groupV2"})
  191. end_session: bool = attr.ib(default=False, metadata={"json": "endSession"})
  192. expires_in_seconds: int = attr.ib(default=0, metadata={"json": "expiresInSeconds"})
  193. profile_key_update: bool = attr.ib(default=False, metadata={"json": "profileKeyUpdate"})
  194. view_once: bool = attr.ib(default=False, metadata={"json": "viewOnce"})
  195. remote_delete: Optional[RemoteDelete] = attr.ib(default=None,
  196. metadata={"json": "remoteDelete"})
  197. @dataclass
  198. class SentSyncMessage(SerializableAttrs['SentSyncMessage']):
  199. message: MessageData
  200. timestamp: int
  201. expiration_start_timestamp: Optional[int] = attr.ib(default=None, metadata={
  202. "json": "expirationStartTimestamp"})
  203. is_recipient_update: bool = attr.ib(default=False, metadata={"json": "isRecipientUpdate"})
  204. unidentified_status: Dict[str, bool] = attr.ib(factory=lambda: {})
  205. destination: Optional[Address] = None
  206. class TypingAction(SerializableEnum):
  207. UNKNOWN = "UNKNOWN"
  208. STARTED = "STARTED"
  209. STOPPED = "STOPPED"
  210. @dataclass
  211. class TypingNotification(SerializableAttrs['TypingNotification']):
  212. action: TypingAction
  213. timestamp: int
  214. group_id: Optional[GroupID] = attr.ib(default=None, metadata={"json": "groupId"})
  215. @dataclass
  216. class OwnReadReceipt(SerializableAttrs['OwnReadReceipt']):
  217. sender: Address
  218. timestamp: int
  219. class ReceiptType(SerializableEnum):
  220. UNKNOWN = "UNKNOWN"
  221. DELIVERY = "DELIVERY"
  222. READ = "READ"
  223. @dataclass
  224. class Receipt(SerializableAttrs['Receipt']):
  225. type: ReceiptType
  226. timestamps: List[int]
  227. when: int
  228. @dataclass
  229. class SyncMessage(SerializableAttrs['SyncMessage']):
  230. sent: Optional[SentSyncMessage] = None
  231. typing: Optional[TypingNotification] = None
  232. read_messages: Optional[List[OwnReadReceipt]] = attr.ib(default=None,
  233. metadata={"json": "readMessages"})
  234. contacts: Optional[Dict[str, Any]] = None
  235. contacts_complete: bool = attr.ib(default=False, metadata={"json": "contactsComplete"})
  236. class MessageType(SerializableEnum):
  237. CIPHERTEXT = "CIPHERTEXT"
  238. UNIDENTIFIED_SENDER = "UNIDENTIFIED_SENDER"
  239. RECEIPT = "RECEIPT"
  240. PREKEY_BUNDLE = "PREKEY_BUNDLE"
  241. KEY_EXCHANGE = "KEY_EXCHANGE"
  242. UNKNOWN = "UNKNOWN"
  243. @dataclass(kw_only=True)
  244. class Message(SerializableAttrs['Message']):
  245. username: str
  246. source: Address
  247. timestamp: int
  248. timestamp_iso: str = attr.ib(metadata={"json": "timestampISO"})
  249. type: MessageType
  250. source_device: Optional[int] = attr.ib(metadata={"json": "sourceDevice"}, default=None)
  251. server_timestamp: Optional[int] = attr.ib(metadata={"json": "serverTimestamp"}, default=None)
  252. server_delivered_timestamp: int = attr.ib(metadata={"json": "serverDeliveredTimestamp"})
  253. has_content: bool = attr.ib(metadata={"json": "hasContent"}, default=False)
  254. is_unidentified_sender: Optional[bool] = attr.ib(metadata={"json": "isUnidentifiedSender"},
  255. default=None)
  256. has_legacy_message: bool = attr.ib(default=False, metadata={"json": "hasLegacyMessage"})
  257. data_message: Optional[MessageData] = attr.ib(default=None, metadata={"json": "dataMessage"})
  258. sync_message: Optional[SyncMessage] = attr.ib(default=None, metadata={"json": "syncMessage"})
  259. typing: Optional[TypingNotification] = None
  260. receipt: Optional[Receipt] = None
  261. class ListenAction(SerializableEnum):
  262. STARTED = "started"
  263. STOPPED = "stopped"
  264. @dataclass
  265. class ListenEvent(SerializableAttrs['ListenEvent']):
  266. action: ListenAction
  267. username: str
  268. exception: Optional[str] = None