gitdataai/libs/api/workspace/members.rs
2026-04-15 09:08:09 +08:00

206 lines
6.4 KiB
Rust

use crate::{ApiResponse, error::ApiError};
use actix_web::{HttpResponse, Result, web};
use service::AppService;
use service::workspace::members::{
PendingInvitationInfo, WorkspaceInviteAcceptParams, WorkspaceInviteParams,
WorkspaceMembersResponse,
};
use session::Session;
use uuid::Uuid;
#[derive(serde::Deserialize, utoipa::IntoParams)]
pub struct MembersQuery {
pub page: Option<u64>,
pub per_page: Option<u64>,
}
#[utoipa::path(
get,
path = "/api/workspaces/{slug}/members",
params(
("slug" = String, Path),
("page" = Option<u64>, Query),
("per_page" = Option<u64>, Query),
),
responses(
(status = 200, description = "List workspace members", body = ApiResponse<WorkspaceMembersResponse>),
(status = 401, description = "Unauthorized"),
(status = 403, description = "Not a member"),
(status = 404, description = "Workspace not found"),
),
tag = "Workspace"
)]
pub async fn workspace_members(
service: web::Data<AppService>,
session: Session,
path: web::Path<String>,
query: web::Query<MembersQuery>,
) -> Result<HttpResponse, ApiError> {
let slug = path.into_inner();
let resp = service
.workspace_members(&session, slug, query.page, query.per_page)
.await?;
Ok(ApiResponse::ok(resp).to_response())
}
#[derive(serde::Deserialize, utoipa::ToSchema)]
pub struct UpdateRoleParams {
pub user_id: Uuid,
pub role: String,
}
#[utoipa::path(
patch,
path = "/api/workspaces/{slug}/members/role",
params(("slug" = String, Path)),
request_body = UpdateRoleParams,
responses(
(status = 200, description = "Update member role"),
(status = 401, description = "Unauthorized"),
(status = 403, description = "Permission denied"),
(status = 404, description = "Workspace or member not found"),
),
tag = "Workspace"
)]
pub async fn workspace_update_member_role(
service: web::Data<AppService>,
session: Session,
path: web::Path<String>,
body: web::Json<UpdateRoleParams>,
) -> Result<HttpResponse, ApiError> {
let slug = path.into_inner();
service
.workspace_update_member_role(&session, slug, body.user_id, body.role.clone())
.await?;
Ok(ApiResponse::ok(serde_json::json!({ "success": true })).to_response())
}
#[utoipa::path(
delete,
path = "/api/workspaces/{slug}/members/{user_id}",
params(
("slug" = String, Path),
("user_id" = Uuid, Path),
),
responses(
(status = 200, description = "Remove member"),
(status = 401, description = "Unauthorized"),
(status = 403, description = "Permission denied"),
(status = 404, description = "Member not found"),
),
tag = "Workspace"
)]
pub async fn workspace_remove_member(
service: web::Data<AppService>,
session: Session,
path: web::Path<(String, Uuid)>,
) -> Result<HttpResponse, ApiError> {
let (slug, user_id) = path.into_inner();
service
.workspace_remove_member(&session, slug, user_id)
.await?;
Ok(ApiResponse::ok(serde_json::json!({ "success": true })).to_response())
}
#[utoipa::path(
get,
path = "/api/workspaces/{slug}/invitations",
params(("slug" = String, Path)),
responses(
(status = 200, description = "List pending invitations", body = ApiResponse<Vec<PendingInvitationInfo>>),
(status = 401, description = "Unauthorized"),
(status = 403, description = "Permission denied"),
(status = 404, description = "Workspace not found"),
),
tag = "Workspace"
)]
pub async fn workspace_pending_invitations(
service: web::Data<AppService>,
session: Session,
path: web::Path<String>,
) -> Result<HttpResponse, ApiError> {
let slug = path.into_inner();
let resp = service
.workspace_pending_invitations(&session, slug)
.await?;
Ok(ApiResponse::ok(resp).to_response())
}
#[utoipa::path(
delete,
path = "/api/workspaces/{slug}/invitations/{user_id}",
params(
("slug" = String, Path),
("user_id" = Uuid, Path),
),
responses(
(status = 200, description = "Cancel invitation"),
(status = 401, description = "Unauthorized"),
(status = 403, description = "Permission denied"),
(status = 404, description = "Invitation not found"),
),
tag = "Workspace"
)]
pub async fn workspace_cancel_invitation(
service: web::Data<AppService>,
session: Session,
path: web::Path<(String, Uuid)>,
) -> Result<HttpResponse, ApiError> {
let (slug, user_id) = path.into_inner();
service
.workspace_cancel_invitation(&session, slug, user_id)
.await?;
Ok(ApiResponse::ok(serde_json::json!({ "success": true })).to_response())
}
#[utoipa::path(
post,
path = "/api/workspaces/{slug}/invitations",
params(("slug" = String, Path)),
request_body = WorkspaceInviteParams,
responses(
(status = 200, description = "Send invitation"),
(status = 401, description = "Unauthorized"),
(status = 403, description = "Permission denied"),
(status = 404, description = "User not found"),
(status = 409, description = "Already a member"),
),
tag = "Workspace"
)]
pub async fn workspace_invite_member(
service: web::Data<AppService>,
session: Session,
path: web::Path<String>,
body: web::Json<WorkspaceInviteParams>,
) -> Result<HttpResponse, ApiError> {
let slug = path.into_inner();
service
.workspace_invite_member(&session, slug, body.into_inner())
.await?;
Ok(ApiResponse::ok(serde_json::json!({ "success": true })).to_response())
}
#[utoipa::path(
post,
path = "/api/workspaces/invitations/accept",
request_body = WorkspaceInviteAcceptParams,
responses(
(status = 200, description = "Accept invitation", body = ApiResponse<service::workspace::info::WorkspaceInfoResponse>),
(status = 400, description = "Invalid or expired token"),
(status = 401, description = "Unauthorized"),
(status = 409, description = "Already accepted"),
),
tag = "Workspace"
)]
pub async fn workspace_accept_invitation(
service: web::Data<AppService>,
session: Session,
body: web::Json<WorkspaceInviteAcceptParams>,
) -> Result<HttpResponse, ApiError> {
let ws = service
.workspace_accept_invitation(&session, body.into_inner())
.await?;
let resp = service.workspace_info(&session, ws.slug).await?;
Ok(ApiResponse::ok(resp).to_response())
}