raisfast 0.2.21

The last backend you'll ever need. Rust-powered headless CMS with built-in blog, ecommerce, wallet, payment and 4 plugin engines.
use super::*;

static COUNTER: std::sync::atomic::AtomicU64 = std::sync::atomic::AtomicU64::new(1);

fn unique_tag() -> u64 {
    COUNTER.fetch_add(1, std::sync::atomic::Ordering::Relaxed)
}

async fn setup_admin() -> (axum::Router, AppState, String) {
    let (app, state) = test_app().await;
    raisfast::models::currencies::create(&state.pool, "CNY", "Chinese Yuan", 2)
        .await
        .unwrap();
    raisfast::models::currencies::create(&state.pool, "USD", "US Dollar", 2)
        .await
        .unwrap();
    let (int_id, id) = create_admin(&state.pool).await;
    let tok = make_token(&id, int_id, raisfast::models::user::UserRole::Admin);
    (app, state, tok)
}

async fn seed_user_get_id(app: &mut axum::Router, tag: u64) -> (String, String) {
    let email = format!("w{tag}@test.com");
    let username = format!("wuser{tag}");
    let (tok, _) = register_and_login(app, &email, &username, "Password123").await;
    let (_, body) = send(app, get_auth("/api/v1/users/me", &tok)).await;
    let id = body["data"]["id"].as_str().unwrap().to_string();
    (tok, id)
}

#[tokio::test]
async fn admin_credit_creates_wallet() {
    let tag = unique_tag();
    let (mut app, _state, tok) = setup_admin().await;
    let (_user_tok, user_id) = seed_user_get_id(&mut app, tag).await;

    let (status, body) = send(
        &mut app,
        post_json_auth(
            "/api/v1/admin/wallets/credit",
            json!({
                "user_id": user_id,
                "currency": "CNY",
                "transaction_no": format!("TXCREDIT{tag}"),
                "amount": 10000,
            }),
            &tok,
        ),
    )
    .await;
    assert!(status.is_success(), "credit: {status} {body:?}");
    assert_eq!(body["data"]["amount"], 10000);
    assert_eq!(body["data"]["entry_type"], "credit");
}

#[tokio::test]
async fn admin_debit_deducts_balance() {
    let tag = unique_tag();
    let (mut app, _state, tok) = setup_admin().await;
    let (_user_tok, user_id) = seed_user_get_id(&mut app, tag).await;

    let _ = send(
        &mut app,
        post_json_auth(
            "/api/v1/admin/wallets/credit",
            json!({
                "user_id": user_id,
                "currency": "CNY",
                "transaction_no": format!("TXPRE{tag}"),
                "amount": 5000,
            }),
            &tok,
        ),
    )
    .await;

    let (status, body) = send(
        &mut app,
        post_json_auth(
            "/api/v1/admin/wallets/debit",
            json!({
                "user_id": user_id,
                "currency": "CNY",
                "transaction_no": format!("TXDEBIT{tag}"),
                "amount": 2000,
            }),
            &tok,
        ),
    )
    .await;
    assert!(status.is_success(), "debit: {status} {body:?}");
    assert_eq!(body["data"]["amount"], 2000);
    assert_eq!(body["data"]["entry_type"], "debit");
}

#[tokio::test]
async fn user_list_wallets_after_credit() {
    let tag = unique_tag();
    let (mut app, _state, tok) = setup_admin().await;
    let (user_tok, user_id) = seed_user_get_id(&mut app, tag).await;

    let _ = send(
        &mut app,
        post_json_auth(
            "/api/v1/admin/wallets/credit",
            json!({
                "user_id": user_id,
                "currency": "CNY",
                "transaction_no": format!("TXLIST{tag}"),
                "amount": 3000,
            }),
            &tok,
        ),
    )
    .await;

    let (status, body) = send(&mut app, get_auth("/api/v1/wallets", &user_tok)).await;
    assert!(status.is_success(), "list wallets: {status} {body:?}");
    let items = body["data"].as_array().unwrap();
    assert!(!items.is_empty());
    let cny = items.iter().find(|w| w["currency"] == "CNY").unwrap();
    assert_eq!(cny["balance"], 3000);
}

#[tokio::test]
async fn user_get_wallet_by_currency() {
    let tag = unique_tag();
    let (mut app, _state, tok) = setup_admin().await;
    let (user_tok, user_id) = seed_user_get_id(&mut app, tag).await;

    let _ = send(
        &mut app,
        post_json_auth(
            "/api/v1/admin/wallets/credit",
            json!({
                "user_id": user_id,
                "currency": "USD",
                "transaction_no": format!("TXUSD{tag}"),
                "amount": 5000,
            }),
            &tok,
        ),
    )
    .await;

    let (status, body) = send(&mut app, get_auth("/api/v1/wallets/USD", &user_tok)).await;
    assert!(status.is_success(), "get wallet: {status} {body:?}");
    assert_eq!(body["data"]["currency"], "USD");
    assert_eq!(body["data"]["balance"], 5000);
}

#[tokio::test]
async fn admin_list_all_wallets() {
    let tag = unique_tag();
    let (mut app, _state, tok) = setup_admin().await;
    let (_user_tok, user_id) = seed_user_get_id(&mut app, tag).await;

    let _ = send(
        &mut app,
        post_json_auth(
            "/api/v1/admin/wallets/credit",
            json!({
                "user_id": user_id,
                "currency": "CNY",
                "transaction_no": format!("TXALIST{tag}"),
                "amount": 1000,
            }),
            &tok,
        ),
    )
    .await;

    let (status, body) = send(
        &mut app,
        get_auth("/api/v1/admin/wallets?page=1&page_size=10", &tok),
    )
    .await;
    assert!(status.is_success(), "admin list wallets: {status} {body:?}");
    let items = body["data"]["items"].as_array().unwrap();
    assert!(!items.is_empty());
}

#[tokio::test]
async fn admin_reversal_restores_balance() {
    let tag = unique_tag();
    let (mut app, _state, tok) = setup_admin().await;
    let (_user_tok, user_id) = seed_user_get_id(&mut app, tag).await;

    let (_, credit_body) = send(
        &mut app,
        post_json_auth(
            "/api/v1/admin/wallets/credit",
            json!({
                "user_id": user_id,
                "currency": "CNY",
                "transaction_no": format!("TXREVPRE{tag}"),
                "amount": 5000,
            }),
            &tok,
        ),
    )
    .await;

    let tx_id = credit_body["data"]["id"].as_str().unwrap();

    let (status, body) = send(
        &mut app,
        post_json_auth(
            &format!("/api/v1/admin/wallets/{tx_id}/reversal"),
            json!({"transaction_no": format!("TXREV{tag}")}),
            &tok,
        ),
    )
    .await;
    assert!(status.is_success(), "reversal: {status} {body:?}");
    assert_eq!(body["data"]["entry_type"], "debit");
}

#[tokio::test]
async fn user_list_transactions() {
    let tag = unique_tag();
    let (mut app, _state, tok) = setup_admin().await;
    let (user_tok, user_id) = seed_user_get_id(&mut app, tag).await;

    let _ = send(
        &mut app,
        post_json_auth(
            "/api/v1/admin/wallets/credit",
            json!({
                "user_id": user_id,
                "currency": "CNY",
                "transaction_no": format!("TXTXN{tag}"),
                "amount": 1000,
            }),
            &tok,
        ),
    )
    .await;

    let (status, body) = send(
        &mut app,
        get_auth(
            "/api/v1/wallets/transactions?page=1&page_size=10",
            &user_tok,
        ),
    )
    .await;
    assert!(status.is_success(), "list txns: {status} {body:?}");
    let items = body["data"]["items"].as_array().unwrap();
    assert!(!items.is_empty());
}

#[tokio::test]
async fn debit_insufficient_balance_rejected() {
    let tag = unique_tag();
    let (mut app, _state, tok) = setup_admin().await;
    let (_user_tok, user_id) = seed_user_get_id(&mut app, tag).await;

    let (status, _) = send(
        &mut app,
        post_json_auth(
            "/api/v1/admin/wallets/debit",
            json!({
                "user_id": user_id,
                "currency": "CNY",
                "transaction_no": format!("TXINSUFF{tag}"),
                "amount": 99999,
            }),
            &tok,
        ),
    )
    .await;
    assert!(!status.is_success(), "should reject insufficient balance");
}