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}