summaryrefslogtreecommitdiff
path: root/src/graphql/mutation.rs
diff options
context:
space:
mode:
authorSanto Cariotti <santo@dcariotti.me>2024-09-07 12:13:12 +0200
committerSanto Cariotti <santo@dcariotti.me>2024-09-07 12:13:12 +0200
commit2c0c99ce01a69c8a5ec14eab737d7858cf11dcb1 (patch)
tree00803615c8eb6a3cde0ba5bce197032843502228 /src/graphql/mutation.rs
parenta434be7b14daee8505e45068db7413d7a8886cd8 (diff)
Use submodules mutations and query for types
Diffstat (limited to 'src/graphql/mutation.rs')
-rw-r--r--src/graphql/mutation.rs241
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
}
}