improved-code

This commit is contained in:
Tony Rewin 2023-09-28 13:12:07 +03:00
parent 235d7417df
commit 7ba39c67f9
2 changed files with 62 additions and 95 deletions

4
README.md Normal file
View File

@ -0,0 +1,4 @@
### ENV
- API_BASE
- REDIS_URL

View File

@ -1,123 +1,86 @@
use actix_sse::Sse; use serde_json::Value;
use actix_web::{web, App, HttpServer, Responder}; use sse_actix_web::{broadcast, Broadcaster};
use tokio::sync::broadcast::Receiver;
use std::collections::HashMap; use std::collections::HashMap;
use std::sync::Mutex;
use std::env; use std::env;
use reqwest::Client; use tokio::sync::broadcast::Receiver;
use serde::{Deserialize, Serialize};
#[derive(Serialize)] #[derive(Debug, Deserialize)]
struct ValidateJWTTokenInput { struct Payload {
token_type: String, reader_id: i32,
token: String,
roles: Option<Vec<String>>,
} }
#[derive(Deserialize)] async fn get_author_id(token: &str) -> Result<i32, Box<dyn std::error::Error>> {
struct ValidateJWTTokenResponse {
is_valid: bool,
claims: HashMap<String, String>,
}
#[derive(Deserialize)]
struct ProfileResponse {
data: ProfileData,
}
#[derive(Deserialize)]
struct ProfileData {
profile: Profile,
}
#[derive(Deserialize)]
struct Profile {
id: i32,
}
#[derive(Deserialize)]
struct AuthorResponse {
data: AuthorData,
}
#[derive(Deserialize)]
struct AuthorData {
author: Author,
}
#[derive(Deserialize)]
struct Author {
id: i32,
}
async fn get_author_id(user_id: i32) -> Result<i32, Box<dyn std::error::Error>> {
let client = Client::new(); let client = Client::new();
let query = format!(r#"{{ let gql = r#"mutation { getSession { user { id } } }"#;
author(user: {}) {{ let api_base = env::var("API_BASE")?;
id let response = client
}} .post(api_base)
}}"#, user_id);
let api_base = env::var("API_BASE").unwrap();
let response = client.post(api_base)
.body(query)
.send()
.await?;
let response_body: AuthorResponse = response.json().await?;
Ok(response_body.data.author.id)
}
async fn get_user_id(token: &str) -> Result<i32, Box<dyn std::error::Error>> {
let client = Client::new();
let query = r#"{ profile { id } }"#;
let authorizer_url = env::var("AUTHORIZER_URL").unwrap();
let response = client.post(authorizer_url)
.bearer_auth(token) .bearer_auth(token)
.body(query) .body(gql)
.send() .send()
.await?; .await?;
let response_body: ProfileResponse = response.json().await?; let response_body: Value = response
Ok(response_body.data.profile.id) .json()
.await
.map_err(|e| format!("Failed to parse response body: {}", e))?;
let id = response_body["data"]["getSession"]["user"]["id"]
.as_i64()
.ok_or("Failed to get user id by token")? as i32;
Ok(id)
} }
async fn sse_handler(receiver: web::Data<Receiver<String>>, token: web::Path<String>) -> impl Responder { async fn sse_handler(
let user_id = match get_user_id(&token).await { broadcaster: web::Data<Broadcaster>,
token: web::Path<String>,
rx: web::Data<Receiver<String>>,
) -> impl Responder {
let author_id = match get_author_id(&token).await {
Ok(id) => id, Ok(id) => id,
Err(e) => { Err(e) => {
eprintln!("Failed to validate token: {}", e); eprintln!("Failed to validate token: {}", e);
return actix_web::HttpResponse::Unauthorized().finish(); return actix_web::HttpResponse::Unauthorized().finish();
} }
}; };
let author_id = match get_author_id(user_id).await.unwrap();
let mut receivers = receiver.lock().unwrap();
let redis_url = env::var("REDIS_URL").unwrap();
let rx = receivers.entry(session_id.into_inner()).or_insert_with(|| {
let (tx, rx) = broadcast::channel(100);
let _handle = tokio::spawn(async move {
let client = redis::Client::open(redis_url).unwrap();
let mut conn = client.get_async_connection().await.unwrap();
let mut pubsub = conn.into_pubsub();
// Subscribe to multiple channels
pubsub.subscribe("new_follower").await.unwrap(); // follow to author
pubsub.subscribe("new_reaction").await.unwrap(); // react on post
pubsub.subscribe("new_shout").await.unwrap(); // post in subscribed topic, author or community
pubsub.subscribe("new_approval").await.unwrap(); // post approved by community
while let Some(msg) = pubsub.on_message().next().await { let mut stream = rx.into_inner().into_stream();
let payload: String = msg.get_payload().unwrap();
tx.send(payload).unwrap(); while let Some(Ok(payload)) = stream.next().await {
let payload: Payload = serde_json::from_str(&payload).unwrap();
if payload.reader_id == author_id {
broadcast(&broadcaster, &payload);
} }
}); }
rx
}); "Subscribed to SSE"
Sse::new(rx.subscribe())
} }
#[actix_web::main] #[actix_web::main]
async fn main() -> std::io::Result<()> { async fn main() -> std::io::Result<()> {
let receivers: web::Data<Mutex<HashMap<String, Receiver<String>>>> = web::Data::new(Mutex::new(HashMap::new())); let broadcaster = Broadcaster::new();
// Create a single Redis Pub/Sub connection and broadcast channel
let (tx, rx) = broadcast::channel(100);
let redis_url = env::var("REDIS_URL").unwrap();
let _handle = tokio::spawn(async move {
let client = redis::Client::open(redis_url).unwrap();
let mut conn = client.get_async_connection().await.unwrap();
let mut pubsub = conn.into_pubsub();
pubsub.subscribe("new_follower").await.unwrap();
pubsub.subscribe("new_reaction").await.unwrap();
pubsub.subscribe("new_shout").await.unwrap();
pubsub.subscribe("new_approval").await.unwrap();
while let Some(msg) = pubsub.on_message().next().await {
let payload: HashMap<String, String> = msg.get_payload().unwrap();
tx.send(serde_json::to_string(&payload).unwrap()).unwrap();
}
});
HttpServer::new(move || { HttpServer::new(move || {
App::new() App::new()
.app_data(receivers.clone()) .data(broadcaster.clone())
.data(rx.clone())
.route("/sse/{token}", web::get().to(sse_handler)) .route("/sse/{token}", web::get().to(sse_handler))
}) })
.bind("127.0.0.1:8080")? .bind("127.0.0.1:8080")?