ruma_identity_service_api/invitation/store_invitation.rs
1//! `POST /_matrix/identity/*/store-invite`
2//!
3//! Store pending invitations to a user's third-party ID.
4
5pub mod v2 {
6 //! `/v2/` ([spec])
7 //!
8 //! [spec]: https://spec.matrix.org/latest/identity-service-api/#post_matrixidentityv2store-invite
9
10 use ruma_common::{
11 api::{request, response, Metadata},
12 metadata,
13 room::RoomType,
14 third_party_invite::IdentityServerBase64PublicKey,
15 thirdparty::Medium,
16 OwnedMxcUri, OwnedRoomAliasId, OwnedRoomId, OwnedUserId,
17 };
18 use serde::{ser::SerializeSeq, Deserialize, Serialize};
19
20 const METADATA: Metadata = metadata! {
21 method: POST,
22 rate_limited: false,
23 authentication: AccessToken,
24 history: {
25 1.0 => "/_matrix/identity/v2/store-invite",
26 }
27 };
28
29 /// Request type for the `store_invitation` endpoint.
30 #[request]
31 pub struct Request {
32 /// The type of the third party identifier for the invited user.
33 ///
34 /// Currently, only `Medium::Email` is supported.
35 pub medium: Medium,
36
37 /// The email address of the invited user.
38 pub address: String,
39
40 /// The Matrix room ID to which the user is invited.
41 pub room_id: OwnedRoomId,
42
43 /// The Matrix user ID of the inviting user.
44 pub sender: OwnedUserId,
45
46 /// The Matrix room alias for the room to which the user is invited.
47 ///
48 /// This should be retrieved from the `m.room.canonical` state event.
49 #[serde(skip_serializing_if = "Option::is_none")]
50 pub room_alias: Option<OwnedRoomAliasId>,
51
52 /// The Content URI for the room to which the user is invited.
53 ///
54 /// This should be retrieved from the `m.room.avatar` state event.
55 #[serde(skip_serializing_if = "Option::is_none")]
56 pub room_avatar_url: Option<OwnedMxcUri>,
57
58 /// The `join_rule` for the room to which the user is invited.
59 ///
60 /// This should be retrieved from the `m.room.join_rules` state event.
61 #[serde(skip_serializing_if = "Option::is_none")]
62 pub room_join_rules: Option<String>,
63
64 /// The name of the room to which the user is invited.
65 ///
66 /// This should be retrieved from the `m.room.name` state event.
67 #[serde(skip_serializing_if = "Option::is_none")]
68 pub room_name: Option<String>,
69
70 /// The type of the room to which the user is invited.
71 ///
72 /// This should be retrieved from the `m.room.create` state event.
73 #[serde(skip_serializing_if = "Option::is_none")]
74 pub room_type: Option<RoomType>,
75
76 /// The display name of the user ID initiating the invite.
77 #[serde(skip_serializing_if = "Option::is_none")]
78 pub sender_display_name: Option<String>,
79
80 /// The Content URI for the avater of the user ID initiating the invite.
81 #[serde(skip_serializing_if = "Option::is_none")]
82 pub sender_avatar_url: Option<OwnedMxcUri>,
83 }
84
85 /// Response type for the `store_invitation` endpoint.
86 #[response]
87 pub struct Response {
88 /// The generated token.
89 ///
90 /// Must be a string consisting of the characters `[0-9a-zA-Z.=_-]`. Its length must not
91 /// exceed 255 characters and it must not be empty.
92 pub token: String,
93
94 /// A list of [server's long-term public key, generated ephemeral public key].
95 pub public_keys: PublicKeys,
96
97 /// The generated (redacted) display_name.
98 ///
99 /// An example is `f...@b...`.
100 pub display_name: String,
101 }
102
103 impl Request {
104 /// Creates a new `Request with the given medium, email address, room ID and sender.
105 pub fn new(
106 medium: Medium,
107 address: String,
108 room_id: OwnedRoomId,
109 sender: OwnedUserId,
110 ) -> Self {
111 Self {
112 medium,
113 address,
114 room_id,
115 sender,
116 room_alias: None,
117 room_avatar_url: None,
118 room_join_rules: None,
119 room_name: None,
120 room_type: None,
121 sender_display_name: None,
122 sender_avatar_url: None,
123 }
124 }
125
126 /// Creates a new `Request` with the given email address, room ID and sender.
127 pub fn email(address: String, room_id: OwnedRoomId, sender: OwnedUserId) -> Self {
128 Self::new(Medium::Email, address, room_id, sender)
129 }
130 }
131
132 impl Response {
133 /// Creates a new `Response` with the given token, public keys and display name.
134 pub fn new(token: String, public_keys: PublicKeys, display_name: String) -> Self {
135 Self { token, public_keys, display_name }
136 }
137 }
138
139 /// The server's long-term public key and generated ephemeral public key.
140 #[derive(Debug, Clone)]
141 #[allow(clippy::exhaustive_structs)]
142 pub struct PublicKeys {
143 /// The server's long-term public key.
144 pub server_key: PublicKey,
145
146 /// The generated ephemeral public key.
147 pub ephemeral_key: PublicKey,
148 }
149
150 impl<'de> Deserialize<'de> for PublicKeys {
151 fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
152 where
153 D: serde::Deserializer<'de>,
154 {
155 let [server_key, ephemeral_key] = <[PublicKey; 2]>::deserialize(deserializer)?;
156
157 Ok(Self { server_key, ephemeral_key })
158 }
159 }
160
161 impl Serialize for PublicKeys {
162 fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
163 where
164 S: serde::Serializer,
165 {
166 let mut seq = serializer.serialize_seq(Some(2))?;
167
168 seq.serialize_element(&self.server_key)?;
169 seq.serialize_element(&self.ephemeral_key)?;
170
171 seq.end()
172 }
173 }
174
175 /// A server's long-term or ephemeral public key.
176 #[derive(Clone, Debug, Serialize, Deserialize)]
177 #[non_exhaustive]
178 pub struct PublicKey {
179 /// The public key, encoded using unpadded base64.
180 pub public_key: IdentityServerBase64PublicKey,
181
182 /// The URI of an endpoint where the validity of this key can be checked by passing it as a
183 /// `public_key` query parameter.
184 pub key_validity_url: String,
185 }
186
187 impl PublicKey {
188 /// Constructs a new `PublicKey` with the given encoded public key and key validity URL.
189 pub fn new(public_key: IdentityServerBase64PublicKey, key_validity_url: String) -> Self {
190 Self { public_key, key_validity_url }
191 }
192 }
193}