1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
|
use crate::{
dates::GraphQLDate,
graphql::types::{
jwt::{self, Authentication},
position,
},
state::AppState,
};
use async_graphql::{Context, Error, FieldResult, Object};
use chrono::Utc;
/// Mutation struct
pub struct Mutation;
#[Object]
impl Mutation {
/// Make GraphQL login
async fn login<'ctx>(
&self,
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))
} else {
Err(Error::new("Invalid email or password"))
}
}
/// Make GraphQL request to create new position to track
async fn new_position<'ctx>(
&self,
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, created_at, ST_Y(location::geometry) AS latitude, ST_X(location::geometry) AS longitude, activity
",
&[
&claims.user_id,
&input.latitude,
&input.longitude,
&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: GraphQLDate(Utc::now()),
latitude: row.get("latitude"),
longitude: row.get("longitude"),
moving_activity: row.get("activity"),
})
.collect();
Ok(positions[0].clone())
}
}
}
}
|