Files
continuwuity/src/api/client/push.rs
T

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

537 lines
16 KiB
Rust
Raw Normal View History

2024-07-16 08:05:25 +00:00
use axum::extract::State;
use conduwuit::{Err, err};
2020-07-30 18:14:47 +02:00
use ruma::{
CanonicalJsonObject, CanonicalJsonValue,
2020-07-30 18:14:47 +02:00
api::client::{
error::ErrorKind,
2022-02-18 15:33:14 +01:00
push::{
delete_pushrule, get_pushers, get_pushrule, get_pushrule_actions,
get_pushrule_enabled, get_pushrules_all, get_pushrules_global_scope, set_pusher,
set_pushrule, set_pushrule_actions, set_pushrule_enabled,
2021-01-24 16:05:52 +01:00
},
2020-07-30 18:14:47 +02:00
},
events::{
GlobalAccountDataEventType,
push_rules::{PushRulesEvent, PushRulesEventContent},
},
push::{
InsertPushRuleError, PredefinedContentRuleId, PredefinedOverrideRuleId,
RemovePushRuleError, Ruleset,
},
2020-07-30 18:14:47 +02:00
};
use service::Services;
2020-07-30 18:14:47 +02:00
2024-07-16 08:05:25 +00:00
use crate::{Error, Result, Ruma};
2024-03-05 19:48:54 -05:00
/// # `GET /_matrix/client/r0/pushrules/`
2021-08-31 19:14:37 +02:00
///
/// Retrieves the push rules event for this user.
2024-04-22 23:48:57 -04:00
pub(crate) async fn get_pushrules_all_route(
State(services): State<crate::State>,
body: Ruma<get_pushrules_all::v3::Request>,
2022-02-18 15:33:14 +01:00
) -> Result<get_pushrules_all::v3::Response> {
let sender_user = body.sender_user();
2024-03-05 19:48:54 -05:00
let Some(content_value) = services
2024-08-08 17:18:30 +00:00
.account_data
.get_global::<CanonicalJsonObject>(sender_user, GlobalAccountDataEventType::PushRules)
.await
.ok()
.and_then(|event| event.get("content").cloned())
.filter(CanonicalJsonValue::is_object)
else {
2024-08-08 17:18:30 +00:00
// user somehow has non-existent push rule event. recreate it and return server
// default silently
return recreate_push_rules_and_return(&services, sender_user).await;
};
2024-03-05 19:48:54 -05:00
let account_data_content =
serde_json::from_value::<PushRulesEventContent>(content_value.into()).map_err(|e| {
err!(Database(warn!("Invalid push rules account data event in database: {e}")))
})?;
2024-10-26 17:26:50 -04:00
let mut global_ruleset = account_data_content.global;
// remove old deprecated mentions push rules as per MSC4210
// and update the stored server default push rules
2024-10-26 17:26:50 -04:00
#[allow(deprecated)]
{
use ruma::push::RuleKind::*;
if global_ruleset
.get(Override, PredefinedOverrideRuleId::ContainsDisplayName.as_str())
.is_some()
|| global_ruleset
.get(Override, PredefinedOverrideRuleId::RoomNotif.as_str())
.is_some()
|| global_ruleset
.get(Content, PredefinedContentRuleId::ContainsUserName.as_str())
.is_some()
{
global_ruleset
.remove(Override, PredefinedOverrideRuleId::ContainsDisplayName)
.ok();
global_ruleset
.remove(Override, PredefinedOverrideRuleId::RoomNotif)
.ok();
global_ruleset
.remove(Content, PredefinedContentRuleId::ContainsUserName)
.ok();
global_ruleset.update_with_server_default(Ruleset::server_default(sender_user));
services
.account_data
.update(
None,
sender_user,
GlobalAccountDataEventType::PushRules.to_string().into(),
&serde_json::to_value(PushRulesEvent {
content: PushRulesEventContent { global: global_ruleset.clone() },
})
.expect("to json always works"),
)
.await?;
}
2024-10-26 17:26:50 -04:00
};
Ok(get_pushrules_all::v3::Response { global: global_ruleset })
2020-07-30 18:14:47 +02:00
}
/// # `GET /_matrix/client/r0/pushrules/global/`
///
/// Retrieves the push rules event for this user.
///
/// This appears to be the exact same as `GET /_matrix/client/r0/pushrules/`.
pub(crate) async fn get_pushrules_global_route(
State(services): State<crate::State>,
body: Ruma<get_pushrules_global_scope::v3::Request>,
) -> Result<get_pushrules_global_scope::v3::Response> {
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
let Some(content_value) = services
.account_data
.get_global::<CanonicalJsonObject>(sender_user, GlobalAccountDataEventType::PushRules)
.await
.ok()
.and_then(|event| event.get("content").cloned())
.filter(CanonicalJsonValue::is_object)
else {
// user somehow has non-existent push rule event. recreate it and return server
// default silently
services
.account_data
.update(
None,
sender_user,
GlobalAccountDataEventType::PushRules.to_string().into(),
&serde_json::to_value(PushRulesEvent {
content: PushRulesEventContent {
global: Ruleset::server_default(sender_user),
},
})
.expect("to json always works"),
)
.await?;
return Ok(get_pushrules_global_scope::v3::Response {
global: Ruleset::server_default(sender_user),
});
};
let account_data_content =
serde_json::from_value::<PushRulesEventContent>(content_value.into()).map_err(|e| {
err!(Database(warn!("Invalid push rules account data event in database: {e}")))
})?;
let mut global_ruleset = account_data_content.global;
// remove old deprecated mentions push rules as per MSC4210
// and update the stored server default push rules
#[allow(deprecated)]
{
use ruma::push::RuleKind::*;
if global_ruleset
.get(Override, PredefinedOverrideRuleId::ContainsDisplayName.as_str())
.is_some()
|| global_ruleset
.get(Override, PredefinedOverrideRuleId::RoomNotif.as_str())
.is_some()
|| global_ruleset
.get(Content, PredefinedContentRuleId::ContainsUserName.as_str())
.is_some()
{
global_ruleset
.remove(Override, PredefinedOverrideRuleId::ContainsDisplayName)
.ok();
global_ruleset
.remove(Override, PredefinedOverrideRuleId::RoomNotif)
.ok();
global_ruleset
.remove(Content, PredefinedContentRuleId::ContainsUserName)
.ok();
global_ruleset.update_with_server_default(Ruleset::server_default(sender_user));
services
.account_data
.update(
None,
sender_user,
GlobalAccountDataEventType::PushRules.to_string().into(),
&serde_json::to_value(PushRulesEvent {
content: PushRulesEventContent { global: global_ruleset.clone() },
})
.expect("to json always works"),
)
.await?;
}
};
Ok(get_pushrules_global_scope::v3::Response { global: global_ruleset })
}
2021-08-31 19:14:37 +02:00
/// # `GET /_matrix/client/r0/pushrules/{scope}/{kind}/{ruleId}`
///
/// Retrieves a single specified push rule for this user.
2024-07-16 08:05:25 +00:00
pub(crate) async fn get_pushrule_route(
State(services): State<crate::State>,
body: Ruma<get_pushrule::v3::Request>,
2024-07-16 08:05:25 +00:00
) -> Result<get_pushrule::v3::Response> {
2021-01-24 16:05:52 +01:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
2024-03-05 19:48:54 -05:00
2024-10-26 17:26:50 -04:00
// remove old deprecated mentions push rules as per MSC4210
#[allow(deprecated)]
if body.rule_id.as_str() == PredefinedContentRuleId::ContainsUserName.as_str()
|| body.rule_id.as_str() == PredefinedOverrideRuleId::ContainsDisplayName.as_str()
|| body.rule_id.as_str() == PredefinedOverrideRuleId::RoomNotif.as_str()
{
return Err!(Request(NotFound("Push rule not found.")));
}
let event: PushRulesEvent = services
2021-01-24 16:05:52 +01:00
.account_data
.get_global(sender_user, GlobalAccountDataEventType::PushRules)
2024-08-08 17:18:30 +00:00
.await
.map_err(|_| err!(Request(NotFound("PushRules event not found."))))?;
2024-03-05 19:48:54 -05:00
let rule = event
.content
2024-03-25 17:05:11 -04:00
.global
.get(body.kind.clone(), &body.rule_id)
.map(Into::into);
2024-03-05 19:48:54 -05:00
2021-01-24 16:05:52 +01:00
if let Some(rule) = rule {
Ok(get_pushrule::v3::Response { rule })
2021-01-24 16:05:52 +01:00
} else {
Err(Error::BadRequest(ErrorKind::NotFound, "Push rule not found."))
2021-01-24 16:05:52 +01:00
}
}
2024-10-26 17:26:50 -04:00
/// # `PUT /_matrix/client/r0/pushrules/global/{kind}/{ruleId}`
2021-08-31 19:14:37 +02:00
///
/// Creates a single specified push rule for this user.
2024-07-16 08:05:25 +00:00
pub(crate) async fn set_pushrule_route(
State(services): State<crate::State>,
body: Ruma<set_pushrule::v3::Request>,
2024-07-16 08:05:25 +00:00
) -> Result<set_pushrule::v3::Response> {
2021-12-15 13:58:25 +01:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
let body = body.body;
2024-03-05 19:48:54 -05:00
let mut account_data: PushRulesEvent = services
2021-01-24 16:05:52 +01:00
.account_data
.get_global(sender_user, GlobalAccountDataEventType::PushRules)
2024-08-08 17:18:30 +00:00
.await
.map_err(|_| err!(Request(NotFound("PushRules event not found."))))?;
2024-03-05 19:48:54 -05:00
if let Err(error) = account_data.content.global.insert(
body.rule.clone(),
body.after.as_deref(),
body.before.as_deref(),
) {
2023-02-26 16:55:42 +01:00
let err = match error {
| InsertPushRuleError::ServerDefaultRuleId => Error::BadRequest(
2023-02-26 16:55:42 +01:00
ErrorKind::InvalidParam,
"Rule IDs starting with a dot are reserved for server-default rules.",
),
| InsertPushRuleError::InvalidRuleId => Error::BadRequest(
ErrorKind::InvalidParam,
"Rule ID containing invalid characters.",
),
| InsertPushRuleError::RelativeToServerDefaultRule => Error::BadRequest(
2023-02-26 16:55:42 +01:00
ErrorKind::InvalidParam,
"Can't place a push rule relatively to a server-default rule.",
),
| InsertPushRuleError::UnknownRuleId => Error::BadRequest(
ErrorKind::NotFound,
"The before or after rule could not be found.",
),
| InsertPushRuleError::BeforeHigherThanAfter => Error::BadRequest(
2023-02-26 16:55:42 +01:00
ErrorKind::InvalidParam,
"The before rule has a higher priority than the after rule.",
),
| _ => Error::BadRequest(ErrorKind::InvalidParam, "Invalid data."),
2023-02-26 16:55:42 +01:00
};
2024-03-05 19:48:54 -05:00
2023-02-26 16:55:42 +01:00
return Err(err);
2021-01-24 16:05:52 +01:00
}
2024-03-05 19:48:54 -05:00
2024-08-08 17:18:30 +00:00
services
.account_data
.update(
None,
sender_user,
GlobalAccountDataEventType::PushRules.to_string().into(),
&serde_json::to_value(account_data).expect("to json value always works"),
)
.await?;
2024-03-05 19:48:54 -05:00
2022-02-18 15:33:14 +01:00
Ok(set_pushrule::v3::Response {})
2020-07-30 18:14:47 +02:00
}
2024-10-26 17:26:50 -04:00
/// # `GET /_matrix/client/r0/pushrules/global/{kind}/{ruleId}/actions`
2021-08-31 19:14:37 +02:00
///
/// Gets the actions of a single specified push rule for this user.
2024-04-22 23:48:57 -04:00
pub(crate) async fn get_pushrule_actions_route(
State(services): State<crate::State>,
body: Ruma<get_pushrule_actions::v3::Request>,
2022-02-18 15:33:14 +01:00
) -> Result<get_pushrule_actions::v3::Response> {
2021-01-24 16:05:52 +01:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
2024-03-05 19:48:54 -05:00
2024-10-26 17:26:50 -04:00
// remove old deprecated mentions push rules as per MSC4210
#[allow(deprecated)]
if body.rule_id.as_str() == PredefinedContentRuleId::ContainsUserName.as_str()
|| body.rule_id.as_str() == PredefinedOverrideRuleId::ContainsDisplayName.as_str()
|| body.rule_id.as_str() == PredefinedOverrideRuleId::RoomNotif.as_str()
{
return Err!(Request(NotFound("Push rule not found.")));
2021-01-24 16:05:52 +01:00
}
2024-03-05 19:48:54 -05:00
let event: PushRulesEvent = services
2021-01-24 16:05:52 +01:00
.account_data
.get_global(sender_user, GlobalAccountDataEventType::PushRules)
2024-08-08 17:18:30 +00:00
.await
.map_err(|_| err!(Request(NotFound("PushRules event not found."))))?;
2024-03-05 19:48:54 -05:00
let actions = event
.content
.global
2023-02-26 16:55:42 +01:00
.get(body.kind.clone(), &body.rule_id)
.map(|rule| rule.actions().to_owned())
.ok_or_else(|| err!(Request(NotFound("Push rule not found."))))?;
2024-03-05 19:48:54 -05:00
Ok(get_pushrule_actions::v3::Response { actions })
2021-01-24 16:05:52 +01:00
}
2024-10-26 17:26:50 -04:00
/// # `PUT /_matrix/client/r0/pushrules/global/{kind}/{ruleId}/actions`
2021-08-31 19:14:37 +02:00
///
/// Sets the actions of a single specified push rule for this user.
2024-04-22 23:48:57 -04:00
pub(crate) async fn set_pushrule_actions_route(
State(services): State<crate::State>,
body: Ruma<set_pushrule_actions::v3::Request>,
2022-02-18 15:33:14 +01:00
) -> Result<set_pushrule_actions::v3::Response> {
2021-01-24 16:05:52 +01:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
2024-03-05 19:48:54 -05:00
let mut account_data: PushRulesEvent = services
2021-01-24 16:05:52 +01:00
.account_data
.get_global(sender_user, GlobalAccountDataEventType::PushRules)
2024-08-08 17:18:30 +00:00
.await
.map_err(|_| err!(Request(NotFound("PushRules event not found."))))?;
2024-03-05 19:48:54 -05:00
2024-03-25 17:05:11 -04:00
if account_data
.content
.global
.set_actions(body.kind.clone(), &body.rule_id, body.actions.clone())
.is_err()
{
2023-02-26 16:55:42 +01:00
return Err(Error::BadRequest(ErrorKind::NotFound, "Push rule not found."));
}
2024-03-05 19:48:54 -05:00
2024-08-08 17:18:30 +00:00
services
.account_data
.update(
None,
sender_user,
GlobalAccountDataEventType::PushRules.to_string().into(),
&serde_json::to_value(account_data).expect("to json value always works"),
)
.await?;
2024-03-05 19:48:54 -05:00
2022-02-18 15:33:14 +01:00
Ok(set_pushrule_actions::v3::Response {})
2021-01-24 16:05:52 +01:00
}
2024-10-26 17:26:50 -04:00
/// # `GET /_matrix/client/r0/pushrules/global/{kind}/{ruleId}/enabled`
2021-08-31 19:14:37 +02:00
///
/// Gets the enabled status of a single specified push rule for this user.
2024-04-22 23:48:57 -04:00
pub(crate) async fn get_pushrule_enabled_route(
State(services): State<crate::State>,
body: Ruma<get_pushrule_enabled::v3::Request>,
2022-02-18 15:33:14 +01:00
) -> Result<get_pushrule_enabled::v3::Response> {
2021-01-24 16:05:52 +01:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
2024-03-05 19:48:54 -05:00
2024-10-26 17:26:50 -04:00
// remove old deprecated mentions push rules as per MSC4210
#[allow(deprecated)]
if body.rule_id.as_str() == PredefinedContentRuleId::ContainsUserName.as_str()
|| body.rule_id.as_str() == PredefinedOverrideRuleId::ContainsDisplayName.as_str()
|| body.rule_id.as_str() == PredefinedOverrideRuleId::RoomNotif.as_str()
{
return Ok(get_pushrule_enabled::v3::Response { enabled: false });
2021-01-24 16:05:52 +01:00
}
2024-03-05 19:48:54 -05:00
let event: PushRulesEvent = services
2021-01-24 16:05:52 +01:00
.account_data
.get_global(sender_user, GlobalAccountDataEventType::PushRules)
2024-08-08 17:18:30 +00:00
.await
.map_err(|_| err!(Request(NotFound("PushRules event not found."))))?;
2024-03-05 19:48:54 -05:00
let enabled = event
.content
.global
2023-02-26 16:55:42 +01:00
.get(body.kind.clone(), &body.rule_id)
2024-03-02 20:55:02 -05:00
.map(ruma::push::AnyPushRuleRef::enabled)
.ok_or_else(|| err!(Request(NotFound("Push rule not found."))))?;
2024-03-05 19:48:54 -05:00
Ok(get_pushrule_enabled::v3::Response { enabled })
2021-01-24 16:05:52 +01:00
}
2024-10-26 17:26:50 -04:00
/// # `PUT /_matrix/client/r0/pushrules/global/{kind}/{ruleId}/enabled`
2021-08-31 19:14:37 +02:00
///
/// Sets the enabled status of a single specified push rule for this user.
2024-04-22 23:48:57 -04:00
pub(crate) async fn set_pushrule_enabled_route(
State(services): State<crate::State>,
body: Ruma<set_pushrule_enabled::v3::Request>,
2022-02-18 15:33:14 +01:00
) -> Result<set_pushrule_enabled::v3::Response> {
2021-01-24 16:05:52 +01:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
2024-03-05 19:48:54 -05:00
let mut account_data: PushRulesEvent = services
2021-01-24 16:05:52 +01:00
.account_data
.get_global(sender_user, GlobalAccountDataEventType::PushRules)
2024-08-08 17:18:30 +00:00
.await
.map_err(|_| err!(Request(NotFound("PushRules event not found."))))?;
2024-03-05 19:48:54 -05:00
2024-03-25 17:05:11 -04:00
if account_data
.content
.global
.set_enabled(body.kind.clone(), &body.rule_id, body.enabled)
.is_err()
{
2023-02-26 16:55:42 +01:00
return Err(Error::BadRequest(ErrorKind::NotFound, "Push rule not found."));
2021-01-24 16:05:52 +01:00
}
2024-03-05 19:48:54 -05:00
2024-08-08 17:18:30 +00:00
services
.account_data
.update(
None,
sender_user,
GlobalAccountDataEventType::PushRules.to_string().into(),
&serde_json::to_value(account_data).expect("to json value always works"),
)
.await?;
2024-03-05 19:48:54 -05:00
2022-02-18 15:33:14 +01:00
Ok(set_pushrule_enabled::v3::Response {})
2020-07-30 18:14:47 +02:00
}
2024-10-26 17:26:50 -04:00
/// # `DELETE /_matrix/client/r0/pushrules/global/{kind}/{ruleId}`
2021-08-31 19:14:37 +02:00
///
/// Deletes a single specified push rule for this user.
2024-04-22 23:48:57 -04:00
pub(crate) async fn delete_pushrule_route(
State(services): State<crate::State>,
body: Ruma<delete_pushrule::v3::Request>,
2024-04-22 23:48:57 -04:00
) -> Result<delete_pushrule::v3::Response> {
2021-01-24 16:05:52 +01:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
2024-03-05 19:48:54 -05:00
let mut account_data: PushRulesEvent = services
2021-01-24 16:05:52 +01:00
.account_data
.get_global(sender_user, GlobalAccountDataEventType::PushRules)
2024-08-08 17:18:30 +00:00
.await
.map_err(|_| err!(Request(NotFound("PushRules event not found."))))?;
2024-03-05 19:48:54 -05:00
2024-03-25 17:05:11 -04:00
if let Err(error) = account_data
.content
.global
.remove(body.kind.clone(), &body.rule_id)
{
2023-02-26 16:55:42 +01:00
let err = match error {
| RemovePushRuleError::ServerDefault => Error::BadRequest(
ErrorKind::InvalidParam,
"Cannot delete a server-default pushrule.",
),
| RemovePushRuleError::NotFound =>
Error::BadRequest(ErrorKind::NotFound, "Push rule not found."),
| _ => Error::BadRequest(ErrorKind::InvalidParam, "Invalid data."),
2023-02-26 16:55:42 +01:00
};
2024-03-05 19:48:54 -05:00
2023-02-26 16:55:42 +01:00
return Err(err);
2021-01-24 16:05:52 +01:00
}
2024-03-05 19:48:54 -05:00
2024-08-08 17:18:30 +00:00
services
.account_data
.update(
None,
sender_user,
GlobalAccountDataEventType::PushRules.to_string().into(),
&serde_json::to_value(account_data).expect("to json value always works"),
)
.await?;
2024-03-05 19:48:54 -05:00
2022-02-18 15:33:14 +01:00
Ok(delete_pushrule::v3::Response {})
2021-01-24 16:05:52 +01:00
}
2021-08-31 19:14:37 +02:00
/// # `GET /_matrix/client/r0/pushers`
///
/// Gets all currently active pushers for the sender user.
2024-07-16 08:05:25 +00:00
pub(crate) async fn get_pushers_route(
State(services): State<crate::State>,
body: Ruma<get_pushers::v3::Request>,
2024-07-16 08:05:25 +00:00
) -> Result<get_pushers::v3::Response> {
2021-03-16 18:00:26 +01:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
2024-03-05 19:48:54 -05:00
2022-02-18 15:33:14 +01:00
Ok(get_pushers::v3::Response {
2024-08-08 17:18:30 +00:00
pushers: services.pusher.get_pushers(sender_user).await,
2022-01-22 16:58:32 +01:00
})
2020-07-30 18:14:47 +02:00
}
2021-08-31 19:14:37 +02:00
/// # `POST /_matrix/client/r0/pushers/set`
///
/// Adds a pusher for the sender user.
///
/// - TODO: Handle `append`
2024-07-16 08:05:25 +00:00
pub(crate) async fn set_pushers_route(
State(services): State<crate::State>,
body: Ruma<set_pusher::v3::Request>,
2024-07-16 08:05:25 +00:00
) -> Result<set_pusher::v3::Response> {
2021-03-16 18:00:26 +01:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
services
.pusher
.set_pusher(sender_user, &body.action)
.await?;
Ok(set_pusher::v3::Response::new())
2020-07-30 18:14:47 +02:00
}
/// user somehow has bad push rules, these must always exist per spec.
/// so recreate it and return server default silently
2024-08-08 17:18:30 +00:00
async fn recreate_push_rules_and_return(
services: &Services,
sender_user: &ruma::UserId,
) -> Result<get_pushrules_all::v3::Response> {
2024-08-08 17:18:30 +00:00
services
.account_data
.update(
None,
sender_user,
GlobalAccountDataEventType::PushRules.to_string().into(),
&serde_json::to_value(PushRulesEvent {
content: PushRulesEventContent {
global: Ruleset::server_default(sender_user),
},
})
.expect("to json always works"),
)
.await?;
Ok(get_pushrules_all::v3::Response {
global: Ruleset::server_default(sender_user),
})
}