1pub mod v3 {
6 use std::collections::BTreeMap;
11
12 use js_int::{uint, UInt};
13 use ruma_common::{
14 api::{request, response, Metadata},
15 metadata,
16 serde::{Raw, StringEnum},
17 OwnedEventId, OwnedMxcUri, OwnedRoomId, OwnedUserId,
18 };
19 use ruma_events::{AnyStateEvent, AnyTimelineEvent};
20 use serde::{Deserialize, Serialize};
21
22 use crate::{filter::RoomEventFilter, PrivOwnedStr};
23
24 const METADATA: Metadata = metadata! {
25 method: POST,
26 rate_limited: true,
27 authentication: AccessToken,
28 history: {
29 1.0 => "/_matrix/client/r0/search",
30 1.1 => "/_matrix/client/v3/search",
31 }
32 };
33
34 #[request(error = crate::Error)]
36 pub struct Request {
37 #[ruma_api(query)]
41 pub next_batch: Option<String>,
42
43 pub search_categories: Categories,
45 }
46
47 #[response(error = crate::Error)]
49 pub struct Response {
50 pub search_categories: ResultCategories,
52 }
53
54 impl Request {
55 pub fn new(search_categories: Categories) -> Self {
57 Self { next_batch: None, search_categories }
58 }
59 }
60
61 impl Response {
62 pub fn new(search_categories: ResultCategories) -> Self {
64 Self { search_categories }
65 }
66 }
67
68 #[derive(Clone, Debug, Default, Deserialize, Serialize)]
70 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
71 pub struct Categories {
72 #[serde(skip_serializing_if = "Option::is_none")]
74 pub room_events: Option<Criteria>,
75 }
76
77 impl Categories {
78 pub fn new() -> Self {
80 Default::default()
81 }
82 }
83
84 #[derive(Clone, Debug, Deserialize, Serialize)]
86 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
87 pub struct Criteria {
88 pub search_term: String,
90
91 #[serde(skip_serializing_if = "Option::is_none")]
95 pub keys: Option<Vec<SearchKeys>>,
96
97 #[serde(skip_serializing_if = "RoomEventFilter::is_empty")]
99 pub filter: RoomEventFilter,
100
101 #[serde(skip_serializing_if = "Option::is_none")]
103 pub order_by: Option<OrderBy>,
104
105 #[serde(default, skip_serializing_if = "EventContext::is_default")]
107 pub event_context: EventContext,
108
109 #[serde(skip_serializing_if = "Option::is_none")]
111 pub include_state: Option<bool>,
112
113 #[serde(default, skip_serializing_if = "Groupings::is_empty")]
115 pub groupings: Groupings,
116 }
117
118 impl Criteria {
119 pub fn new(search_term: String) -> Self {
121 Self {
122 search_term,
123 keys: None,
124 filter: RoomEventFilter::default(),
125 order_by: None,
126 event_context: Default::default(),
127 include_state: None,
128 groupings: Default::default(),
129 }
130 }
131 }
132
133 #[derive(Clone, Debug, Deserialize, Serialize)]
135 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
136 pub struct EventContext {
137 #[serde(
139 default = "default_event_context_limit",
140 skip_serializing_if = "is_default_event_context_limit"
141 )]
142 pub before_limit: UInt,
143
144 #[serde(
146 default = "default_event_context_limit",
147 skip_serializing_if = "is_default_event_context_limit"
148 )]
149 pub after_limit: UInt,
150
151 #[serde(default, skip_serializing_if = "ruma_common::serde::is_default")]
154 pub include_profile: bool,
155 }
156
157 fn default_event_context_limit() -> UInt {
158 uint!(5)
159 }
160
161 #[allow(clippy::trivially_copy_pass_by_ref)]
162 fn is_default_event_context_limit(val: &UInt) -> bool {
163 *val == default_event_context_limit()
164 }
165
166 impl EventContext {
167 pub fn new() -> Self {
169 Self {
170 before_limit: default_event_context_limit(),
171 after_limit: default_event_context_limit(),
172 include_profile: false,
173 }
174 }
175
176 pub fn is_default(&self) -> bool {
178 self.before_limit == default_event_context_limit()
179 && self.after_limit == default_event_context_limit()
180 && !self.include_profile
181 }
182 }
183
184 impl Default for EventContext {
185 fn default() -> Self {
186 Self::new()
187 }
188 }
189
190 #[derive(Clone, Debug, Default, Deserialize, Serialize)]
192 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
193 pub struct EventContextResult {
194 #[serde(skip_serializing_if = "Option::is_none")]
196 pub end: Option<String>,
197
198 #[serde(default, skip_serializing_if = "Vec::is_empty")]
200 pub events_after: Vec<Raw<AnyTimelineEvent>>,
201
202 #[serde(default, skip_serializing_if = "Vec::is_empty")]
204 pub events_before: Vec<Raw<AnyTimelineEvent>>,
205
206 #[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
208 pub profile_info: BTreeMap<OwnedUserId, UserProfile>,
209
210 #[serde(skip_serializing_if = "Option::is_none")]
212 pub start: Option<String>,
213 }
214
215 impl EventContextResult {
216 pub fn new() -> Self {
218 Default::default()
219 }
220
221 pub fn is_empty(&self) -> bool {
223 self.end.is_none()
224 && self.events_after.is_empty()
225 && self.events_before.is_empty()
226 && self.profile_info.is_empty()
227 && self.start.is_none()
228 }
229 }
230
231 #[derive(Clone, Default, Debug, Deserialize, Serialize)]
233 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
234 pub struct Grouping {
235 pub key: Option<GroupingKey>,
237 }
238
239 impl Grouping {
240 pub fn new() -> Self {
242 Default::default()
243 }
244
245 pub fn is_empty(&self) -> bool {
247 self.key.is_none()
248 }
249 }
250
251 #[doc = include_str!(concat!(env!("CARGO_MANIFEST_DIR"), "/src/doc/string_enum.md"))]
253 #[derive(Clone, PartialEq, Eq, PartialOrd, Ord, StringEnum)]
254 #[ruma_enum(rename_all = "snake_case")]
255 #[non_exhaustive]
256 pub enum GroupingKey {
257 RoomId,
259
260 Sender,
262
263 #[doc(hidden)]
264 _Custom(PrivOwnedStr),
265 }
266
267 #[derive(Clone, Default, Debug, Deserialize, Serialize)]
269 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
270 pub struct Groupings {
271 #[serde(default, skip_serializing_if = "<[_]>::is_empty")]
273 pub group_by: Vec<Grouping>,
274 }
275
276 impl Groupings {
277 pub fn new() -> Self {
279 Default::default()
280 }
281
282 pub fn is_empty(&self) -> bool {
284 self.group_by.is_empty()
285 }
286 }
287
288 #[doc = include_str!(concat!(env!("CARGO_MANIFEST_DIR"), "/src/doc/string_enum.md"))]
290 #[derive(Clone, PartialEq, Eq, StringEnum)]
291 #[non_exhaustive]
292 pub enum SearchKeys {
293 #[ruma_enum(rename = "content.body")]
295 ContentBody,
296
297 #[ruma_enum(rename = "content.name")]
299 ContentName,
300
301 #[ruma_enum(rename = "content.topic")]
303 ContentTopic,
304
305 #[doc(hidden)]
306 _Custom(PrivOwnedStr),
307 }
308
309 #[doc = include_str!(concat!(env!("CARGO_MANIFEST_DIR"), "/src/doc/string_enum.md"))]
311 #[derive(Clone, PartialEq, Eq, StringEnum)]
312 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
313 #[ruma_enum(rename_all = "snake_case")]
314 pub enum OrderBy {
315 Recent,
317
318 Rank,
321
322 #[doc(hidden)]
323 _Custom(PrivOwnedStr),
324 }
325
326 #[derive(Clone, Default, Debug, Deserialize, Serialize)]
328 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
329 pub struct ResultCategories {
330 #[serde(default, skip_serializing_if = "ResultRoomEvents::is_empty")]
332 pub room_events: ResultRoomEvents,
333 }
334
335 impl ResultCategories {
336 pub fn new() -> Self {
338 Default::default()
339 }
340 }
341
342 #[derive(Clone, Debug, Default, Deserialize, Serialize)]
344 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
345 pub struct ResultRoomEvents {
346 #[serde(skip_serializing_if = "Option::is_none")]
348 pub count: Option<UInt>,
349
350 #[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
352 pub groups: BTreeMap<GroupingKey, BTreeMap<OwnedRoomIdOrUserId, ResultGroup>>,
353
354 #[serde(skip_serializing_if = "Option::is_none")]
359 pub next_batch: Option<String>,
360
361 #[serde(default, skip_serializing_if = "Vec::is_empty")]
363 pub results: Vec<SearchResult>,
364
365 #[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
369 pub state: BTreeMap<OwnedRoomId, Vec<Raw<AnyStateEvent>>>,
370
371 #[serde(default, skip_serializing_if = "Vec::is_empty")]
374 pub highlights: Vec<String>,
375 }
376
377 impl ResultRoomEvents {
378 pub fn new() -> Self {
380 Default::default()
381 }
382
383 pub fn is_empty(&self) -> bool {
385 self.count.is_none()
386 && self.groups.is_empty()
387 && self.next_batch.is_none()
388 && self.results.is_empty()
389 && self.state.is_empty()
390 && self.highlights.is_empty()
391 }
392 }
393
394 #[derive(Clone, Debug, Default, Deserialize, Serialize)]
396 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
397 pub struct ResultGroup {
398 #[serde(skip_serializing_if = "Option::is_none")]
403 pub next_batch: Option<String>,
404
405 #[serde(skip_serializing_if = "Option::is_none")]
407 pub order: Option<UInt>,
408
409 #[serde(default, skip_serializing_if = "Vec::is_empty")]
411 pub results: Vec<OwnedEventId>,
412 }
413
414 impl ResultGroup {
415 pub fn new() -> Self {
417 Default::default()
418 }
419
420 pub fn is_empty(&self) -> bool {
422 self.next_batch.is_none() && self.order.is_none() && self.results.is_empty()
423 }
424 }
425
426 #[derive(Clone, Debug, Default, Deserialize, Serialize)]
428 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
429 pub struct SearchResult {
430 #[serde(skip_serializing_if = "EventContextResult::is_empty")]
432 pub context: EventContextResult,
433
434 #[serde(skip_serializing_if = "Option::is_none")]
438 pub rank: Option<f64>,
439
440 #[serde(skip_serializing_if = "Option::is_none")]
442 pub result: Option<Raw<AnyTimelineEvent>>,
443 }
444
445 impl SearchResult {
446 pub fn new() -> Self {
448 Default::default()
449 }
450
451 pub fn is_empty(&self) -> bool {
453 self.context.is_empty() && self.rank.is_none() && self.result.is_none()
454 }
455 }
456
457 #[derive(Clone, Debug, Default, Deserialize, Serialize)]
459 #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
460 pub struct UserProfile {
461 #[serde(skip_serializing_if = "Option::is_none")]
466 #[cfg_attr(
467 feature = "compat-empty-string-null",
468 serde(default, deserialize_with = "ruma_common::serde::empty_string_as_none")
469 )]
470 pub avatar_url: Option<OwnedMxcUri>,
471
472 #[serde(skip_serializing_if = "Option::is_none")]
474 pub displayname: Option<String>,
475 }
476
477 impl UserProfile {
478 pub fn new() -> Self {
480 Default::default()
481 }
482
483 pub fn is_empty(&self) -> bool {
485 self.avatar_url.is_none() && self.displayname.is_none()
486 }
487 }
488
489 #[derive(Clone, Debug, Deserialize, Eq, Ord, PartialEq, PartialOrd, Serialize)]
491 #[allow(clippy::exhaustive_enums)]
492 pub enum OwnedRoomIdOrUserId {
493 RoomId(OwnedRoomId),
495
496 UserId(OwnedUserId),
498 }
499}