this repo has no description
1mod common; 2 3use axum::{Router, extract::Request, http::StatusCode, routing::any}; 4use base64::{Engine as _, engine::general_purpose::URL_SAFE_NO_PAD}; 5use reqwest::Client; 6use std::sync::Arc; 7use tokio::net::TcpListener; 8 9async fn spawn_mock_upstream() -> ( 10 String, 11 tokio::sync::mpsc::Receiver<(String, String, Option<String>)>, 12) { 13 let (tx, rx) = tokio::sync::mpsc::channel(10); 14 let tx = Arc::new(tx); 15 16 let app = Router::new().fallback(any(move |req: Request| { 17 let tx = tx.clone(); 18 async move { 19 let method = req.method().to_string(); 20 let uri = req.uri().to_string(); 21 let auth = req 22 .headers() 23 .get("Authorization") 24 .and_then(|h| h.to_str().ok()) 25 .map(|s| s.to_string()); 26 27 let _ = tx.send((method, uri, auth)).await; 28 (StatusCode::OK, "Mock Response") 29 } 30 })); 31 32 let listener = TcpListener::bind("127.0.0.1:0").await.unwrap(); 33 let addr = listener.local_addr().unwrap(); 34 35 tokio::spawn(async move { 36 axum::serve(listener, app).await.unwrap(); 37 }); 38 39 (format!("http://{}", addr), rx) 40} 41 42#[tokio::test] 43async fn test_proxy_via_header() { 44 let app_url = common::base_url().await; 45 let (upstream_url, mut rx) = spawn_mock_upstream().await; 46 let client = Client::new(); 47 48 let res = client 49 .get(format!("{}/xrpc/com.example.test", app_url)) 50 .header("atproto-proxy", &upstream_url) 51 .header("Authorization", "Bearer test-token") 52 .send() 53 .await 54 .unwrap(); 55 56 assert_eq!(res.status(), StatusCode::OK); 57 58 let (method, uri, auth) = rx.recv().await.expect("Upstream should receive request"); 59 assert_eq!(method, "GET"); 60 assert_eq!(uri, "/xrpc/com.example.test"); 61 assert_eq!(auth, Some("Bearer test-token".to_string())); 62} 63 64 65#[tokio::test] 66async fn test_proxy_auth_signing() { 67 let app_url = common::base_url().await; 68 let (upstream_url, mut rx) = spawn_mock_upstream().await; 69 let client = Client::new(); 70 71 let (access_jwt, did) = common::create_account_and_login(&client).await; 72 73 let res = client 74 .get(format!("{}/xrpc/com.example.signed", app_url)) 75 .header("atproto-proxy", &upstream_url) 76 .header("Authorization", format!("Bearer {}", access_jwt)) 77 .send() 78 .await 79 .unwrap(); 80 81 assert_eq!(res.status(), StatusCode::OK); 82 83 let (method, uri, auth) = rx.recv().await.expect("Upstream receive"); 84 assert_eq!(method, "GET"); 85 assert_eq!(uri, "/xrpc/com.example.signed"); 86 87 let received_token = auth.expect("No auth header").replace("Bearer ", ""); 88 assert_ne!(received_token, access_jwt, "Token should be replaced"); 89 90 let parts: Vec<&str> = received_token.split('.').collect(); 91 assert_eq!(parts.len(), 3); 92 93 let payload_bytes = URL_SAFE_NO_PAD.decode(parts[1]).expect("payload b64"); 94 let claims: serde_json::Value = serde_json::from_slice(&payload_bytes).expect("payload json"); 95 96 assert_eq!(claims["iss"], did); 97 assert_eq!(claims["sub"], did); 98 assert_eq!(claims["aud"], upstream_url); 99 assert_eq!(claims["lxm"], "com.example.signed"); 100} 101 102#[tokio::test] 103async fn test_proxy_post_with_body() { 104 let app_url = common::base_url().await; 105 let (upstream_url, mut rx) = spawn_mock_upstream().await; 106 let client = Client::new(); 107 108 let payload = serde_json::json!({ 109 "text": "Hello from proxy test", 110 "createdAt": "2024-01-01T00:00:00Z" 111 }); 112 113 let res = client 114 .post(format!("{}/xrpc/com.example.postMethod", app_url)) 115 .header("atproto-proxy", &upstream_url) 116 .header("Authorization", "Bearer test-token") 117 .json(&payload) 118 .send() 119 .await 120 .unwrap(); 121 122 assert_eq!(res.status(), StatusCode::OK); 123 124 let (method, uri, auth) = rx.recv().await.expect("Upstream should receive request"); 125 assert_eq!(method, "POST"); 126 assert_eq!(uri, "/xrpc/com.example.postMethod"); 127 assert_eq!(auth, Some("Bearer test-token".to_string())); 128} 129 130#[tokio::test] 131async fn test_proxy_with_query_params() { 132 let app_url = common::base_url().await; 133 let (upstream_url, mut rx) = spawn_mock_upstream().await; 134 let client = Client::new(); 135 136 let res = client 137 .get(format!( 138 "{}/xrpc/com.example.query?repo=did:plc:test&collection=app.bsky.feed.post&limit=50", 139 app_url 140 )) 141 .header("atproto-proxy", &upstream_url) 142 .header("Authorization", "Bearer test-token") 143 .send() 144 .await 145 .unwrap(); 146 147 assert_eq!(res.status(), StatusCode::OK); 148 149 let (method, uri, _auth) = rx.recv().await.expect("Upstream should receive request"); 150 assert_eq!(method, "GET"); 151 assert!( 152 uri.contains("repo=did") || uri.contains("repo=did%3Aplc%3Atest"), 153 "URI should contain repo param, got: {}", 154 uri 155 ); 156 assert!( 157 uri.contains("collection=app.bsky.feed.post") || uri.contains("collection=app.bsky"), 158 "URI should contain collection param, got: {}", 159 uri 160 ); 161 assert!( 162 uri.contains("limit=50"), 163 "URI should contain limit param, got: {}", 164 uri 165 ); 166}