Admin API to un-revoke and edit registration tokens (#4637)

This commit is contained in:
Quentin Gliech
2025-06-06 12:56:43 +02:00
committed by GitHub
11 changed files with 1357 additions and 0 deletions

View File

@@ -136,6 +136,10 @@ where
get_with(
self::user_registration_tokens::get,
self::user_registration_tokens::get_doc,
)
.put_with(
self::user_registration_tokens::update,
self::user_registration_tokens::update_doc,
),
)
.api_route(
@@ -145,6 +149,13 @@ where
self::user_registration_tokens::revoke_doc,
),
)
.api_route(
"/user-registration-tokens/{id}/unrevoke",
post_with(
self::user_registration_tokens::unrevoke,
self::user_registration_tokens::unrevoke_doc,
),
)
.api_route(
"/upstream-oauth-links",
get_with(

View File

@@ -25,6 +25,9 @@ use crate::{
#[derive(Debug, thiserror::Error, OperationIo)]
#[aide(output_with = "Json<ErrorResponse>")]
pub enum RouteError {
#[error("A registration token with the same token already exists")]
Conflict(mas_data_model::UserRegistrationToken),
#[error(transparent)]
Internal(Box<dyn std::error::Error + Send + Sync + 'static>),
}
@@ -36,6 +39,7 @@ impl IntoResponse for RouteError {
let error = ErrorResponse::from_error(&self);
let sentry_event_id = record_error!(self, Self::Internal(_));
let status = match self {
Self::Conflict(_) => StatusCode::CONFLICT,
Self::Internal(_) => StatusCode::INTERNAL_SERVER_ERROR,
};
(status, sentry_event_id, Json(error)).into_response()
@@ -83,6 +87,12 @@ pub async fn handler(
.token
.unwrap_or_else(|| Alphanumeric.sample_string(&mut rng, 12));
// See if we have an existing token with the same token
let existing_token = repo.user_registration_token().find_by_token(&token).await?;
if let Some(existing_token) = existing_token {
return Err(RouteError::Conflict(existing_token));
}
let registration_token = repo
.user_registration_token()
.add(
@@ -196,4 +206,56 @@ mod tests {
}
"#);
}
#[sqlx::test(migrator = "mas_storage_pg::MIGRATOR")]
async fn test_create_conflict(pool: PgPool) {
setup();
let mut state = TestState::from_pool(pool).await.unwrap();
let token = state.token_with_scope("urn:mas:admin").await;
let request = Request::post("/api/admin/v1/user-registration-tokens")
.bearer(&token)
.json(serde_json::json!({
"token": "test_token_123",
"usage_limit": 5
}));
let response = state.request(request).await;
response.assert_status(StatusCode::CREATED);
let body: serde_json::Value = response.json();
assert_json_snapshot!(body, @r#"
{
"data": {
"type": "user-registration_token",
"id": "01FSHN9AG0MZAA6S4AF7CTV32E",
"attributes": {
"token": "test_token_123",
"valid": true,
"usage_limit": 5,
"times_used": 0,
"created_at": "2022-01-16T14:40:00Z",
"last_used_at": null,
"expires_at": null,
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
}
"#);
let request = Request::post("/api/admin/v1/user-registration-tokens")
.bearer(&token)
.json(serde_json::json!({
"token": "test_token_123",
"usage_limit": 5
}));
let response = state.request(request).await;
response.assert_status(StatusCode::CONFLICT);
}
}

View File

@@ -7,10 +7,14 @@ mod add;
mod get;
mod list;
mod revoke;
mod unrevoke;
mod update;
pub use self::{
add::{doc as add_doc, handler as add},
get::{doc as get_doc, handler as get},
list::{doc as list_doc, handler as list},
revoke::{doc as revoke_doc, handler as revoke},
unrevoke::{doc as unrevoke_doc, handler as unrevoke},
update::{doc as update_doc, handler as update},
};

View File

@@ -0,0 +1,237 @@
// Copyright 2025 The Matrix.org Foundation C.I.C.
//
// SPDX-License-Identifier: AGPL-3.0-only
// Please see LICENSE in the repository root for full details.
use aide::{OperationIo, transform::TransformOperation};
use axum::{Json, response::IntoResponse};
use hyper::StatusCode;
use mas_axum_utils::record_error;
use ulid::Ulid;
use crate::{
admin::{
call_context::CallContext,
model::{Resource, UserRegistrationToken},
params::UlidPathParam,
response::{ErrorResponse, SingleResponse},
},
impl_from_error_for_route,
};
#[derive(Debug, thiserror::Error, OperationIo)]
#[aide(output_with = "Json<ErrorResponse>")]
pub enum RouteError {
#[error(transparent)]
Internal(Box<dyn std::error::Error + Send + Sync + 'static>),
#[error("Registration token with ID {0} not found")]
NotFound(Ulid),
#[error("Registration token with ID {0} is not revoked")]
NotRevoked(Ulid),
}
impl_from_error_for_route!(mas_storage::RepositoryError);
impl IntoResponse for RouteError {
fn into_response(self) -> axum::response::Response {
let error = ErrorResponse::from_error(&self);
let sentry_event_id = record_error!(self, Self::Internal(_));
let status = match self {
Self::Internal(_) => StatusCode::INTERNAL_SERVER_ERROR,
Self::NotFound(_) => StatusCode::NOT_FOUND,
Self::NotRevoked(_) => StatusCode::BAD_REQUEST,
};
(status, sentry_event_id, Json(error)).into_response()
}
}
pub fn doc(operation: TransformOperation) -> TransformOperation {
operation
.id("unrevokeUserRegistrationToken")
.summary("Unrevoke a user registration token")
.description("Calling this endpoint will unrevoke a previously revoked user registration token, allowing it to be used for registrations again (subject to its usage limits and expiration).")
.tag("user-registration-token")
.response_with::<200, Json<SingleResponse<UserRegistrationToken>>, _>(|t| {
// Get the valid token sample
let [valid_token, _] = UserRegistrationToken::samples();
let id = valid_token.id();
let response = SingleResponse::new(valid_token, format!("/api/admin/v1/user-registration-tokens/{id}/unrevoke"));
t.description("Registration token was unrevoked").example(response)
})
.response_with::<400, RouteError, _>(|t| {
let response = ErrorResponse::from_error(&RouteError::NotRevoked(Ulid::nil()));
t.description("Token is not revoked").example(response)
})
.response_with::<404, RouteError, _>(|t| {
let response = ErrorResponse::from_error(&RouteError::NotFound(Ulid::nil()));
t.description("Registration token was not found").example(response)
})
}
#[tracing::instrument(name = "handler.admin.v1.user_registration_tokens.unrevoke", skip_all)]
pub async fn handler(
CallContext {
mut repo, clock, ..
}: CallContext,
id: UlidPathParam,
) -> Result<Json<SingleResponse<UserRegistrationToken>>, RouteError> {
let id = *id;
let token = repo
.user_registration_token()
.lookup(id)
.await?
.ok_or(RouteError::NotFound(id))?;
// Check if the token is not revoked
if token.revoked_at.is_none() {
return Err(RouteError::NotRevoked(id));
}
// Unrevoke the token using the repository method
let token = repo.user_registration_token().unrevoke(token).await?;
repo.save().await?;
Ok(Json(SingleResponse::new(
UserRegistrationToken::new(token, clock.now()),
format!("/api/admin/v1/user-registration-tokens/{id}/unrevoke"),
)))
}
#[cfg(test)]
mod tests {
use hyper::{Request, StatusCode};
use sqlx::PgPool;
use crate::test_utils::{RequestBuilderExt, ResponseExt, TestState, setup};
#[sqlx::test(migrator = "mas_storage_pg::MIGRATOR")]
async fn test_unrevoke_token(pool: PgPool) {
setup();
let mut state = TestState::from_pool(pool).await.unwrap();
let token = state.token_with_scope("urn:mas:admin").await;
let mut repo = state.repository().await.unwrap();
// Create a token
let registration_token = repo
.user_registration_token()
.add(
&mut state.rng(),
&state.clock,
"test_token_456".to_owned(),
Some(5),
None,
)
.await
.unwrap();
// Revoke it
let registration_token = repo
.user_registration_token()
.revoke(&state.clock, registration_token)
.await
.unwrap();
repo.save().await.unwrap();
// Now unrevoke it
let request = Request::post(format!(
"/api/admin/v1/user-registration-tokens/{}/unrevoke",
registration_token.id
))
.bearer(&token)
.empty();
let response = state.request(request).await;
response.assert_status(StatusCode::OK);
let body: serde_json::Value = response.json();
// The revoked_at timestamp should be null
insta::assert_json_snapshot!(body, @r#"
{
"data": {
"type": "user-registration_token",
"id": "01FSHN9AG0MZAA6S4AF7CTV32E",
"attributes": {
"token": "test_token_456",
"valid": true,
"usage_limit": 5,
"times_used": 0,
"created_at": "2022-01-16T14:40:00Z",
"last_used_at": null,
"expires_at": null,
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E/unrevoke"
}
}
"#);
}
#[sqlx::test(migrator = "mas_storage_pg::MIGRATOR")]
async fn test_unrevoke_not_revoked_token(pool: PgPool) {
setup();
let mut state = TestState::from_pool(pool).await.unwrap();
let token = state.token_with_scope("urn:mas:admin").await;
let mut repo = state.repository().await.unwrap();
let registration_token = repo
.user_registration_token()
.add(
&mut state.rng(),
&state.clock,
"test_token_789".to_owned(),
None,
None,
)
.await
.unwrap();
repo.save().await.unwrap();
// Try to unrevoke a token that's not revoked
let request = Request::post(format!(
"/api/admin/v1/user-registration-tokens/{}/unrevoke",
registration_token.id
))
.bearer(&token)
.empty();
let response = state.request(request).await;
response.assert_status(StatusCode::BAD_REQUEST);
let body: serde_json::Value = response.json();
assert_eq!(
body["errors"][0]["title"],
format!(
"Registration token with ID {} is not revoked",
registration_token.id
)
);
}
#[sqlx::test(migrator = "mas_storage_pg::MIGRATOR")]
async fn test_unrevoke_unknown_token(pool: PgPool) {
setup();
let mut state = TestState::from_pool(pool).await.unwrap();
let token = state.token_with_scope("urn:mas:admin").await;
let request = Request::post(
"/api/admin/v1/user-registration-tokens/01040G2081040G2081040G2081/unrevoke",
)
.bearer(&token)
.empty();
let response = state.request(request).await;
response.assert_status(StatusCode::NOT_FOUND);
let body: serde_json::Value = response.json();
assert_eq!(
body["errors"][0]["title"],
"Registration token with ID 01040G2081040G2081040G2081 not found"
);
}
}

View File

@@ -0,0 +1,511 @@
// Copyright 2025 The Matrix.org Foundation C.I.C.
//
// SPDX-License-Identifier: AGPL-3.0-only
// Please see LICENSE in the repository root for full details.
use aide::{OperationIo, transform::TransformOperation};
use axum::{Json, response::IntoResponse};
use chrono::{DateTime, Utc};
use hyper::StatusCode;
use mas_axum_utils::record_error;
use schemars::JsonSchema;
use serde::{Deserialize, Deserializer};
use ulid::Ulid;
use crate::{
admin::{
call_context::CallContext,
model::{Resource, UserRegistrationToken},
params::UlidPathParam,
response::{ErrorResponse, SingleResponse},
},
impl_from_error_for_route,
};
// Any value that is present is considered Some value, including null.
fn deserialize_some<'de, T, D>(deserializer: D) -> Result<Option<T>, D::Error>
where
T: Deserialize<'de>,
D: Deserializer<'de>,
{
Deserialize::deserialize(deserializer).map(Some)
}
/// # JSON payload for the `PUT /api/admin/v1/user-registration-tokens/{id}` endpoint
#[derive(Deserialize, JsonSchema)]
#[serde(rename = "EditUserRegistrationTokenRequest")]
pub struct Request {
/// New expiration date for the token, or null to remove expiration
#[serde(
skip_serializing_if = "Option::is_none",
default,
deserialize_with = "deserialize_some"
)]
#[expect(clippy::option_option)]
expires_at: Option<Option<DateTime<Utc>>>,
/// New usage limit for the token, or null to remove the limit
#[expect(clippy::option_option)]
#[serde(
skip_serializing_if = "Option::is_none",
default,
deserialize_with = "deserialize_some"
)]
usage_limit: Option<Option<u32>>,
}
#[derive(Debug, thiserror::Error, OperationIo)]
#[aide(output_with = "Json<ErrorResponse>")]
pub enum RouteError {
#[error(transparent)]
Internal(Box<dyn std::error::Error + Send + Sync + 'static>),
#[error("Registration token with ID {0} not found")]
NotFound(Ulid),
}
impl_from_error_for_route!(mas_storage::RepositoryError);
impl IntoResponse for RouteError {
fn into_response(self) -> axum::response::Response {
let error = ErrorResponse::from_error(&self);
let sentry_event_id = record_error!(self, Self::Internal(_));
let status = match self {
Self::Internal(_) => StatusCode::INTERNAL_SERVER_ERROR,
Self::NotFound(_) => StatusCode::NOT_FOUND,
};
(status, sentry_event_id, Json(error)).into_response()
}
}
pub fn doc(operation: TransformOperation) -> TransformOperation {
operation
.id("updateUserRegistrationToken")
.summary("Update a user registration token")
.description("Update properties of a user registration token such as expiration and usage limit. To set a field to null (removing the limit/expiration), include the field with a null value. To leave a field unchanged, omit it from the request body.")
.tag("user-registration-token")
.response_with::<200, Json<SingleResponse<UserRegistrationToken>>, _>(|t| {
// Get the valid token sample
let [valid_token, _] = UserRegistrationToken::samples();
let id = valid_token.id();
let response = SingleResponse::new(valid_token, format!("/api/admin/v1/user-registration-tokens/{id}"));
t.description("Registration token was updated").example(response)
})
.response_with::<404, RouteError, _>(|t| {
let response = ErrorResponse::from_error(&RouteError::NotFound(Ulid::nil()));
t.description("Registration token was not found").example(response)
})
}
#[tracing::instrument(name = "handler.admin.v1.user_registration_tokens.update", skip_all)]
pub async fn handler(
CallContext {
mut repo, clock, ..
}: CallContext,
id: UlidPathParam,
Json(request): Json<Request>,
) -> Result<Json<SingleResponse<UserRegistrationToken>>, RouteError> {
let id = *id;
// Get the token
let mut token = repo
.user_registration_token()
.lookup(id)
.await?
.ok_or(RouteError::NotFound(id))?;
// Update expiration if present in the request
if let Some(expires_at) = request.expires_at {
token = repo
.user_registration_token()
.set_expiry(token, expires_at)
.await?;
}
// Update usage limit if present in the request
if let Some(usage_limit) = request.usage_limit {
token = repo
.user_registration_token()
.set_usage_limit(token, usage_limit)
.await?;
}
repo.save().await?;
Ok(Json(SingleResponse::new(
UserRegistrationToken::new(token, clock.now()),
format!("/api/admin/v1/user-registration-tokens/{id}"),
)))
}
#[cfg(test)]
mod tests {
use chrono::Duration;
use hyper::{Request, StatusCode};
use mas_storage::Clock as _;
use serde_json::json;
use sqlx::PgPool;
use crate::test_utils::{RequestBuilderExt, ResponseExt, TestState, setup};
#[sqlx::test(migrator = "mas_storage_pg::MIGRATOR")]
async fn test_update_expiry(pool: PgPool) {
setup();
let mut state = TestState::from_pool(pool).await.unwrap();
let token = state.token_with_scope("urn:mas:admin").await;
let mut repo = state.repository().await.unwrap();
// Create a token without expiry
let registration_token = repo
.user_registration_token()
.add(
&mut state.rng(),
&state.clock,
"test_update_expiry".to_owned(),
None,
None,
)
.await
.unwrap();
repo.save().await.unwrap();
// Update with an expiry date
let future_date = state.clock.now() + Duration::days(30);
let request = Request::put(format!(
"/api/admin/v1/user-registration-tokens/{}",
registration_token.id
))
.bearer(&token)
.json(json!({
"expires_at": future_date
}));
let response = state.request(request).await;
response.assert_status(StatusCode::OK);
let body: serde_json::Value = response.json();
// Verify expiry was updated
insta::assert_json_snapshot!(body, @r#"
{
"data": {
"type": "user-registration_token",
"id": "01FSHN9AG0MZAA6S4AF7CTV32E",
"attributes": {
"token": "test_update_expiry",
"valid": true,
"usage_limit": null,
"times_used": 0,
"created_at": "2022-01-16T14:40:00Z",
"last_used_at": null,
"expires_at": "2022-02-15T14:40:00Z",
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
}
"#);
// Now remove the expiry
let request = Request::put(format!(
"/api/admin/v1/user-registration-tokens/{}",
registration_token.id
))
.bearer(&token)
.json(json!({
"expires_at": null
}));
let response = state.request(request).await;
response.assert_status(StatusCode::OK);
let body: serde_json::Value = response.json();
// Verify expiry was removed
insta::assert_json_snapshot!(body, @r#"
{
"data": {
"type": "user-registration_token",
"id": "01FSHN9AG0MZAA6S4AF7CTV32E",
"attributes": {
"token": "test_update_expiry",
"valid": true,
"usage_limit": null,
"times_used": 0,
"created_at": "2022-01-16T14:40:00Z",
"last_used_at": null,
"expires_at": null,
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
}
"#);
}
#[sqlx::test(migrator = "mas_storage_pg::MIGRATOR")]
async fn test_update_usage_limit(pool: PgPool) {
setup();
let mut state = TestState::from_pool(pool).await.unwrap();
let token = state.token_with_scope("urn:mas:admin").await;
let mut repo = state.repository().await.unwrap();
// Create a token with usage limit
let registration_token = repo
.user_registration_token()
.add(
&mut state.rng(),
&state.clock,
"test_update_limit".to_owned(),
Some(5),
None,
)
.await
.unwrap();
repo.save().await.unwrap();
// Update the usage limit
let request = Request::put(format!(
"/api/admin/v1/user-registration-tokens/{}",
registration_token.id
))
.bearer(&token)
.json(json!({
"usage_limit": 10
}));
let response = state.request(request).await;
response.assert_status(StatusCode::OK);
let body: serde_json::Value = response.json();
// Verify usage limit was updated
insta::assert_json_snapshot!(body, @r#"
{
"data": {
"type": "user-registration_token",
"id": "01FSHN9AG0MZAA6S4AF7CTV32E",
"attributes": {
"token": "test_update_limit",
"valid": true,
"usage_limit": 10,
"times_used": 0,
"created_at": "2022-01-16T14:40:00Z",
"last_used_at": null,
"expires_at": null,
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
}
"#);
// Now remove the usage limit
let request = Request::put(format!(
"/api/admin/v1/user-registration-tokens/{}",
registration_token.id
))
.bearer(&token)
.json(json!({
"usage_limit": null
}));
let response = state.request(request).await;
response.assert_status(StatusCode::OK);
let body: serde_json::Value = response.json();
// Verify usage limit was removed
insta::assert_json_snapshot!(body, @r#"
{
"data": {
"type": "user-registration_token",
"id": "01FSHN9AG0MZAA6S4AF7CTV32E",
"attributes": {
"token": "test_update_limit",
"valid": true,
"usage_limit": null,
"times_used": 0,
"created_at": "2022-01-16T14:40:00Z",
"last_used_at": null,
"expires_at": null,
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
}
"#);
}
#[sqlx::test(migrator = "mas_storage_pg::MIGRATOR")]
async fn test_update_multiple_fields(pool: PgPool) {
setup();
let mut state = TestState::from_pool(pool).await.unwrap();
let token = state.token_with_scope("urn:mas:admin").await;
let mut repo = state.repository().await.unwrap();
// Create a token
let registration_token = repo
.user_registration_token()
.add(
&mut state.rng(),
&state.clock,
"test_update_multiple".to_owned(),
None,
None,
)
.await
.unwrap();
repo.save().await.unwrap();
// Update both fields
let future_date = state.clock.now() + Duration::days(30);
let request = Request::put(format!(
"/api/admin/v1/user-registration-tokens/{}",
registration_token.id
))
.bearer(&token)
.json(json!({
"expires_at": future_date,
"usage_limit": 20
}));
let response = state.request(request).await;
response.assert_status(StatusCode::OK);
let body: serde_json::Value = response.json();
// Both fields were updated
insta::assert_json_snapshot!(body, @r#"
{
"data": {
"type": "user-registration_token",
"id": "01FSHN9AG0MZAA6S4AF7CTV32E",
"attributes": {
"token": "test_update_multiple",
"valid": true,
"usage_limit": 20,
"times_used": 0,
"created_at": "2022-01-16T14:40:00Z",
"last_used_at": null,
"expires_at": "2022-02-15T14:40:00Z",
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
}
"#);
}
#[sqlx::test(migrator = "mas_storage_pg::MIGRATOR")]
async fn test_update_no_fields(pool: PgPool) {
setup();
let mut state = TestState::from_pool(pool).await.unwrap();
let token = state.token_with_scope("urn:mas:admin").await;
let mut repo = state.repository().await.unwrap();
// Create a token
let registration_token = repo
.user_registration_token()
.add(
&mut state.rng(),
&state.clock,
"test_update_none".to_owned(),
Some(5),
Some(state.clock.now() + Duration::days(30)),
)
.await
.unwrap();
repo.save().await.unwrap();
// Send empty update
let request = Request::put(format!(
"/api/admin/v1/user-registration-tokens/{}",
registration_token.id
))
.bearer(&token)
.json(json!({}));
let response = state.request(request).await;
response.assert_status(StatusCode::OK);
let body: serde_json::Value = response.json();
// It shouldn't have updated the token
insta::assert_json_snapshot!(body, @r#"
{
"data": {
"type": "user-registration_token",
"id": "01FSHN9AG0MZAA6S4AF7CTV32E",
"attributes": {
"token": "test_update_none",
"valid": true,
"usage_limit": 5,
"times_used": 0,
"created_at": "2022-01-16T14:40:00Z",
"last_used_at": null,
"expires_at": "2022-02-15T14:40:00Z",
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01FSHN9AG0MZAA6S4AF7CTV32E"
}
}
"#);
}
#[sqlx::test(migrator = "mas_storage_pg::MIGRATOR")]
async fn test_update_unknown_token(pool: PgPool) {
setup();
let mut state = TestState::from_pool(pool).await.unwrap();
let token = state.token_with_scope("urn:mas:admin").await;
// Try to update a non-existent token
let request =
Request::put("/api/admin/v1/user-registration-tokens/01040G2081040G2081040G2081")
.bearer(&token)
.json(json!({
"usage_limit": 5
}));
let response = state.request(request).await;
response.assert_status(StatusCode::NOT_FOUND);
let body: serde_json::Value = response.json();
assert_eq!(
body["errors"][0]["title"],
"Registration token with ID 01040G2081040G2081040G2081 not found"
);
}
}

View File

@@ -0,0 +1,15 @@
{
"db_name": "PostgreSQL",
"query": "\n UPDATE user_registration_tokens\n SET expires_at = $2\n WHERE user_registration_token_id = $1\n ",
"describe": {
"columns": [],
"parameters": {
"Left": [
"Uuid",
"Timestamptz"
]
},
"nullable": []
},
"hash": "21b9e39ffd89de288305765c339a991d2471667cf2981770447cde6fd025fbb7"
}

View File

@@ -0,0 +1,14 @@
{
"db_name": "PostgreSQL",
"query": "\n UPDATE user_registration_tokens\n SET revoked_at = NULL\n WHERE user_registration_token_id = $1\n ",
"describe": {
"columns": [],
"parameters": {
"Left": [
"Uuid"
]
},
"nullable": []
},
"hash": "3c7fc3e386ce51187f6344ad65e1d78a7f026e8311bdc7d5ccc2f39d962e898f"
}

View File

@@ -0,0 +1,15 @@
{
"db_name": "PostgreSQL",
"query": "\n UPDATE user_registration_tokens\n SET usage_limit = $2\n WHERE user_registration_token_id = $1\n ",
"describe": {
"columns": [],
"parameters": {
"Left": [
"Uuid",
"Int4"
]
},
"nullable": []
},
"hash": "7e414c29745cf5c85fa4e7cb5d661b07f43ab168956470d120166ed7eab631d9"
}

View File

@@ -546,6 +546,111 @@ impl UserRegistrationTokenRepository for PgUserRegistrationTokenRepository<'_> {
Ok(token)
}
#[tracing::instrument(
name = "db.user_registration_token.unrevoke",
skip_all,
fields(
db.query.text,
user_registration_token.id = %token.id,
),
err,
)]
async fn unrevoke(
&mut self,
mut token: UserRegistrationToken,
) -> Result<UserRegistrationToken, Self::Error> {
let res = sqlx::query!(
r#"
UPDATE user_registration_tokens
SET revoked_at = NULL
WHERE user_registration_token_id = $1
"#,
Uuid::from(token.id),
)
.traced()
.execute(&mut *self.conn)
.await?;
DatabaseError::ensure_affected_rows(&res, 1)?;
token.revoked_at = None;
Ok(token)
}
#[tracing::instrument(
name = "db.user_registration_token.set_expiry",
skip_all,
fields(
db.query.text,
user_registration_token.id = %token.id,
),
err,
)]
async fn set_expiry(
&mut self,
mut token: UserRegistrationToken,
expires_at: Option<DateTime<Utc>>,
) -> Result<UserRegistrationToken, Self::Error> {
let res = sqlx::query!(
r#"
UPDATE user_registration_tokens
SET expires_at = $2
WHERE user_registration_token_id = $1
"#,
Uuid::from(token.id),
expires_at,
)
.traced()
.execute(&mut *self.conn)
.await?;
DatabaseError::ensure_affected_rows(&res, 1)?;
token.expires_at = expires_at;
Ok(token)
}
#[tracing::instrument(
name = "db.user_registration_token.set_usage_limit",
skip_all,
fields(
db.query.text,
user_registration_token.id = %token.id,
),
err,
)]
async fn set_usage_limit(
&mut self,
mut token: UserRegistrationToken,
usage_limit: Option<u32>,
) -> Result<UserRegistrationToken, Self::Error> {
let usage_limit_i32 = usage_limit
.map(i32::try_from)
.transpose()
.map_err(DatabaseError::to_invalid_operation)?;
let res = sqlx::query!(
r#"
UPDATE user_registration_tokens
SET usage_limit = $2
WHERE user_registration_token_id = $1
"#,
Uuid::from(token.id),
usage_limit_i32,
)
.traced()
.execute(&mut *self.conn)
.await?;
DatabaseError::ensure_affected_rows(&res, 1)?;
token.usage_limit = usage_limit;
Ok(token)
}
}
#[cfg(test)]
@@ -560,6 +665,138 @@ mod tests {
use crate::PgRepository;
#[sqlx::test(migrator = "crate::MIGRATOR")]
async fn test_unrevoke(pool: PgPool) {
let mut rng = ChaChaRng::seed_from_u64(42);
let clock = MockClock::default();
let mut repo = PgRepository::from_pool(&pool).await.unwrap().boxed();
// Create a token
let token = repo
.user_registration_token()
.add(&mut rng, &clock, "test_token".to_owned(), None, None)
.await
.unwrap();
// Revoke the token
let revoked_token = repo
.user_registration_token()
.revoke(&clock, token)
.await
.unwrap();
// Verify it's revoked
assert!(revoked_token.revoked_at.is_some());
// Unrevoke the token
let unrevoked_token = repo
.user_registration_token()
.unrevoke(revoked_token)
.await
.unwrap();
// Verify it's no longer revoked
assert!(unrevoked_token.revoked_at.is_none());
// Check that we can find it with the non-revoked filter
let non_revoked_filter = UserRegistrationTokenFilter::new(clock.now()).with_revoked(false);
let page = repo
.user_registration_token()
.list(non_revoked_filter, Pagination::first(10))
.await
.unwrap();
assert!(page.edges.iter().any(|t| t.id == unrevoked_token.id));
}
#[sqlx::test(migrator = "crate::MIGRATOR")]
async fn test_set_expiry(pool: PgPool) {
let mut rng = ChaChaRng::seed_from_u64(42);
let clock = MockClock::default();
let mut repo = PgRepository::from_pool(&pool).await.unwrap().boxed();
// Create a token without expiry
let token = repo
.user_registration_token()
.add(&mut rng, &clock, "test_token_expiry".to_owned(), None, None)
.await
.unwrap();
// Verify it has no expiration
assert!(token.expires_at.is_none());
// Set an expiration
let future_time = clock.now() + Duration::days(30);
let updated_token = repo
.user_registration_token()
.set_expiry(token, Some(future_time))
.await
.unwrap();
// Verify expiration is set
assert_eq!(updated_token.expires_at, Some(future_time));
// Remove the expiration
let final_token = repo
.user_registration_token()
.set_expiry(updated_token, None)
.await
.unwrap();
// Verify expiration is removed
assert!(final_token.expires_at.is_none());
}
#[sqlx::test(migrator = "crate::MIGRATOR")]
async fn test_set_usage_limit(pool: PgPool) {
let mut rng = ChaChaRng::seed_from_u64(42);
let clock = MockClock::default();
let mut repo = PgRepository::from_pool(&pool).await.unwrap().boxed();
// Create a token without usage limit
let token = repo
.user_registration_token()
.add(&mut rng, &clock, "test_token_limit".to_owned(), None, None)
.await
.unwrap();
// Verify it has no usage limit
assert!(token.usage_limit.is_none());
// Set a usage limit
let updated_token = repo
.user_registration_token()
.set_usage_limit(token, Some(5))
.await
.unwrap();
// Verify usage limit is set
assert_eq!(updated_token.usage_limit, Some(5));
// Change the usage limit
let changed_token = repo
.user_registration_token()
.set_usage_limit(updated_token, Some(10))
.await
.unwrap();
// Verify usage limit is changed
assert_eq!(changed_token.usage_limit, Some(10));
// Remove the usage limit
let final_token = repo
.user_registration_token()
.set_usage_limit(changed_token, None)
.await
.unwrap();
// Verify usage limit is removed
assert!(final_token.usage_limit.is_none());
}
#[sqlx::test(migrator = "crate::MIGRATOR")]
async fn test_list_and_count(pool: PgPool) {
let mut rng = ChaChaRng::seed_from_u64(42);

View File

@@ -196,6 +196,53 @@ pub trait UserRegistrationTokenRepository: Send + Sync {
token: UserRegistrationToken,
) -> Result<UserRegistrationToken, Self::Error>;
/// Unrevoke a previously revoked [`UserRegistrationToken`]
///
/// # Parameters
///
/// * `token`: The [`UserRegistrationToken`] to unrevoke
///
/// # Errors
///
/// Returns [`Self::Error`] if the underlying repository fails
async fn unrevoke(
&mut self,
token: UserRegistrationToken,
) -> Result<UserRegistrationToken, Self::Error>;
/// Set the expiration time of a [`UserRegistrationToken`]
///
/// # Parameters
///
/// * `token`: The [`UserRegistrationToken`] to update
/// * `expires_at`: The new expiration time, or `None` to remove the
/// expiration
///
/// # Errors
///
/// Returns [`Self::Error`] if the underlying repository fails
async fn set_expiry(
&mut self,
token: UserRegistrationToken,
expires_at: Option<DateTime<Utc>>,
) -> Result<UserRegistrationToken, Self::Error>;
/// Set the usage limit of a [`UserRegistrationToken`]
///
/// # Parameters
///
/// * `token`: The [`UserRegistrationToken`] to update
/// * `usage_limit`: The new usage limit, or `None` to remove the limit
///
/// # Errors
///
/// Returns [`Self::Error`] if the underlying repository fails
async fn set_usage_limit(
&mut self,
token: UserRegistrationToken,
usage_limit: Option<u32>,
) -> Result<UserRegistrationToken, Self::Error>;
/// List [`UserRegistrationToken`]s based on the provided filter
///
/// Returns a list of matching [`UserRegistrationToken`]s
@@ -249,6 +296,20 @@ repository_impl!(UserRegistrationTokenRepository:
clock: &dyn Clock,
token: UserRegistrationToken,
) -> Result<UserRegistrationToken, Self::Error>;
async fn unrevoke(
&mut self,
token: UserRegistrationToken,
) -> Result<UserRegistrationToken, Self::Error>;
async fn set_expiry(
&mut self,
token: UserRegistrationToken,
expires_at: Option<DateTime<Utc>>,
) -> Result<UserRegistrationToken, Self::Error>;
async fn set_usage_limit(
&mut self,
token: UserRegistrationToken,
usage_limit: Option<u32>,
) -> Result<UserRegistrationToken, Self::Error>;
async fn list(
&mut self,
filter: UserRegistrationTokenFilter,

View File

@@ -2415,6 +2415,87 @@
}
}
}
},
"put": {
"tags": [
"user-registration-token"
],
"summary": "Update a user registration token",
"description": "Update properties of a user registration token such as expiration and usage limit. To set a field to null (removing the limit/expiration), include the field with a null value. To leave a field unchanged, omit it from the request body.",
"operationId": "updateUserRegistrationToken",
"parameters": [
{
"in": "path",
"name": "id",
"required": true,
"schema": {
"title": "The ID of the resource",
"$ref": "#/components/schemas/ULID"
},
"style": "simple"
}
],
"requestBody": {
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/EditUserRegistrationTokenRequest"
}
}
},
"required": true
},
"responses": {
"200": {
"description": "Registration token was updated",
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/SingleResponse_for_UserRegistrationToken"
},
"example": {
"data": {
"type": "user-registration_token",
"id": "01040G2081040G2081040G2081",
"attributes": {
"token": "abc123def456",
"valid": true,
"usage_limit": 10,
"times_used": 5,
"created_at": "1970-01-01T00:00:00Z",
"last_used_at": "1970-01-01T00:00:00Z",
"expires_at": "1970-01-31T00:00:00Z",
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01040G2081040G2081040G2081"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01040G2081040G2081040G2081"
}
}
}
}
},
"404": {
"description": "Registration token was not found",
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/ErrorResponse"
},
"example": {
"errors": [
{
"title": "Registration token with ID 00000000000000000000000000 not found"
}
]
}
}
}
}
}
}
},
"/api/admin/v1/user-registration-tokens/{id}/revoke": {
@@ -2507,6 +2588,96 @@
}
}
},
"/api/admin/v1/user-registration-tokens/{id}/unrevoke": {
"post": {
"tags": [
"user-registration-token"
],
"summary": "Unrevoke a user registration token",
"description": "Calling this endpoint will unrevoke a previously revoked user registration token, allowing it to be used for registrations again (subject to its usage limits and expiration).",
"operationId": "unrevokeUserRegistrationToken",
"parameters": [
{
"in": "path",
"name": "id",
"required": true,
"schema": {
"title": "The ID of the resource",
"$ref": "#/components/schemas/ULID"
},
"style": "simple"
}
],
"responses": {
"200": {
"description": "Registration token was unrevoked",
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/SingleResponse_for_UserRegistrationToken"
},
"example": {
"data": {
"type": "user-registration_token",
"id": "01040G2081040G2081040G2081",
"attributes": {
"token": "abc123def456",
"valid": true,
"usage_limit": 10,
"times_used": 5,
"created_at": "1970-01-01T00:00:00Z",
"last_used_at": "1970-01-01T00:00:00Z",
"expires_at": "1970-01-31T00:00:00Z",
"revoked_at": null
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01040G2081040G2081040G2081"
}
},
"links": {
"self": "/api/admin/v1/user-registration-tokens/01040G2081040G2081040G2081/unrevoke"
}
}
}
}
},
"400": {
"description": "Token is not revoked",
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/ErrorResponse"
},
"example": {
"errors": [
{
"title": "Registration token with ID 00000000000000000000000000 is not revoked"
}
]
}
}
}
},
"404": {
"description": "Registration token was not found",
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/ErrorResponse"
},
"example": {
"errors": [
{
"title": "Registration token with ID 00000000000000000000000000 not found"
}
]
}
}
}
}
}
}
},
"/api/admin/v1/upstream-oauth-links": {
"get": {
"tags": [
@@ -4138,6 +4309,25 @@
}
}
},
"EditUserRegistrationTokenRequest": {
"title": "JSON payload for the `PUT /api/admin/v1/user-registration-tokens/{id}` endpoint",
"type": "object",
"properties": {
"expires_at": {
"description": "New expiration date for the token, or null to remove expiration",
"type": "string",
"format": "date-time",
"nullable": true
},
"usage_limit": {
"description": "New usage limit for the token, or null to remove the limit",
"type": "integer",
"format": "uint32",
"minimum": 0.0,
"nullable": true
}
}
},
"UpstreamOAuthLinkFilter": {
"type": "object",
"properties": {