diff options
author | Santo Cariotti <santo@dcariotti.me> | 2024-09-07 12:13:12 +0200 |
---|---|---|
committer | Santo Cariotti <santo@dcariotti.me> | 2024-09-07 12:13:12 +0200 |
commit | 2c0c99ce01a69c8a5ec14eab737d7858cf11dcb1 (patch) | |
tree | 00803615c8eb6a3cde0ba5bce197032843502228 /src/graphql/mutation.rs | |
parent | a434be7b14daee8505e45068db7413d7a8886cd8 (diff) |
Use submodules mutations and query for types
Diffstat (limited to 'src/graphql/mutation.rs')
-rw-r--r-- | src/graphql/mutation.rs | 241 |
1 files changed, 10 insertions, 231 deletions
diff --git a/src/graphql/mutation.rs b/src/graphql/mutation.rs index 6e0a265..cef4b22 100644 --- a/src/graphql/mutation.rs +++ b/src/graphql/mutation.rs @@ -1,14 +1,10 @@ -use crate::{ - expo, - graphql::types::{ - alert, - jwt::{self, Authentication}, - notification, position, - user::{self, find_user}, - }, - state::AppState, +use crate::graphql::types::{ + alert, + jwt::{self}, + position, + user::{self}, }; -use async_graphql::{Context, Error, FieldResult, Object}; +use async_graphql::{Context,FieldResult, Object}; /// Mutation struct pub struct Mutation; @@ -36,27 +32,7 @@ impl Mutation { ctx: &Context<'ctx>, input: jwt::LoginCredentials, ) -> FieldResult<jwt::AuthBody> { - let state = ctx.data::<AppState>().expect("Can't connect to db"); - let client = &*state.client; - - let password = sha256::digest(input.password); - let rows = client - .query( - "SELECT id FROM users WHERE email = $1 AND password = $2", - &[&input.email, &password], - ) - .await - .unwrap(); - - let id: Vec<i32> = rows.iter().map(|row| row.get(0)).collect(); - if id.len() == 1 { - // Create a new claim using the found ID - let claims = jwt::Claims::new(id[0]); - let token = claims.get_token().unwrap(); - Ok(jwt::AuthBody::new(token, id[0])) - } else { - Err(Error::new("Invalid email or password")) - } + jwt::mutations::login(ctx, input).await } /// Make GraphQL call to register a notification device token for the user. @@ -79,28 +55,7 @@ impl Mutation { ctx: &Context<'ctx>, input: user::RegisterNotificationToken, ) -> FieldResult<user::User> { - let state = ctx.data::<AppState>().expect("Can't connect to db"); - let client = &*state.client; - - let auth: &Authentication = ctx.data().unwrap(); - match auth { - Authentication::NotLogged => Err(Error::new("Can't find the owner")), - Authentication::Logged(claims) => { - let user = find_user(client, claims.user_id) - .await - .expect("Should not be here"); - - client - .query( - "UPDATE users SET notification_token = $1 WHERE id = $2", - &[&input.token, &claims.user_id], - ) - .await - .unwrap(); - - Ok(user) - } - } + user::mutations::register_device(ctx, input).await } /// Make GraphQL request to create new position to track @@ -126,48 +81,7 @@ impl Mutation { ctx: &Context<'ctx>, input: position::PositionInput, ) -> FieldResult<position::Position> { - let state = ctx.data::<AppState>().expect("Can't connect to db"); - let client = &*state.client; - - let auth: &Authentication = ctx.data().unwrap(); - match auth { - Authentication::NotLogged => Err(Error::new("Can't find the owner")), - Authentication::Logged(claims) => { - let rows = client - .query( - "INSERT INTO positions (user_id, location, activity) - VALUES ( - $1, - ST_SetSRID(ST_MakePoint($2, $3), 4326), - $4 - ) - RETURNING id, user_id, extract(epoch from created_at)::double precision as created_at, ST_Y(location::geometry) AS latitude, ST_X(location::geometry) AS longitude, activity - ", - &[ - &claims.user_id, - &input.longitude, - &input.latitude, - &input.moving_activity, - ], - ) - .await - .unwrap(); - - let positions: Vec<position::Position> = rows - .iter() - .map(|row| position::Position { - id: row.get("id"), - user_id: row.get("user_id"), - created_at: row.get::<_, f64>("created_at") as i64, - latitude: row.get("latitude"), - longitude: row.get("longitude"), - moving_activity: row.get("activity"), - }) - .collect(); - - Ok(positions[0].clone()) - } - } + position::mutations::new_position(ctx, input).await } /// Make GraphQL request to create new alert. Only for admins. @@ -197,141 +111,6 @@ impl Mutation { ctx: &Context<'ctx>, input: alert::AlertInput, ) -> FieldResult<alert::Alert> { - let state = ctx.data::<AppState>().expect("Can't connect to db"); - let client = &*state.client; - - let auth: &Authentication = ctx.data().unwrap(); - match auth { - Authentication::NotLogged => Err(Error::new("Can't find the owner")), - Authentication::Logged(claims) => { - let claim_user = find_user(client, claims.user_id).await.unwrap(); - if !claim_user.is_admin { - return Err(Error::new("Unauthorized")); - } - - let points: String = input - .points - .iter() - .map(|x| { - format!( - "ST_SetSRID(ST_MakePoint({}, {}), 4326)", - x.longitude, x.latitude - ) - }) - .collect::<Vec<String>>() - .join(","); - - let polygon = format!("ST_MakePolygon(ST_MakeLine(ARRAY[{}]))", points); - - let valid_query = format!("SELECT ST_IsValid({}) as is_valid", polygon); - let rows = client.query(&valid_query, &[]).await.unwrap(); - - let is_valid: bool = rows[0].get("is_valid"); - if !is_valid { - return Err(Error::new("Polygon is not valid")); - } - - let insert_query = format!( - "INSERT INTO alerts (user_id, area, level) - VALUES($1, {}, $2) - RETURNING id, user_id, extract(epoch from created_at)::double precision as created_at, ST_AsText(area) as area, - ST_AsText(ST_Buffer(area::geography, CASE WHEN level = 'One' THEN 0 WHEN level = 'Two' THEN 1000 WHEN level = 'Three' THEN 2000 ELSE 0 END)) as extended_area, level, reached_users", - polygon - ); - - let rows = client - .query(&insert_query, &[&claims.user_id, &input.level]) - .await - .unwrap(); - let mut alert = rows - .iter() - .map(|row| alert::Alert { - id: row.get("id"), - user_id: row.get("user_id"), - created_at: row.get::<_, f64>("created_at") as i64, - area: row.get("area"), - extended_area: row.get("extended_area"), - level: row.get("level"), - reached_users: row.get("reached_users"), - }) - .collect::<Vec<alert::Alert>>() - .first() - .cloned() - .ok_or_else(|| Error::new("Failed to create alert"))?; - - let distance: f64 = match alert.level { - alert::LevelAlert::One => 0.0, - alert::LevelAlert::Two => 1000.0, - alert::LevelAlert::Three => 2000.0, - }; - - let position_ids: Vec<i32> = client - .query( - " - SELECT id FROM positions - WHERE ST_DWithin( - location::geography, - (SELECT area::geography FROM alerts WHERE id = $1), - $2 - )", - &[&alert.id, &distance], - ) - .await - .unwrap() - .iter() - .map(|row| row.get(0)) - .collect(); - - let mut notification_ids = vec![]; - for id in &position_ids { - let notification = notification::Notification::insert_db(client, alert.id, *id) - .await - .unwrap(); - notification_ids.push(notification); - } - - alert.reached_users = notification_ids.len() as i32; - client - .query( - "UPDATE alerts SET reached_users = $1 WHERE id = $2", - &[&alert.reached_users, &alert.id], - ) - .await - .unwrap(); - - let placeholders: Vec<String> = (1..=position_ids.len()) - .map(|i| format!("${}", i)) - .collect(); - let query = format!( - "SELECT u.notification_token FROM positions p JOIN users u ON u.id = p.user_id - WHERE p.id IN ({}) AND notification_token IS NOT NULL", - placeholders.join(", ") - ); - - let tokens: Vec<String> = client - .query( - &query, - &position_ids - .iter() - .map(|id| id as &(dyn tokio_postgres::types::ToSql + Sync)) - .collect::<Vec<&(dyn tokio_postgres::types::ToSql + Sync)>>(), - ) - .await - .unwrap() - .iter() - .map(|row| format!("ExponentPushToken[{}]", row.get::<usize, String>(0))) - .collect(); - - expo::send( - tokens, - "New Alert!".to_string(), - "Keep an eye open".to_string(), - ) - .await - .unwrap(); - - Ok(alert) - } - } + alert::mutations::new_alert(ctx, input).await } } |