ruma_common/
api.rs

1//! Core types used to define the requests and responses for each endpoint in the various
2//! [Matrix API specifications][apis].
3//!
4//! When implementing a new Matrix API, each endpoint has a request type which implements
5//! [`IncomingRequest`] and [`OutgoingRequest`], and a response type connected via an associated
6//! type.
7//!
8//! An implementation of [`IncomingRequest`] or [`OutgoingRequest`] contains all the information
9//! about the HTTP method, the path and input parameters for requests, and the structure of a
10//! successful response. Such types can then be used by client code to make requests, and by server
11//! code to fulfill those requests.
12//!
13//! [apis]: https://spec.matrix.org/latest/#matrix-apis
14
15use std::{convert::TryInto as _, error::Error as StdError};
16
17use as_variant::as_variant;
18use bytes::BufMut;
19use serde::{Deserialize, Serialize};
20
21use self::error::{FromHttpRequestError, FromHttpResponseError, IntoHttpError};
22use crate::UserId;
23
24/// Convenient constructor for [`Metadata`] constants.
25///
26/// Usage:
27///
28/// ```
29/// # use ruma_common::{metadata, api::Metadata};
30/// const _: Metadata = metadata! {
31///     method: GET, // one of the associated constants of http::Method
32///     rate_limited: true,
33///     authentication: AccessToken, // one of the variants of api::AuthScheme
34///
35///     // history of endpoint paths
36///     // there must be at least one path but otherwise everything is optional
37///     history: {
38///         unstable => "/_matrix/foo/org.bar.msc9000/baz",
39///         unstable => "/_matrix/foo/org.bar.msc9000/qux",
40///         1.0 => "/_matrix/media/r0/qux",
41///         1.1 => "/_matrix/media/v3/qux",
42///         1.2 => deprecated,
43///         1.3 => removed,
44///     }
45/// };
46/// ```
47#[macro_export]
48macro_rules! metadata {
49    ( $( $field:ident: $rhs:tt ),+ $(,)? ) => {
50        $crate::api::Metadata {
51            $( $field: $crate::metadata!(@field $field: $rhs) ),+
52        }
53    };
54
55    ( @field method: $method:ident ) => { $crate::exports::http::Method::$method };
56
57    ( @field authentication: $scheme:ident ) => { $crate::api::AuthScheme::$scheme };
58
59    ( @field history: {
60        $( unstable => $unstable_path:literal, )*
61        $( $( $version:literal => $rhs:tt, )+ )?
62    } ) => {
63        $crate::metadata! {
64            @history_impl
65            [ $($unstable_path),* ]
66            // Flip left and right to avoid macro parsing ambiguities
67            $( $( $rhs = $version ),+ )?
68        }
69    };
70
71    // Simple literal case: used for description, name, rate_limited
72    ( @field $_field:ident: $rhs:expr ) => { $rhs };
73
74    ( @history_impl
75        [ $($unstable_path:literal),* ]
76        $(
77            $( $stable_path:literal = $version:literal ),+
78            $(,
79                deprecated = $deprecated_version:literal
80                $(, removed = $removed_version:literal )?
81            )?
82        )?
83    ) => {
84        $crate::api::VersionHistory::new(
85            &[ $( $unstable_path ),* ],
86            &[ $($(
87                ($crate::api::MatrixVersion::from_lit(stringify!($version)), $stable_path)
88            ),+)? ],
89            $crate::metadata!(@optional_version $($( $deprecated_version )?)?),
90            $crate::metadata!(@optional_version $($($( $removed_version )?)?)?),
91        )
92    };
93
94    ( @optional_version ) => { None };
95    ( @optional_version $version:literal ) => { Some($crate::api::MatrixVersion::from_lit(stringify!($version))) }
96}
97
98/// Generates [`OutgoingRequest`] and [`IncomingRequest`] implementations.
99///
100/// The `OutgoingRequest` impl is on the `Request` type this attribute is used on. It is
101/// feature-gated behind `cfg(feature = "client")`.
102///
103/// The `IncomingRequest` impl is on `IncomingRequest`, which is either a type alias to
104/// `Request` or a fully-owned version of the same, depending of whether `Request` has any
105/// lifetime parameters. It is feature-gated behind `cfg(feature = "server")`.
106///
107/// The generated code expects a `METADATA` constant of type [`Metadata`] to be in scope,
108/// alongside a `Response` type that implements [`OutgoingResponse`] (for
109/// `cfg(feature = "server")`) and / or [`IncomingResponse`] (for `cfg(feature = "client")`).
110///
111/// By default, the type this macro is used on gets a `#[non_exhaustive]` attribute. This
112/// behavior can be controlled by setting the `ruma_unstable_exhaustive_types` compile-time
113/// `cfg` setting as `--cfg=ruma_unstable_exhaustive_types` using `RUSTFLAGS` or
114/// `.cargo/config.toml` (under `[build]` -> `rustflags = ["..."]`). When that setting is
115/// activated, the attribute is not applied so the type is exhaustive.
116///
117/// ## Attributes
118///
119/// To declare which part of the request a field belongs to:
120///
121/// * `#[ruma_api(header = HEADER_NAME)]`: Fields with this attribute will be treated as HTTP
122///   headers on the request. The value must implement `ToString` and `FromStr`. Generally this
123///   is a `String`. The attribute value shown above as `HEADER_NAME` must be a `const`
124///   expression of the type `http::header::HeaderName`, like one of the constants from
125///   `http::header`, e.g. `CONTENT_TYPE`. During deserialization of the request, if the field
126///   is an `Option` and parsing the header fails, the error will be ignored and the value will
127///   be `None`.
128/// * `#[ruma_api(path)]`: Fields with this attribute will be inserted into the matching path
129///   component of the request URL. If there are multiple of these fields, the order in which
130///   they are declared must match the order in which they occur in the request path.
131/// * `#[ruma_api(query)]`: Fields with this attribute will be inserting into the URL's query
132///   string.
133/// * `#[ruma_api(query_all)]`: Instead of individual query fields, one query_all field, of any
134///   type that can be (de)serialized by [serde_html_form], can be used for cases where
135///   multiple endpoints should share a query fields type, the query fields are better
136///   expressed as an `enum` rather than a `struct`, or the endpoint supports arbitrary query
137///   parameters.
138/// * No attribute: Fields without an attribute are part of the body. They can use `#[serde]`
139///   attributes to customize (de)serialization.
140/// * `#[ruma_api(body)]`: Use this if multiple endpoints should share a request body type, or
141///   the request body is better expressed as an `enum` rather than a `struct`. The value of
142///   the field will be used as the JSON body (rather than being a field in the request body
143///   object).
144/// * `#[ruma_api(raw_body)]`: Like `body` in that the field annotated with it represents the
145///   entire request body, but this attribute is for endpoints where the body can be anything,
146///   not just JSON. The field type must be `Vec<u8>`.
147///
148/// ## Examples
149///
150/// ```
151/// pub mod do_a_thing {
152///     use ruma_common::{api::request, OwnedRoomId};
153///     # use ruma_common::{
154///     #     api::{response, Metadata},
155///     #     metadata,
156///     # };
157///
158///     // const METADATA: Metadata = metadata! { ... };
159///     # const METADATA: Metadata = metadata! {
160///     #     method: POST,
161///     #     rate_limited: false,
162///     #     authentication: None,
163///     #     history: {
164///     #         unstable => "/_matrix/some/endpoint/:room_id",
165///     #     },
166///     # };
167///
168///     #[request]
169///     pub struct Request {
170///         #[ruma_api(path)]
171///         pub room_id: OwnedRoomId,
172///
173///         #[ruma_api(query)]
174///         pub bar: String,
175///
176///         #[serde(default)]
177///         pub foo: String,
178///     }
179///
180///     // #[response]
181///     // pub struct Response { ... }
182///     # #[response]
183///     # pub struct Response {}
184/// }
185///
186/// pub mod upload_file {
187///     use http::header::CONTENT_TYPE;
188///     use ruma_common::api::request;
189///     # use ruma_common::{
190///     #     api::{response, Metadata},
191///     #     metadata,
192///     # };
193///
194///     // const METADATA: Metadata = metadata! { ... };
195///     # const METADATA: Metadata = metadata! {
196///     #     method: POST,
197///     #     rate_limited: false,
198///     #     authentication: None,
199///     #     history: {
200///     #         unstable => "/_matrix/some/endpoint/:file_name",
201///     #     },
202///     # };
203///
204///     #[request]
205///     pub struct Request {
206///         #[ruma_api(path)]
207///         pub file_name: String,
208///
209///         #[ruma_api(header = CONTENT_TYPE)]
210///         pub content_type: String,
211///
212///         #[ruma_api(raw_body)]
213///         pub file: Vec<u8>,
214///     }
215///
216///     // #[response]
217///     // pub struct Response { ... }
218///     # #[response]
219///     # pub struct Response {}
220/// }
221/// ```
222///
223/// [serde_html_form]: https://crates.io/crates/serde_html_form
224pub use ruma_macros::request;
225/// Generates [`OutgoingResponse`] and [`IncomingResponse`] implementations.
226///
227/// The `OutgoingResponse` impl is feature-gated behind `cfg(feature = "server")`.
228/// The `IncomingResponse` impl is feature-gated behind `cfg(feature = "client")`.
229///
230/// The generated code expects a `METADATA` constant of type [`Metadata`] to be in scope.
231///
232/// By default, the type this macro is used on gets a `#[non_exhaustive]` attribute. This
233/// behavior can be controlled by setting the `ruma_unstable_exhaustive_types` compile-time
234/// `cfg` setting as `--cfg=ruma_unstable_exhaustive_types` using `RUSTFLAGS` or
235/// `.cargo/config.toml` (under `[build]` -> `rustflags = ["..."]`). When that setting is
236/// activated, the attribute is not applied so the type is exhaustive.
237///
238/// The status code of `OutgoingResponse` can be optionally overridden by adding the `status`
239/// attribute to `response`. The attribute value must be a status code constant from
240/// `http::StatusCode`, e.g. `IM_A_TEAPOT`.
241///
242/// ## Attributes
243///
244/// To declare which part of the response a field belongs to:
245///
246/// * `#[ruma_api(header = HEADER_NAME)]`: Fields with this attribute will be treated as HTTP
247///   headers on the response. The value must implement `ToString` and `FromStr`. Generally
248///   this is a `String`. The attribute value shown above as `HEADER_NAME` must be a header
249///   name constant from `http::header`, e.g. `CONTENT_TYPE`. During deserialization of the
250///   response, if the field is an `Option` and parsing the header fails, the error will be
251///   ignored and the value will be `None`.
252/// * No attribute: Fields without an attribute are part of the body. They can use `#[serde]`
253///   attributes to customize (de)serialization.
254/// * `#[ruma_api(body)]`: Use this if multiple endpoints should share a response body type, or
255///   the response body is better expressed as an `enum` rather than a `struct`. The value of
256///   the field will be used as the JSON body (rather than being a field in the response body
257///   object).
258/// * `#[ruma_api(raw_body)]`: Like `body` in that the field annotated with it represents the
259///   entire response body, but this attribute is for endpoints where the body can be anything,
260///   not just JSON. The field type must be `Vec<u8>`.
261///
262/// ## Examples
263///
264/// ```
265/// pub mod do_a_thing {
266///     use ruma_common::{api::response, OwnedRoomId};
267///     # use ruma_common::{
268///     #     api::{request, Metadata},
269///     #     metadata,
270///     # };
271///
272///     // const METADATA: Metadata = metadata! { ... };
273///     # const METADATA: Metadata = metadata! {
274///     #     method: POST,
275///     #     rate_limited: false,
276///     #     authentication: None,
277///     #     history: {
278///     #         unstable => "/_matrix/some/endpoint",
279///     #     },
280///     # };
281///
282///     // #[request]
283///     // pub struct Request { ... }
284///     # #[request]
285///     # pub struct Request { }
286///
287///     #[response(status = IM_A_TEAPOT)]
288///     pub struct Response {
289///         #[serde(skip_serializing_if = "Option::is_none")]
290///         pub foo: Option<String>,
291///     }
292/// }
293///
294/// pub mod download_file {
295///     use http::header::CONTENT_TYPE;
296///     use ruma_common::api::response;
297///     # use ruma_common::{
298///     #     api::{request, Metadata},
299///     #     metadata,
300///     # };
301///
302///     // const METADATA: Metadata = metadata! { ... };
303///     # const METADATA: Metadata = metadata! {
304///     #     method: POST,
305///     #     rate_limited: false,
306///     #     authentication: None,
307///     #     history: {
308///     #         unstable => "/_matrix/some/endpoint",
309///     #     },
310///     # };
311///
312///     // #[request]
313///     // pub struct Request { ... }
314///     # #[request]
315///     # pub struct Request { }
316///
317///     #[response]
318///     pub struct Response {
319///         #[ruma_api(header = CONTENT_TYPE)]
320///         pub content_type: String,
321///
322///         #[ruma_api(raw_body)]
323///         pub file: Vec<u8>,
324///     }
325/// }
326/// ```
327pub use ruma_macros::response;
328
329pub mod error;
330mod metadata;
331
332pub use self::metadata::{MatrixVersion, Metadata, VersionHistory, VersioningDecision};
333
334/// An enum to control whether an access token should be added to outgoing requests
335#[derive(Clone, Copy, Debug)]
336#[allow(clippy::exhaustive_enums)]
337pub enum SendAccessToken<'a> {
338    /// Add the given access token to the request only if the `METADATA` on the request requires
339    /// it.
340    IfRequired(&'a str),
341
342    /// Always add the access token.
343    Always(&'a str),
344
345    /// Add the given appservice token to the request only if the `METADATA` on the request
346    /// requires it.
347    Appservice(&'a str),
348
349    /// Don't add an access token.
350    ///
351    /// This will lead to an error if the request endpoint requires authentication
352    None,
353}
354
355impl<'a> SendAccessToken<'a> {
356    /// Get the access token for an endpoint that requires one.
357    ///
358    /// Returns `Some(_)` if `self` contains an access token.
359    pub fn get_required_for_endpoint(self) -> Option<&'a str> {
360        as_variant!(self, Self::IfRequired | Self::Appservice | Self::Always)
361    }
362
363    /// Get the access token for an endpoint that should not require one.
364    ///
365    /// Returns `Some(_)` only if `self` is `SendAccessToken::Always(_)`.
366    pub fn get_not_required_for_endpoint(self) -> Option<&'a str> {
367        as_variant!(self, Self::Always)
368    }
369
370    /// Gets the access token for an endpoint that requires one for appservices.
371    ///
372    /// Returns `Some(_)` if `self` is either `SendAccessToken::Appservice(_)`
373    /// or `SendAccessToken::Always(_)`
374    pub fn get_required_for_appservice(self) -> Option<&'a str> {
375        as_variant!(self, Self::Appservice | Self::Always)
376    }
377}
378
379/// A request type for a Matrix API endpoint, used for sending requests.
380pub trait OutgoingRequest: Sized + Clone {
381    /// A type capturing the expected error conditions the server can return.
382    type EndpointError: EndpointError;
383
384    /// Response type returned when the request is successful.
385    type IncomingResponse: IncomingResponse<EndpointError = Self::EndpointError>;
386
387    /// Metadata about the endpoint.
388    const METADATA: Metadata;
389
390    /// Tries to convert this request into an `http::Request`.
391    ///
392    /// On endpoints with authentication, when adequate information isn't provided through
393    /// access_token, this could result in an error. It may also fail with a serialization error
394    /// in case of bugs in Ruma though.
395    ///
396    /// It may also fail if, for every version in `considering_versions`;
397    /// - The endpoint is too old, and has been removed in all versions.
398    ///   ([`EndpointRemoved`](error::IntoHttpError::EndpointRemoved))
399    /// - The endpoint is too new, and no unstable path is known for this endpoint.
400    ///   ([`NoUnstablePath`](error::IntoHttpError::NoUnstablePath))
401    ///
402    /// Finally, this will emit a warning through `tracing` if it detects if any version in
403    /// `considering_versions` has deprecated this endpoint.
404    ///
405    /// The endpoints path will be appended to the given `base_url`, for example
406    /// `https://matrix.org`. Since all paths begin with a slash, it is not necessary for the
407    /// `base_url` to have a trailing slash. If it has one however, it will be ignored.
408    fn try_into_http_request<T: Default + BufMut>(
409        self,
410        base_url: &str,
411        access_token: SendAccessToken<'_>,
412        considering_versions: &'_ [MatrixVersion],
413    ) -> Result<http::Request<T>, IntoHttpError>;
414}
415
416/// A response type for a Matrix API endpoint, used for receiving responses.
417pub trait IncomingResponse: Sized {
418    /// A type capturing the expected error conditions the server can return.
419    type EndpointError: EndpointError;
420
421    /// Tries to convert the given `http::Response` into this response type.
422    fn try_from_http_response<T: AsRef<[u8]>>(
423        response: http::Response<T>,
424    ) -> Result<Self, FromHttpResponseError<Self::EndpointError>>;
425}
426
427/// An extension to [`OutgoingRequest`] which provides Appservice specific methods.
428pub trait OutgoingRequestAppserviceExt: OutgoingRequest {
429    /// Tries to convert this request into an `http::Request` and appends a virtual `user_id` to
430    /// [assert Appservice identity][id_assert].
431    ///
432    /// [id_assert]: https://spec.matrix.org/latest/application-service-api/#identity-assertion
433    fn try_into_http_request_with_user_id<T: Default + BufMut>(
434        self,
435        base_url: &str,
436        access_token: SendAccessToken<'_>,
437        user_id: &UserId,
438        considering_versions: &'_ [MatrixVersion],
439    ) -> Result<http::Request<T>, IntoHttpError> {
440        let mut http_request =
441            self.try_into_http_request(base_url, access_token, considering_versions)?;
442        let user_id_query = serde_html_form::to_string([("user_id", user_id)])?;
443
444        let uri = http_request.uri().to_owned();
445        let mut parts = uri.into_parts();
446
447        let path_and_query_with_user_id = match &parts.path_and_query {
448            Some(path_and_query) => match path_and_query.query() {
449                Some(_) => format!("{path_and_query}&{user_id_query}"),
450                None => format!("{path_and_query}?{user_id_query}"),
451            },
452            None => format!("/?{user_id_query}"),
453        };
454
455        parts.path_and_query =
456            Some(path_and_query_with_user_id.try_into().map_err(http::Error::from)?);
457
458        *http_request.uri_mut() = parts.try_into().map_err(http::Error::from)?;
459
460        Ok(http_request)
461    }
462}
463
464impl<T: OutgoingRequest> OutgoingRequestAppserviceExt for T {}
465
466/// A request type for a Matrix API endpoint, used for receiving requests.
467pub trait IncomingRequest: Sized {
468    /// A type capturing the error conditions that can be returned in the response.
469    type EndpointError: EndpointError;
470
471    /// Response type to return when the request is successful.
472    type OutgoingResponse: OutgoingResponse;
473
474    /// Metadata about the endpoint.
475    const METADATA: Metadata;
476
477    /// Tries to turn the given `http::Request` into this request type,
478    /// together with the corresponding path arguments.
479    ///
480    /// Note: The strings in path_args need to be percent-decoded.
481    fn try_from_http_request<B, S>(
482        req: http::Request<B>,
483        path_args: &[S],
484    ) -> Result<Self, FromHttpRequestError>
485    where
486        B: AsRef<[u8]>,
487        S: AsRef<str>;
488}
489
490/// A request type for a Matrix API endpoint, used for sending responses.
491pub trait OutgoingResponse {
492    /// Tries to convert this response into an `http::Response`.
493    ///
494    /// This method should only fail when when invalid header values are specified. It may also
495    /// fail with a serialization error in case of bugs in Ruma though.
496    fn try_into_http_response<T: Default + BufMut>(
497        self,
498    ) -> Result<http::Response<T>, IntoHttpError>;
499}
500
501/// Gives users the ability to define their own serializable / deserializable errors.
502pub trait EndpointError: OutgoingResponse + StdError + Sized + Send + 'static {
503    /// Tries to construct `Self` from an `http::Response`.
504    ///
505    /// This will always return `Err` variant when no `error` field is defined in
506    /// the `ruma_api` macro.
507    fn from_http_response<T: AsRef<[u8]>>(response: http::Response<T>) -> Self;
508}
509
510/// Authentication scheme used by the endpoint.
511#[derive(Copy, Clone, Debug, PartialEq, Eq)]
512#[allow(clippy::exhaustive_enums)]
513pub enum AuthScheme {
514    /// No authentication is performed.
515    None,
516
517    /// Authentication is performed by including an access token in the `Authentication` http
518    /// header, or an `access_token` query parameter.
519    ///
520    /// Using the query parameter is deprecated since Matrix 1.11.
521    AccessToken,
522
523    /// Authentication is optional, and it is performed by including an access token in the
524    /// `Authentication` http header, or an `access_token` query parameter.
525    ///
526    /// Using the query parameter is deprecated since Matrix 1.11.
527    AccessTokenOptional,
528
529    /// Authentication is required, and can only be performed for appservices, by including an
530    /// appservice access token in the `Authentication` http header, or `access_token` query
531    /// parameter.
532    ///
533    /// Using the query parameter is deprecated since Matrix 1.11.
534    AppserviceToken,
535
536    /// No authentication is performed for clients, but it can be performed for appservices, by
537    /// including an appservice access token in the `Authentication` http header, or an
538    /// `access_token` query parameter.
539    ///
540    /// Using the query parameter is deprecated since Matrix 1.11.
541    AppserviceTokenOptional,
542
543    /// Authentication is performed by including X-Matrix signatures in the request headers,
544    /// as defined in the federation API.
545    ServerSignatures,
546}
547
548/// The direction to return events from.
549#[derive(Clone, Copy, Debug, Default, PartialEq, Eq, Deserialize, Serialize)]
550#[allow(clippy::exhaustive_enums)]
551pub enum Direction {
552    /// Return events backwards in time from the requested `from` token.
553    #[default]
554    #[serde(rename = "b")]
555    Backward,
556
557    /// Return events forwards in time from the requested `from` token.
558    #[serde(rename = "f")]
559    Forward,
560}