1//! Matrix event identifiers.
23use ruma_macros::IdZst;
45use super::ServerName;
67/// A Matrix [event ID].
8///
9/// An `EventId` is generated randomly or converted from a string slice, and can be converted back
10/// into a string as needed.
11///
12/// # Room versions
13///
14/// Matrix specifies multiple [room versions] and the format of event identifiers differ between
15/// them. The original format used by room versions 1 and 2 uses a short pseudorandom "localpart"
16/// followed by the hostname and port of the originating homeserver. Later room versions change
17/// event identifiers to be a hash of the event encoded with Base64. Some of the methods provided by
18/// `EventId` are only relevant to the original event format.
19///
20/// ```
21/// # use ruma_common::EventId;
22/// // Original format
23/// assert_eq!(<&EventId>::try_from("$h29iv0s8:example.com").unwrap(), "$h29iv0s8:example.com");
24/// // Room version 3 format
25/// assert_eq!(
26/// <&EventId>::try_from("$acR1l0raoZnm60CBwAVgqbZqoO/mYU81xysh1u7XcJk").unwrap(),
27/// "$acR1l0raoZnm60CBwAVgqbZqoO/mYU81xysh1u7XcJk"
28/// );
29/// // Room version 4 format
30/// assert_eq!(
31/// <&EventId>::try_from("$Rqnc-F-dvnEYJTyHq_iKxU2bZ1CI92-kuZq3a5lr5Zg").unwrap(),
32/// "$Rqnc-F-dvnEYJTyHq_iKxU2bZ1CI92-kuZq3a5lr5Zg"
33/// );
34/// ```
35///
36/// [event ID]: https://spec.matrix.org/latest/appendices/#event-ids
37/// [room versions]: https://spec.matrix.org/latest/rooms/#complete-list-of-room-versions
38#[repr(transparent)]
39#[derive(PartialEq, Eq, PartialOrd, Ord, Hash, IdZst)]
40#[ruma_id(validate = ruma_identifiers_validation::event_id::validate)]
41pub struct EventId(str);
4243impl EventId {
44/// Attempts to generate an `EventId` for the given origin server with a localpart consisting
45 /// of 18 random ASCII characters.
46 ///
47 /// This should only be used for events in the original format as used by Matrix room versions
48 /// 1 and 2.
49#[cfg(feature = "rand")]
50 #[allow(clippy::new_ret_no_self)]
51pub fn new(server_name: &ServerName) -> OwnedEventId {
52Self::from_borrowed(&format!("${}:{server_name}", super::generate_localpart(18))).to_owned()
53 }
5455/// Returns the event's unique ID.
56 ///
57 /// For the original event format as used by Matrix room versions 1 and 2, this is the
58 /// "localpart" that precedes the homeserver. For later formats, this is the entire ID without
59 /// the leading `$` sigil.
60pub fn localpart(&self) -> &str {
61let idx = self.colon_idx().unwrap_or_else(|| self.as_str().len());
62&self.as_str()[1..idx]
63 }
6465/// Returns the server name of the event ID.
66 ///
67 /// Only applicable to events in the original format as used by Matrix room versions 1 and 2.
68pub fn server_name(&self) -> Option<&ServerName> {
69self.colon_idx().map(|idx| ServerName::from_borrowed(&self.as_str()[idx + 1..]))
70 }
7172fn colon_idx(&self) -> Option<usize> {
73self.as_str().find(':')
74 }
75}
7677#[cfg(test)]
78mod tests {
79use super::{EventId, OwnedEventId};
80use crate::IdParseError;
8182#[test]
83fn valid_original_event_id() {
84assert_eq!(
85 <&EventId>::try_from("$39hvsi03hlne:example.com").expect("Failed to create EventId."),
86"$39hvsi03hlne:example.com"
87);
88 }
8990#[test]
91fn valid_base64_event_id() {
92assert_eq!(
93 <&EventId>::try_from("$acR1l0raoZnm60CBwAVgqbZqoO/mYU81xysh1u7XcJk")
94 .expect("Failed to create EventId."),
95"$acR1l0raoZnm60CBwAVgqbZqoO/mYU81xysh1u7XcJk"
96);
97 }
9899#[test]
100fn valid_url_safe_base64_event_id() {
101assert_eq!(
102 <&EventId>::try_from("$Rqnc-F-dvnEYJTyHq_iKxU2bZ1CI92-kuZq3a5lr5Zg")
103 .expect("Failed to create EventId."),
104"$Rqnc-F-dvnEYJTyHq_iKxU2bZ1CI92-kuZq3a5lr5Zg"
105);
106 }
107108#[cfg(feature = "rand")]
109 #[test]
110fn generate_random_valid_event_id() {
111use crate::server_name;
112113let event_id = EventId::new(server_name!("example.com"));
114let id_str = event_id.as_str();
115116assert!(id_str.starts_with('$'));
117assert_eq!(id_str.len(), 31);
118 }
119120#[test]
121fn serialize_valid_original_event_id() {
122assert_eq!(
123 serde_json::to_string(
124 <&EventId>::try_from("$39hvsi03hlne:example.com")
125 .expect("Failed to create EventId.")
126 )
127 .expect("Failed to convert EventId to JSON."),
128r#""$39hvsi03hlne:example.com""#
129);
130 }
131132#[test]
133fn serialize_valid_base64_event_id() {
134assert_eq!(
135 serde_json::to_string(
136 <&EventId>::try_from("$acR1l0raoZnm60CBwAVgqbZqoO/mYU81xysh1u7XcJk")
137 .expect("Failed to create EventId.")
138 )
139 .expect("Failed to convert EventId to JSON."),
140r#""$acR1l0raoZnm60CBwAVgqbZqoO/mYU81xysh1u7XcJk""#
141);
142 }
143144#[test]
145fn serialize_valid_url_safe_base64_event_id() {
146assert_eq!(
147 serde_json::to_string(
148 <&EventId>::try_from("$Rqnc-F-dvnEYJTyHq_iKxU2bZ1CI92-kuZq3a5lr5Zg")
149 .expect("Failed to create EventId.")
150 )
151 .expect("Failed to convert EventId to JSON."),
152r#""$Rqnc-F-dvnEYJTyHq_iKxU2bZ1CI92-kuZq3a5lr5Zg""#
153);
154 }
155156#[test]
157fn deserialize_valid_original_event_id() {
158assert_eq!(
159 serde_json::from_str::<OwnedEventId>(r#""$39hvsi03hlne:example.com""#)
160 .expect("Failed to convert JSON to EventId"),
161 <&EventId>::try_from("$39hvsi03hlne:example.com").expect("Failed to create EventId.")
162 );
163 }
164165#[test]
166fn deserialize_valid_base64_event_id() {
167assert_eq!(
168 serde_json::from_str::<OwnedEventId>(
169r#""$acR1l0raoZnm60CBwAVgqbZqoO/mYU81xysh1u7XcJk""#
170)
171 .expect("Failed to convert JSON to EventId"),
172 <&EventId>::try_from("$acR1l0raoZnm60CBwAVgqbZqoO/mYU81xysh1u7XcJk")
173 .expect("Failed to create EventId.")
174 );
175 }
176177#[test]
178fn deserialize_valid_url_safe_base64_event_id() {
179assert_eq!(
180 serde_json::from_str::<OwnedEventId>(
181r#""$Rqnc-F-dvnEYJTyHq_iKxU2bZ1CI92-kuZq3a5lr5Zg""#
182)
183 .expect("Failed to convert JSON to EventId"),
184 <&EventId>::try_from("$Rqnc-F-dvnEYJTyHq_iKxU2bZ1CI92-kuZq3a5lr5Zg")
185 .expect("Failed to create EventId.")
186 );
187 }
188189#[test]
190fn valid_original_event_id_with_explicit_standard_port() {
191assert_eq!(
192 <&EventId>::try_from("$39hvsi03hlne:example.com:443")
193 .expect("Failed to create EventId."),
194"$39hvsi03hlne:example.com:443"
195);
196 }
197198#[test]
199fn valid_original_event_id_with_non_standard_port() {
200assert_eq!(
201 <&EventId>::try_from("$39hvsi03hlne:example.com:5000")
202 .expect("Failed to create EventId."),
203"$39hvsi03hlne:example.com:5000"
204);
205 }
206207#[test]
208fn missing_original_event_id_sigil() {
209assert_eq!(
210 <&EventId>::try_from("39hvsi03hlne:example.com").unwrap_err(),
211 IdParseError::MissingLeadingSigil
212 );
213 }
214215#[test]
216fn missing_base64_event_id_sigil() {
217assert_eq!(
218 <&EventId>::try_from("acR1l0raoZnm60CBwAVgqbZqoO/mYU81xysh1u7XcJk").unwrap_err(),
219 IdParseError::MissingLeadingSigil
220 );
221 }
222223#[test]
224fn missing_url_safe_base64_event_id_sigil() {
225assert_eq!(
226 <&EventId>::try_from("Rqnc-F-dvnEYJTyHq_iKxU2bZ1CI92-kuZq3a5lr5Zg").unwrap_err(),
227 IdParseError::MissingLeadingSigil
228 );
229 }
230231#[test]
232fn invalid_event_id_host() {
233assert_eq!(
234 <&EventId>::try_from("$39hvsi03hlne:/").unwrap_err(),
235 IdParseError::InvalidServerName
236 );
237 }
238239#[test]
240fn invalid_event_id_port() {
241assert_eq!(
242 <&EventId>::try_from("$39hvsi03hlne:example.com:notaport").unwrap_err(),
243 IdParseError::InvalidServerName
244 );
245 }
246}