summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSanto Cariotti <santo@dcariotti.me>2024-09-05 12:29:49 +0200
committerSanto Cariotti <santo@dcariotti.me>2024-09-05 12:29:49 +0200
commitb592d19f1a74a32df0aaed7dbf7484d0ef5ad35a (patch)
tree51d1452393cc282421da923dafa33f751138dac7
parent7e9dbd60f55f02ab065c764f8230aabaa6778eed (diff)
Create notifications from a new alert
-rw-r--r--src/graphql/mutation.rs139
-rw-r--r--src/graphql/types/notification.rs26
2 files changed, 97 insertions, 68 deletions
diff --git a/src/graphql/mutation.rs b/src/graphql/mutation.rs
index b5de3fb..4387b77 100644
--- a/src/graphql/mutation.rs
+++ b/src/graphql/mutation.rs
@@ -2,7 +2,7 @@ use crate::{
graphql::types::{
alert,
jwt::{self, Authentication},
- position,
+ notification, position,
user::{self, find_user},
},
state::AppState,
@@ -203,10 +203,7 @@ impl Mutation {
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
- .expect("Should not be here");
-
+ let claim_user = find_user(client, claims.user_id).await.unwrap();
if !claim_user.is_admin {
return Err(Error::new("Unauthorized"));
}
@@ -223,79 +220,85 @@ impl Mutation {
.collect::<Vec<String>>()
.join(",");
- let polygon = format!(
- "ST_MakePolygon(
- ST_MakeLine(
- ARRAY[{}]
- )
- )",
- points
- );
+ let polygon = format!("ST_MakePolygon(ST_MakeLine(ARRAY[{}]))", points);
- match client
- .query(&format!("SELECT ST_IsValid({}) as is_valid", polygon), &[])
- .await
- {
- Ok(rows) => {
- let valids: Vec<alert::PolygonValid> = rows
- .iter()
- .map(|row| alert::PolygonValid {
- is_valid: row.get("is_valid"),
- })
- .collect();
+ let valid_query = format!("SELECT ST_IsValid({}) as is_valid", polygon);
+ let rows = client.query(&valid_query, &[]).await.unwrap();
- if valids[0].is_valid == false {
- return Err(Error::new("Polygon is not valid"));
- }
- }
- Err(e) => return Err(e.into()),
- };
+ let is_valid: bool = rows[0].get("is_valid");
+ if !is_valid {
+ return Err(Error::new("Polygon is not valid"));
+ }
- let query = format!(
+ 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",
+ 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
);
- match client.query(&query, &[&claims.user_id, &input.level]).await {
- Ok(rows) => {
- let alerts: Vec<alert::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();
+ 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"))?;
- // TODO: Send notifications
+ 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();
- Ok(alerts[0].clone())
- }
- Err(e) => Err(e.into()),
+ let mut notification_ids = vec![];
+ for id in position_ids {
+ let notification = notification::Notification::new(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();
+
+ Ok(alert)
}
}
}
diff --git a/src/graphql/types/notification.rs b/src/graphql/types/notification.rs
index 67fb42f..a8c327e 100644
--- a/src/graphql/types/notification.rs
+++ b/src/graphql/types/notification.rs
@@ -1,9 +1,11 @@
use crate::{
+ errors::AppError,
graphql::types::{alert::Alert, jwt::Authentication, position::Position, user::find_user},
state::AppState,
};
use async_graphql::{Context, SimpleObject};
use serde::{Deserialize, Serialize};
+use tokio_postgres::Client;
#[derive(SimpleObject, Clone, Debug, Serialize, Deserialize)]
/// Notification struct
@@ -14,6 +16,30 @@ pub struct Notification {
pub seen: bool,
pub created_at: i64,
}
+
+impl Notification {
+ /// Create a new notification into the database from an alert_id and a position_id.
+ /// Returns the new ID.
+ pub async fn new(client: &Client, alert_id: i32, position_id: i32) -> Result<i32, AppError> {
+ match client
+ .query(
+ "INSERT INTO notifications(alert_id, position_id)
+ VALUES($1, $2)
+ RETURNING id
+ ",
+ &[&alert_id, &position_id],
+ )
+ .await
+ {
+ Ok(rows) => {
+ let row = rows[0].clone();
+ Ok(row.get("id"))
+ }
+ Err(_) => Err(AppError::Database),
+ }
+ }
+}
+
/// Get notifications from the database
pub async fn get_notifications<'ctx>(
ctx: &Context<'ctx>,