1373 lines
49 KiB
Rust
1373 lines
49 KiB
Rust
use axum::{
|
|
Json,
|
|
extract::{Extension, Path, State},
|
|
http::StatusCode,
|
|
response::Response,
|
|
};
|
|
use module_runtime::{
|
|
PROFILE_RECHARGE_PAYMENT_CHANNEL_MOCK, RuntimeProfileInviteCodeRecord,
|
|
RuntimeProfileMembershipBenefitRecord, RuntimeProfileRechargeCenterRecord,
|
|
RuntimeProfileRechargeOrderRecord, RuntimeProfileRechargeProductRecord,
|
|
RuntimeProfileRedeemCodeMode, RuntimeProfileRedeemCodeRecord,
|
|
RuntimeProfileRewardCodeRedeemRecord, RuntimeProfileTaskCenterRecord,
|
|
RuntimeProfileTaskClaimRecord, RuntimeProfileTaskConfigRecord, RuntimeProfileTaskCycle,
|
|
RuntimeProfileTaskItemRecord, RuntimeProfileTaskStatus, RuntimeProfileWalletLedgerSourceType,
|
|
RuntimeReferralInviteCenterRecord, RuntimeTrackingScopeKind,
|
|
};
|
|
use serde_json::{Value, json};
|
|
use shared_contracts::runtime::{
|
|
AdminDisableProfileRedeemCodeRequest, AdminDisableProfileTaskConfigRequest,
|
|
AdminUpsertProfileInviteCodeRequest, AdminUpsertProfileRedeemCodeRequest,
|
|
AdminUpsertProfileTaskConfigRequest, ClaimProfileTaskRewardResponse,
|
|
CreateProfileRechargeOrderRequest, CreateProfileRechargeOrderResponse,
|
|
PROFILE_TASK_CYCLE_DAILY, PROFILE_TASK_STATUS_CLAIMABLE, PROFILE_TASK_STATUS_CLAIMED,
|
|
PROFILE_TASK_STATUS_DISABLED, PROFILE_TASK_STATUS_INCOMPLETE,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_ASSET_OPERATION_CONSUME,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_ASSET_OPERATION_REFUND,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_DAILY_TASK_REWARD,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_INVITE_INVITEE_REWARD,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_INVITE_INVITER_REWARD,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_NEW_USER_REGISTRATION_REWARD,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_POINTS_RECHARGE,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_PUZZLE_AUTHOR_INCENTIVE_CLAIM,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_REDEEM_CODE_REWARD,
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_SNAPSHOT_SYNC, ProfileDashboardSummaryResponse,
|
|
ProfileInviteCodeAdminListResponse, ProfileInviteCodeAdminResponse,
|
|
ProfileMembershipBenefitResponse, ProfileMembershipResponse, ProfilePlayStatsResponse,
|
|
ProfilePlayedWorkSummaryResponse, ProfileRechargeCenterResponse, ProfileRechargeOrderResponse,
|
|
ProfileRechargeProductResponse, ProfileRedeemCodeAdminListResponse,
|
|
ProfileRedeemCodeAdminResponse, ProfileReferralInviteCenterResponse,
|
|
ProfileReferralInvitedUserResponse, ProfileTaskCenterResponse,
|
|
ProfileTaskConfigAdminListResponse, ProfileTaskConfigAdminResponse, ProfileTaskItemResponse,
|
|
ProfileWalletLedgerEntryResponse, ProfileWalletLedgerResponse,
|
|
RedeemProfileReferralInviteCodeRequest, RedeemProfileReferralInviteCodeResponse,
|
|
RedeemProfileRewardCodeRequest, RedeemProfileRewardCodeResponse, TRACKING_SCOPE_KIND_MODULE,
|
|
TRACKING_SCOPE_KIND_SITE, TRACKING_SCOPE_KIND_USER, TRACKING_SCOPE_KIND_WORK,
|
|
};
|
|
use spacetime_client::SpacetimeClientError;
|
|
use time::OffsetDateTime;
|
|
|
|
use crate::{
|
|
admin::AuthenticatedAdmin, api_response::json_success_body, auth::AuthenticatedAccessToken,
|
|
http_error::AppError, request_context::RequestContext, state::AppState,
|
|
};
|
|
|
|
pub async fn get_profile_dashboard(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let record = state
|
|
.spacetime_client()
|
|
.get_profile_dashboard(user_id)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
ProfileDashboardSummaryResponse {
|
|
wallet_balance: record.wallet_balance,
|
|
total_play_time_ms: record.total_play_time_ms,
|
|
played_world_count: record.played_world_count,
|
|
updated_at: record.updated_at,
|
|
},
|
|
))
|
|
}
|
|
|
|
pub async fn get_profile_wallet_ledger(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let entries = state
|
|
.spacetime_client()
|
|
.list_profile_wallet_ledger(user_id)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
ProfileWalletLedgerResponse {
|
|
entries: entries
|
|
.into_iter()
|
|
.map(build_profile_wallet_ledger_entry_response)
|
|
.collect(),
|
|
},
|
|
))
|
|
}
|
|
|
|
fn format_profile_wallet_ledger_source_type(
|
|
source_type: RuntimeProfileWalletLedgerSourceType,
|
|
) -> &'static str {
|
|
match source_type {
|
|
RuntimeProfileWalletLedgerSourceType::SnapshotSync => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_SNAPSHOT_SYNC
|
|
}
|
|
RuntimeProfileWalletLedgerSourceType::NewUserRegistrationReward => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_NEW_USER_REGISTRATION_REWARD
|
|
}
|
|
RuntimeProfileWalletLedgerSourceType::InviteInviterReward => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_INVITE_INVITER_REWARD
|
|
}
|
|
RuntimeProfileWalletLedgerSourceType::InviteInviteeReward => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_INVITE_INVITEE_REWARD
|
|
}
|
|
RuntimeProfileWalletLedgerSourceType::PointsRecharge => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_POINTS_RECHARGE
|
|
}
|
|
RuntimeProfileWalletLedgerSourceType::AssetOperationConsume => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_ASSET_OPERATION_CONSUME
|
|
}
|
|
RuntimeProfileWalletLedgerSourceType::AssetOperationRefund => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_ASSET_OPERATION_REFUND
|
|
}
|
|
RuntimeProfileWalletLedgerSourceType::RedeemCodeReward => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_REDEEM_CODE_REWARD
|
|
}
|
|
RuntimeProfileWalletLedgerSourceType::PuzzleAuthorIncentiveClaim => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_PUZZLE_AUTHOR_INCENTIVE_CLAIM
|
|
}
|
|
RuntimeProfileWalletLedgerSourceType::DailyTaskReward => {
|
|
PROFILE_WALLET_LEDGER_SOURCE_TYPE_DAILY_TASK_REWARD
|
|
}
|
|
}
|
|
}
|
|
|
|
pub async fn get_profile_recharge_center(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let record = state
|
|
.spacetime_client()
|
|
.get_profile_recharge_center(user_id)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_profile_recharge_center_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn create_profile_recharge_order(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
Json(payload): Json<CreateProfileRechargeOrderRequest>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let payment_channel = payload
|
|
.payment_channel
|
|
.unwrap_or_else(|| PROFILE_RECHARGE_PAYMENT_CHANNEL_MOCK.to_string());
|
|
let created_at_micros = OffsetDateTime::now_utc().unix_timestamp_nanos() / 1_000;
|
|
let (center, order) = state
|
|
.spacetime_client()
|
|
.create_profile_recharge_order(
|
|
user_id,
|
|
payload.product_id,
|
|
payment_channel,
|
|
created_at_micros as i64,
|
|
)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
CreateProfileRechargeOrderResponse {
|
|
order: build_profile_recharge_order_response(order),
|
|
center: build_profile_recharge_center_response(center),
|
|
},
|
|
))
|
|
}
|
|
|
|
pub async fn get_profile_referral_invite_center(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let record = state
|
|
.spacetime_client()
|
|
.get_profile_referral_invite_center(user_id)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_profile_referral_invite_center_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn redeem_profile_referral_invite_code(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
Json(payload): Json<RedeemProfileReferralInviteCodeRequest>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let updated_at_micros = OffsetDateTime::now_utc().unix_timestamp_nanos() / 1_000;
|
|
let record = state
|
|
.spacetime_client()
|
|
.redeem_profile_referral_invite_code(user_id, payload.invite_code, updated_at_micros as i64)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_redeem_profile_referral_invite_code_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn redeem_profile_reward_code(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
Json(payload): Json<RedeemProfileRewardCodeRequest>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let redeemed_at_micros = OffsetDateTime::now_utc().unix_timestamp_nanos() / 1_000;
|
|
let record = state
|
|
.spacetime_client()
|
|
.redeem_profile_reward_code(user_id, payload.code, redeemed_at_micros as i64)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_redeem_profile_reward_code_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn get_profile_task_center(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let record = state
|
|
.spacetime_client()
|
|
.get_profile_task_center(user_id)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_profile_task_center_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn claim_profile_task_reward(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
Path(task_id): Path<String>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let record = state
|
|
.spacetime_client()
|
|
.claim_profile_task_reward(user_id, task_id)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_claim_profile_task_reward_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn admin_list_profile_task_configs(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(admin): Extension<AuthenticatedAdmin>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let entries = state
|
|
.spacetime_client()
|
|
.admin_list_profile_task_configs(admin.session().subject.clone())
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
ProfileTaskConfigAdminListResponse {
|
|
entries: entries
|
|
.into_iter()
|
|
.map(build_profile_task_config_admin_response)
|
|
.collect(),
|
|
},
|
|
))
|
|
}
|
|
|
|
pub async fn admin_upsert_profile_task_config(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(admin): Extension<AuthenticatedAdmin>,
|
|
Json(payload): Json<AdminUpsertProfileTaskConfigRequest>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let cycle = parse_profile_task_cycle(&payload.cycle).map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
AppError::from_status(StatusCode::BAD_REQUEST).with_message(error),
|
|
)
|
|
})?;
|
|
let scope_kind = parse_tracking_scope_kind(&payload.scope_kind).map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
AppError::from_status(StatusCode::BAD_REQUEST).with_message(error),
|
|
)
|
|
})?;
|
|
let updated_at_micros = OffsetDateTime::now_utc().unix_timestamp_nanos() / 1_000;
|
|
let record = state
|
|
.spacetime_client()
|
|
.admin_upsert_profile_task_config(
|
|
admin.session().subject.clone(),
|
|
payload.task_id,
|
|
payload.title,
|
|
payload.description.unwrap_or_default(),
|
|
payload.event_key,
|
|
cycle,
|
|
scope_kind,
|
|
payload.threshold,
|
|
payload.reward_points,
|
|
payload.enabled,
|
|
payload.sort_order.unwrap_or(10),
|
|
updated_at_micros as i64,
|
|
)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_profile_task_config_admin_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn admin_disable_profile_task_config(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(admin): Extension<AuthenticatedAdmin>,
|
|
Json(payload): Json<AdminDisableProfileTaskConfigRequest>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let updated_at_micros = OffsetDateTime::now_utc().unix_timestamp_nanos() / 1_000;
|
|
let record = state
|
|
.spacetime_client()
|
|
.admin_disable_profile_task_config(
|
|
admin.session().subject.clone(),
|
|
payload.task_id,
|
|
updated_at_micros as i64,
|
|
)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_profile_task_config_admin_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn admin_list_profile_redeem_codes(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(admin): Extension<AuthenticatedAdmin>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let entries = state
|
|
.spacetime_client()
|
|
.admin_list_profile_redeem_codes(admin.session().subject.clone())
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
ProfileRedeemCodeAdminListResponse {
|
|
entries: entries
|
|
.into_iter()
|
|
.map(build_profile_redeem_code_admin_response)
|
|
.collect(),
|
|
},
|
|
))
|
|
}
|
|
|
|
pub async fn admin_upsert_profile_redeem_code(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(admin): Extension<AuthenticatedAdmin>,
|
|
Json(payload): Json<AdminUpsertProfileRedeemCodeRequest>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let updated_at_micros = OffsetDateTime::now_utc().unix_timestamp_nanos() / 1_000;
|
|
let mode = parse_profile_redeem_code_mode(&payload.mode).map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
AppError::from_status(StatusCode::BAD_REQUEST).with_message(error),
|
|
)
|
|
})?;
|
|
let record = state
|
|
.spacetime_client()
|
|
.admin_upsert_profile_redeem_code(
|
|
admin.session().subject.clone(),
|
|
payload.code,
|
|
mode,
|
|
payload.reward_points,
|
|
payload.max_uses,
|
|
payload.enabled,
|
|
payload.allowed_user_ids,
|
|
payload.allowed_public_user_codes,
|
|
updated_at_micros as i64,
|
|
)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_profile_redeem_code_admin_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn admin_disable_profile_redeem_code(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(admin): Extension<AuthenticatedAdmin>,
|
|
Json(payload): Json<AdminDisableProfileRedeemCodeRequest>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let updated_at_micros = OffsetDateTime::now_utc().unix_timestamp_nanos() / 1_000;
|
|
let record = state
|
|
.spacetime_client()
|
|
.admin_disable_profile_redeem_code(
|
|
admin.session().subject.clone(),
|
|
payload.code,
|
|
updated_at_micros as i64,
|
|
)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_profile_redeem_code_admin_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn admin_list_profile_invite_codes(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(admin): Extension<AuthenticatedAdmin>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let entries = state
|
|
.spacetime_client()
|
|
.admin_list_profile_invite_codes(admin.session().subject.clone())
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
ProfileInviteCodeAdminListResponse {
|
|
entries: entries
|
|
.into_iter()
|
|
.map(build_profile_invite_code_admin_response)
|
|
.collect(),
|
|
},
|
|
))
|
|
}
|
|
|
|
pub async fn admin_upsert_profile_invite_code(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(admin): Extension<AuthenticatedAdmin>,
|
|
Json(payload): Json<AdminUpsertProfileInviteCodeRequest>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let metadata_json = normalize_admin_invite_code_metadata(payload.metadata)
|
|
.map_err(|error| runtime_profile_error_response(&request_context, error))?;
|
|
let updated_at_micros = OffsetDateTime::now_utc().unix_timestamp_nanos() / 1_000;
|
|
let record = state
|
|
.spacetime_client()
|
|
.admin_upsert_profile_invite_code(
|
|
admin.session().username.clone(),
|
|
payload.invite_code,
|
|
metadata_json,
|
|
updated_at_micros as i64,
|
|
)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
build_profile_invite_code_admin_response(record),
|
|
))
|
|
}
|
|
|
|
pub async fn get_profile_play_stats(
|
|
State(state): State<AppState>,
|
|
Extension(request_context): Extension<RequestContext>,
|
|
Extension(authenticated): Extension<AuthenticatedAccessToken>,
|
|
) -> Result<Json<Value>, Response> {
|
|
let user_id = authenticated.claims().user_id().to_string();
|
|
let record = state
|
|
.spacetime_client()
|
|
.get_profile_play_stats(user_id)
|
|
.await
|
|
.map_err(|error| {
|
|
runtime_profile_error_response(
|
|
&request_context,
|
|
map_runtime_profile_client_error(error),
|
|
)
|
|
})?;
|
|
|
|
Ok(json_success_body(
|
|
Some(&request_context),
|
|
ProfilePlayStatsResponse {
|
|
total_play_time_ms: record.total_play_time_ms,
|
|
played_works: record
|
|
.played_works
|
|
.into_iter()
|
|
.map(|entry| ProfilePlayedWorkSummaryResponse {
|
|
world_key: entry.world_key,
|
|
owner_user_id: entry.owner_user_id,
|
|
profile_id: entry.profile_id,
|
|
world_type: entry.world_type,
|
|
world_title: entry.world_title,
|
|
world_subtitle: entry.world_subtitle,
|
|
first_played_at: entry.first_played_at,
|
|
last_played_at: entry.last_played_at,
|
|
last_observed_play_time_ms: entry.last_observed_play_time_ms,
|
|
})
|
|
.collect(),
|
|
updated_at: record.updated_at,
|
|
},
|
|
))
|
|
}
|
|
|
|
fn map_runtime_profile_client_error(error: SpacetimeClientError) -> AppError {
|
|
let (status, provider) = match error {
|
|
SpacetimeClientError::Runtime(_) => (StatusCode::BAD_REQUEST, "runtime-profile"),
|
|
_ => (StatusCode::BAD_GATEWAY, "spacetimedb"),
|
|
};
|
|
|
|
AppError::from_status(status).with_details(json!({
|
|
"provider": provider,
|
|
"message": error.to_string(),
|
|
}))
|
|
}
|
|
|
|
fn runtime_profile_error_response(request_context: &RequestContext, error: AppError) -> Response {
|
|
error.into_response_with_context(Some(request_context))
|
|
}
|
|
|
|
fn build_profile_recharge_center_response(
|
|
record: RuntimeProfileRechargeCenterRecord,
|
|
) -> ProfileRechargeCenterResponse {
|
|
ProfileRechargeCenterResponse {
|
|
wallet_balance: record.wallet_balance,
|
|
membership: ProfileMembershipResponse {
|
|
status: record.membership.status.as_str().to_string(),
|
|
tier: record.membership.tier.as_str().to_string(),
|
|
started_at: record.membership.started_at,
|
|
expires_at: record.membership.expires_at,
|
|
updated_at: record.membership.updated_at,
|
|
},
|
|
point_products: record
|
|
.point_products
|
|
.into_iter()
|
|
.map(build_profile_recharge_product_response)
|
|
.collect(),
|
|
membership_products: record
|
|
.membership_products
|
|
.into_iter()
|
|
.map(build_profile_recharge_product_response)
|
|
.collect(),
|
|
benefits: record
|
|
.benefits
|
|
.into_iter()
|
|
.map(build_profile_membership_benefit_response)
|
|
.collect(),
|
|
latest_order: record
|
|
.latest_order
|
|
.map(build_profile_recharge_order_response),
|
|
has_points_recharged: record.has_points_recharged,
|
|
}
|
|
}
|
|
|
|
fn build_profile_recharge_product_response(
|
|
record: RuntimeProfileRechargeProductRecord,
|
|
) -> ProfileRechargeProductResponse {
|
|
ProfileRechargeProductResponse {
|
|
product_id: record.product_id,
|
|
title: record.title,
|
|
price_cents: record.price_cents,
|
|
kind: record.kind.as_str().to_string(),
|
|
points_amount: record.points_amount,
|
|
bonus_points: record.bonus_points,
|
|
duration_days: record.duration_days,
|
|
badge_label: record.badge_label,
|
|
description: record.description,
|
|
tier: record.tier.as_str().to_string(),
|
|
}
|
|
}
|
|
|
|
fn build_profile_membership_benefit_response(
|
|
record: RuntimeProfileMembershipBenefitRecord,
|
|
) -> ProfileMembershipBenefitResponse {
|
|
ProfileMembershipBenefitResponse {
|
|
benefit_name: record.benefit_name,
|
|
normal_value: record.normal_value,
|
|
month_value: record.month_value,
|
|
season_value: record.season_value,
|
|
year_value: record.year_value,
|
|
}
|
|
}
|
|
|
|
fn build_profile_recharge_order_response(
|
|
record: RuntimeProfileRechargeOrderRecord,
|
|
) -> ProfileRechargeOrderResponse {
|
|
ProfileRechargeOrderResponse {
|
|
order_id: record.order_id,
|
|
product_id: record.product_id,
|
|
product_title: record.product_title,
|
|
kind: record.kind.as_str().to_string(),
|
|
amount_cents: record.amount_cents,
|
|
status: record.status.as_str().to_string(),
|
|
payment_channel: record.payment_channel,
|
|
paid_at: record.paid_at,
|
|
created_at: record.created_at,
|
|
points_delta: record.points_delta,
|
|
membership_expires_at: record.membership_expires_at,
|
|
}
|
|
}
|
|
|
|
fn build_profile_referral_invite_center_response(
|
|
record: RuntimeReferralInviteCenterRecord,
|
|
) -> ProfileReferralInviteCenterResponse {
|
|
ProfileReferralInviteCenterResponse {
|
|
invite_code: record.invite_code,
|
|
invite_link_path: record.invite_link_path,
|
|
invited_count: record.invited_count,
|
|
rewarded_invite_count: record.rewarded_invite_count,
|
|
today_inviter_reward_count: record.today_inviter_reward_count,
|
|
today_inviter_reward_remaining: record.today_inviter_reward_remaining,
|
|
reward_points: record.reward_points,
|
|
invited_users: record
|
|
.invited_users
|
|
.into_iter()
|
|
.map(|user| ProfileReferralInvitedUserResponse {
|
|
user_id: user.user_id,
|
|
display_name: user.display_name,
|
|
avatar_url: user.avatar_url,
|
|
bound_at: user.bound_at,
|
|
})
|
|
.collect(),
|
|
has_redeemed_code: record.has_redeemed_code,
|
|
bound_inviter_user_id: record.bound_inviter_user_id,
|
|
bound_at: record.bound_at,
|
|
updated_at: record.updated_at,
|
|
}
|
|
}
|
|
|
|
fn build_redeem_profile_referral_invite_code_response(
|
|
record: module_runtime::RuntimeReferralRedeemRecord,
|
|
) -> RedeemProfileReferralInviteCodeResponse {
|
|
RedeemProfileReferralInviteCodeResponse {
|
|
center: build_profile_referral_invite_center_response(record.center),
|
|
invitee_reward_granted: record.invitee_reward_granted,
|
|
inviter_reward_granted: record.inviter_reward_granted,
|
|
invitee_balance_after: record.invitee_balance_after,
|
|
inviter_balance_after: record.inviter_balance_after,
|
|
}
|
|
}
|
|
|
|
fn build_redeem_profile_reward_code_response(
|
|
record: RuntimeProfileRewardCodeRedeemRecord,
|
|
) -> RedeemProfileRewardCodeResponse {
|
|
RedeemProfileRewardCodeResponse {
|
|
wallet_balance: record.wallet_balance,
|
|
amount_granted: record.amount_granted,
|
|
ledger_entry: build_profile_wallet_ledger_entry_response(record.ledger_entry),
|
|
}
|
|
}
|
|
|
|
fn build_profile_wallet_ledger_entry_response(
|
|
record: module_runtime::RuntimeProfileWalletLedgerEntryRecord,
|
|
) -> ProfileWalletLedgerEntryResponse {
|
|
ProfileWalletLedgerEntryResponse {
|
|
id: record.wallet_ledger_id,
|
|
amount_delta: record.amount_delta,
|
|
balance_after: record.balance_after,
|
|
source_type: format_profile_wallet_ledger_source_type(record.source_type).to_string(),
|
|
created_at: record.created_at,
|
|
}
|
|
}
|
|
|
|
fn build_profile_task_center_response(
|
|
record: RuntimeProfileTaskCenterRecord,
|
|
) -> ProfileTaskCenterResponse {
|
|
ProfileTaskCenterResponse {
|
|
day_key: record.day_key,
|
|
wallet_balance: record.wallet_balance,
|
|
tasks: record
|
|
.tasks
|
|
.into_iter()
|
|
.map(build_profile_task_item_response)
|
|
.collect(),
|
|
updated_at: record.updated_at,
|
|
}
|
|
}
|
|
|
|
fn build_profile_task_item_response(
|
|
record: RuntimeProfileTaskItemRecord,
|
|
) -> ProfileTaskItemResponse {
|
|
ProfileTaskItemResponse {
|
|
task_id: record.task_id,
|
|
title: record.title,
|
|
description: record.description,
|
|
event_key: record.event_key,
|
|
cycle: format_profile_task_cycle(record.cycle).to_string(),
|
|
threshold: record.threshold,
|
|
progress_count: record.progress_count,
|
|
reward_points: record.reward_points,
|
|
status: format_profile_task_status(record.status).to_string(),
|
|
day_key: record.day_key,
|
|
claimed_at: record.claimed_at,
|
|
updated_at: record.updated_at,
|
|
}
|
|
}
|
|
|
|
fn build_claim_profile_task_reward_response(
|
|
record: RuntimeProfileTaskClaimRecord,
|
|
) -> ClaimProfileTaskRewardResponse {
|
|
ClaimProfileTaskRewardResponse {
|
|
task_id: record.task_id,
|
|
day_key: record.day_key,
|
|
reward_points: record.reward_points,
|
|
wallet_balance: record.wallet_balance,
|
|
ledger_entry: build_profile_wallet_ledger_entry_response(record.ledger_entry),
|
|
center: build_profile_task_center_response(record.center),
|
|
}
|
|
}
|
|
|
|
fn build_profile_task_config_admin_response(
|
|
record: RuntimeProfileTaskConfigRecord,
|
|
) -> ProfileTaskConfigAdminResponse {
|
|
ProfileTaskConfigAdminResponse {
|
|
task_id: record.task_id,
|
|
title: record.title,
|
|
description: record.description,
|
|
event_key: record.event_key,
|
|
cycle: format_profile_task_cycle(record.cycle).to_string(),
|
|
scope_kind: format_tracking_scope_kind(record.scope_kind).to_string(),
|
|
threshold: record.threshold,
|
|
reward_points: record.reward_points,
|
|
enabled: record.enabled,
|
|
sort_order: record.sort_order,
|
|
created_by: record.created_by,
|
|
created_at: record.created_at,
|
|
updated_by: record.updated_by,
|
|
updated_at: record.updated_at,
|
|
}
|
|
}
|
|
|
|
fn normalize_admin_invite_code_metadata(metadata: Option<Value>) -> Result<String, AppError> {
|
|
let metadata = match metadata {
|
|
Some(Value::Null) | None => json!({}),
|
|
Some(value) if value.is_object() => value,
|
|
Some(_) => {
|
|
return Err(AppError::from_status(StatusCode::BAD_REQUEST)
|
|
.with_message("邀请码 metadata 必须是 JSON 对象")
|
|
.with_details(json!({ "field": "metadata" })));
|
|
}
|
|
};
|
|
let metadata_json = serde_json::to_string(&metadata).map_err(|error| {
|
|
AppError::from_status(StatusCode::BAD_REQUEST)
|
|
.with_message(format!("邀请码 metadata 序列化失败:{error}"))
|
|
.with_details(json!({ "field": "metadata" }))
|
|
})?;
|
|
if metadata_json.len() > 4096 {
|
|
return Err(AppError::from_status(StatusCode::BAD_REQUEST)
|
|
.with_message("邀请码 metadata 不能超过 4096 bytes")
|
|
.with_details(json!({ "field": "metadata" })));
|
|
}
|
|
|
|
Ok(metadata_json)
|
|
}
|
|
|
|
fn parse_profile_redeem_code_mode(raw: &str) -> Result<RuntimeProfileRedeemCodeMode, String> {
|
|
match raw.trim().to_ascii_lowercase().as_str() {
|
|
"public" => Ok(RuntimeProfileRedeemCodeMode::Public),
|
|
"unique" => Ok(RuntimeProfileRedeemCodeMode::Unique),
|
|
"private" => Ok(RuntimeProfileRedeemCodeMode::Private),
|
|
_ => Err("兑换码类型无效".to_string()),
|
|
}
|
|
}
|
|
|
|
fn parse_profile_task_cycle(raw: &str) -> Result<RuntimeProfileTaskCycle, String> {
|
|
match raw.trim().to_ascii_lowercase().as_str() {
|
|
PROFILE_TASK_CYCLE_DAILY => Ok(RuntimeProfileTaskCycle::Daily),
|
|
_ => Err("任务周期无效".to_string()),
|
|
}
|
|
}
|
|
|
|
fn parse_tracking_scope_kind(raw: &str) -> Result<RuntimeTrackingScopeKind, String> {
|
|
match raw.trim().to_ascii_lowercase().as_str() {
|
|
TRACKING_SCOPE_KIND_SITE => Ok(RuntimeTrackingScopeKind::Site),
|
|
TRACKING_SCOPE_KIND_WORK => Ok(RuntimeTrackingScopeKind::Work),
|
|
TRACKING_SCOPE_KIND_MODULE => Ok(RuntimeTrackingScopeKind::Module),
|
|
TRACKING_SCOPE_KIND_USER => Ok(RuntimeTrackingScopeKind::User),
|
|
_ => Err("埋点范围无效".to_string()),
|
|
}
|
|
}
|
|
|
|
fn format_profile_task_cycle(cycle: RuntimeProfileTaskCycle) -> &'static str {
|
|
match cycle {
|
|
RuntimeProfileTaskCycle::Daily => PROFILE_TASK_CYCLE_DAILY,
|
|
}
|
|
}
|
|
|
|
fn format_profile_task_status(status: RuntimeProfileTaskStatus) -> &'static str {
|
|
match status {
|
|
RuntimeProfileTaskStatus::Incomplete => PROFILE_TASK_STATUS_INCOMPLETE,
|
|
RuntimeProfileTaskStatus::Claimable => PROFILE_TASK_STATUS_CLAIMABLE,
|
|
RuntimeProfileTaskStatus::Claimed => PROFILE_TASK_STATUS_CLAIMED,
|
|
RuntimeProfileTaskStatus::Disabled => PROFILE_TASK_STATUS_DISABLED,
|
|
}
|
|
}
|
|
|
|
fn format_tracking_scope_kind(scope_kind: RuntimeTrackingScopeKind) -> &'static str {
|
|
match scope_kind {
|
|
RuntimeTrackingScopeKind::Site => TRACKING_SCOPE_KIND_SITE,
|
|
RuntimeTrackingScopeKind::Work => TRACKING_SCOPE_KIND_WORK,
|
|
RuntimeTrackingScopeKind::Module => TRACKING_SCOPE_KIND_MODULE,
|
|
RuntimeTrackingScopeKind::User => TRACKING_SCOPE_KIND_USER,
|
|
}
|
|
}
|
|
|
|
fn build_profile_invite_code_admin_response(
|
|
record: RuntimeProfileInviteCodeRecord,
|
|
) -> ProfileInviteCodeAdminResponse {
|
|
let metadata =
|
|
serde_json::from_str::<Value>(&record.metadata_json).unwrap_or_else(|_| json!({}));
|
|
ProfileInviteCodeAdminResponse {
|
|
user_id: record.user_id,
|
|
invite_code: record.invite_code,
|
|
metadata,
|
|
created_at: record.created_at,
|
|
updated_at: record.updated_at,
|
|
}
|
|
}
|
|
|
|
fn build_profile_redeem_code_admin_response(
|
|
record: RuntimeProfileRedeemCodeRecord,
|
|
) -> ProfileRedeemCodeAdminResponse {
|
|
ProfileRedeemCodeAdminResponse {
|
|
code: record.code,
|
|
mode: record.mode.as_str().to_string(),
|
|
reward_points: record.reward_points,
|
|
max_uses: record.max_uses,
|
|
global_used_count: record.global_used_count,
|
|
enabled: record.enabled,
|
|
allowed_user_ids: record.allowed_user_ids,
|
|
created_by: record.created_by,
|
|
created_at: record.created_at,
|
|
updated_at: record.updated_at,
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use module_runtime::RuntimeProfileWalletLedgerSourceType;
|
|
|
|
use super::{format_profile_wallet_ledger_source_type, normalize_admin_invite_code_metadata};
|
|
|
|
use axum::{
|
|
body::Body,
|
|
http::{Request, StatusCode},
|
|
};
|
|
use http_body_util::BodyExt;
|
|
use platform_auth::{
|
|
AccessTokenClaims, AccessTokenClaimsInput, AuthProvider, BindingStatus, sign_access_token,
|
|
};
|
|
use serde_json::Value;
|
|
use std::time::Duration;
|
|
use time::OffsetDateTime;
|
|
use tower::ServiceExt;
|
|
|
|
use crate::{app::build_router, config::AppConfig, state::AppState};
|
|
|
|
#[test]
|
|
fn profile_wallet_ledger_source_type_formats_backend_values() {
|
|
assert_eq!(
|
|
format_profile_wallet_ledger_source_type(
|
|
RuntimeProfileWalletLedgerSourceType::NewUserRegistrationReward
|
|
),
|
|
shared_contracts::runtime::PROFILE_WALLET_LEDGER_SOURCE_TYPE_NEW_USER_REGISTRATION_REWARD
|
|
);
|
|
assert_eq!(
|
|
format_profile_wallet_ledger_source_type(
|
|
RuntimeProfileWalletLedgerSourceType::AssetOperationConsume
|
|
),
|
|
shared_contracts::runtime::PROFILE_WALLET_LEDGER_SOURCE_TYPE_ASSET_OPERATION_CONSUME
|
|
);
|
|
assert_eq!(
|
|
format_profile_wallet_ledger_source_type(
|
|
RuntimeProfileWalletLedgerSourceType::AssetOperationRefund
|
|
),
|
|
shared_contracts::runtime::PROFILE_WALLET_LEDGER_SOURCE_TYPE_ASSET_OPERATION_REFUND
|
|
);
|
|
assert_eq!(
|
|
format_profile_wallet_ledger_source_type(
|
|
RuntimeProfileWalletLedgerSourceType::PuzzleAuthorIncentiveClaim
|
|
),
|
|
shared_contracts::runtime::PROFILE_WALLET_LEDGER_SOURCE_TYPE_PUZZLE_AUTHOR_INCENTIVE_CLAIM
|
|
);
|
|
assert_eq!(
|
|
format_profile_wallet_ledger_source_type(
|
|
RuntimeProfileWalletLedgerSourceType::DailyTaskReward
|
|
),
|
|
shared_contracts::runtime::PROFILE_WALLET_LEDGER_SOURCE_TYPE_DAILY_TASK_REWARD
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn profile_dashboard_requires_authentication() {
|
|
let app = build_router(AppState::new(AppConfig::default()).expect("state should build"));
|
|
|
|
let response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("GET")
|
|
.uri("/api/profile/dashboard")
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::UNAUTHORIZED);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn profile_wallet_ledger_requires_authentication() {
|
|
let app = build_router(AppState::new(AppConfig::default()).expect("state should build"));
|
|
|
|
let response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("GET")
|
|
.uri("/api/profile/wallet-ledger")
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::UNAUTHORIZED);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn profile_tasks_require_authentication() {
|
|
let app = build_router(AppState::new(AppConfig::default()).expect("state should build"));
|
|
|
|
let list_response = app
|
|
.clone()
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("GET")
|
|
.uri("/api/profile/tasks")
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
let claim_response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("POST")
|
|
.uri("/api/profile/tasks/daily_login/claim")
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(list_response.status(), StatusCode::UNAUTHORIZED);
|
|
assert_eq!(claim_response.status(), StatusCode::UNAUTHORIZED);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn profile_play_stats_requires_authentication() {
|
|
let app = build_router(AppState::new(AppConfig::default()).expect("state should build"));
|
|
|
|
let response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("GET")
|
|
.uri("/api/profile/play-stats")
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::UNAUTHORIZED);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn profile_recharge_center_requires_authentication() {
|
|
let app = build_router(AppState::new(AppConfig::default()).expect("state should build"));
|
|
|
|
let response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("GET")
|
|
.uri("/api/profile/recharge-center")
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::UNAUTHORIZED);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn profile_recharge_order_requires_authentication() {
|
|
let app = build_router(AppState::new(AppConfig::default()).expect("state should build"));
|
|
|
|
let response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("POST")
|
|
.uri("/api/profile/recharge/orders")
|
|
.header("content-type", "application/json")
|
|
.body(Body::from(r#"{"productId":"points_60"}"#))
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::UNAUTHORIZED);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn profile_referral_invite_center_requires_authentication() {
|
|
let app = build_router(AppState::new(AppConfig::default()).expect("state should build"));
|
|
|
|
let response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("GET")
|
|
.uri("/api/profile/referrals/invite-center")
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::UNAUTHORIZED);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn profile_referral_redeem_code_requires_authentication() {
|
|
let app = build_router(AppState::new(AppConfig::default()).expect("state should build"));
|
|
|
|
let response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("POST")
|
|
.uri("/api/profile/referrals/redeem-code")
|
|
.header("content-type", "application/json")
|
|
.body(Body::from(r#"{"inviteCode":"SY12345678"}"#))
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::UNAUTHORIZED);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn profile_referral_redeem_code_calls_spacetime_for_authenticated_user() {
|
|
let state = seed_authenticated_state().await;
|
|
let token = issue_access_token(&state);
|
|
let app = build_router(state);
|
|
|
|
let response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("POST")
|
|
.uri("/api/profile/referrals/redeem-code")
|
|
.header("authorization", format!("Bearer {token}"))
|
|
.header("content-type", "application/json")
|
|
.body(Body::from(r#"{"inviteCode":"SY12345678"}"#))
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::BAD_GATEWAY);
|
|
let body = response
|
|
.into_body()
|
|
.collect()
|
|
.await
|
|
.expect("body should collect")
|
|
.to_bytes();
|
|
let payload: Value =
|
|
serde_json::from_slice(&body).expect("response body should be valid json");
|
|
assert_eq!(
|
|
payload["error"]["details"]["provider"],
|
|
Value::String("spacetimedb".to_string())
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn admin_invite_code_metadata_accepts_only_json_object() {
|
|
assert_eq!(
|
|
normalize_admin_invite_code_metadata(None).expect("empty metadata should default"),
|
|
"{}"
|
|
);
|
|
assert_eq!(
|
|
normalize_admin_invite_code_metadata(Some(serde_json::json!({
|
|
"channel": "spring",
|
|
"source": "banner"
|
|
})))
|
|
.expect("object metadata should serialize"),
|
|
r#"{"channel":"spring","source":"banner"}"#
|
|
);
|
|
|
|
let error = normalize_admin_invite_code_metadata(Some(serde_json::json!("spring")))
|
|
.expect_err("non-object metadata should reject");
|
|
assert_eq!(error.message(), "邀请码 metadata 必须是 JSON 对象");
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn runtime_profile_legacy_routes_are_not_mounted() {
|
|
let app = build_router(AppState::new(AppConfig::default()).expect("state should build"));
|
|
|
|
for uri in [
|
|
"/api/runtime/profile/dashboard",
|
|
"/api/runtime/profile/wallet-ledger",
|
|
"/api/runtime/profile/recharge-center",
|
|
"/api/runtime/profile/recharge/orders",
|
|
"/api/runtime/profile/referrals/invite-center",
|
|
"/api/runtime/profile/referrals/redeem-code",
|
|
"/api/runtime/profile/redeem-codes/redeem",
|
|
"/api/runtime/profile/play-stats",
|
|
"/api/runtime/profile/save-archives",
|
|
"/api/runtime/profile/save-archives/world-1",
|
|
"/api/runtime/profile/browse-history",
|
|
] {
|
|
let response = app
|
|
.clone()
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("GET")
|
|
.uri(uri)
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::NOT_FOUND, "{uri}");
|
|
}
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn admin_profile_task_routes_require_admin_authentication() {
|
|
let app = build_router(
|
|
AppState::new(admin_enabled_test_config()).expect("state should build"),
|
|
);
|
|
|
|
let list_response = app
|
|
.clone()
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("GET")
|
|
.uri("/admin/api/profile/tasks")
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
let upsert_response = app
|
|
.clone()
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("POST")
|
|
.uri("/admin/api/profile/tasks")
|
|
.header("content-type", "application/json")
|
|
.body(Body::from(r#"{"taskId":"daily_login"}"#))
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
let disable_response = app
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("POST")
|
|
.uri("/admin/api/profile/tasks/disable")
|
|
.header("content-type", "application/json")
|
|
.body(Body::from(r#"{"taskId":"daily_login"}"#))
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(list_response.status(), StatusCode::UNAUTHORIZED);
|
|
assert_eq!(upsert_response.status(), StatusCode::UNAUTHORIZED);
|
|
assert_eq!(disable_response.status(), StatusCode::UNAUTHORIZED);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn admin_profile_code_list_routes_require_admin_authentication() {
|
|
let app = build_router(
|
|
AppState::new(admin_enabled_test_config()).expect("state should build"),
|
|
);
|
|
|
|
for uri in [
|
|
"/admin/api/profile/redeem-codes",
|
|
"/admin/api/profile/invite-codes",
|
|
] {
|
|
let response = app
|
|
.clone()
|
|
.oneshot(
|
|
Request::builder()
|
|
.method("GET")
|
|
.uri(uri)
|
|
.body(Body::empty())
|
|
.expect("request should build"),
|
|
)
|
|
.await
|
|
.expect("request should succeed");
|
|
|
|
assert_eq!(response.status(), StatusCode::UNAUTHORIZED, "{uri}");
|
|
}
|
|
}
|
|
|
|
async fn seed_authenticated_state() -> AppState {
|
|
let state = AppState::new(fast_spacetime_timeout_config()).expect("state should build");
|
|
state
|
|
.seed_test_phone_user_with_password("13800138104", "secret123")
|
|
.await
|
|
.id;
|
|
state
|
|
}
|
|
|
|
fn fast_spacetime_timeout_config() -> AppConfig {
|
|
AppConfig {
|
|
spacetime_procedure_timeout: Duration::from_secs(1),
|
|
..AppConfig::default()
|
|
}
|
|
}
|
|
|
|
fn admin_enabled_test_config() -> AppConfig {
|
|
AppConfig {
|
|
admin_username: Some("root".to_string()),
|
|
admin_password: Some("secret123".to_string()),
|
|
..AppConfig::default()
|
|
}
|
|
}
|
|
|
|
fn issue_access_token(state: &AppState) -> String {
|
|
let claims = AccessTokenClaims::from_input(
|
|
AccessTokenClaimsInput {
|
|
user_id: "user_00000001".to_string(),
|
|
session_id: "sess_runtime_profile".to_string(),
|
|
provider: AuthProvider::Password,
|
|
roles: vec!["user".to_string()],
|
|
token_version: 2,
|
|
phone_verified: true,
|
|
binding_status: BindingStatus::Active,
|
|
display_name: Some("资料页用户".to_string()),
|
|
},
|
|
state.auth_jwt_config(),
|
|
OffsetDateTime::now_utc(),
|
|
)
|
|
.expect("claims should build");
|
|
|
|
sign_access_token(&claims, state.auth_jwt_config()).expect("token should sign")
|
|
}
|
|
}
|