Compare commits
4 Commits
ab81d241df
...
bf84ff6e33
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
bf84ff6e33 | ||
|
|
48bb2ab839 | ||
|
|
35971d14d1 | ||
|
|
72f73a89e1 |
1653
Cargo.lock
generated
1653
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
@ -7,4 +7,6 @@ edition = "2021"
|
|||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
rocket = {version = "0.5.0", features = ["json"] }
|
rocket = {version = "0.5.0", features = ["json"] }
|
||||||
serde = { version = "1.0", features = ["derive"] }
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
|
sea-orm = { version = "^0.12.0", features = [ "sqlx-sqlite", "runtime-tokio-native-tls", "macros", "mock" ] }
|
||||||
|
futures = "0.3.28"
|
||||||
34
src/entities/members.rs
Normal file
34
src/entities/members.rs
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
//! `SeaORM` Entity. Generated by sea-orm-codegen 0.12.10
|
||||||
|
|
||||||
|
use sea_orm::entity::prelude::*;
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq)]
|
||||||
|
#[sea_orm(table_name = "members")]
|
||||||
|
pub struct Model {
|
||||||
|
#[sea_orm(primary_key, auto_increment = false)]
|
||||||
|
pub id: i32,
|
||||||
|
#[sea_orm(column_name = "ntnuUsername")]
|
||||||
|
pub ntnu_username: String,
|
||||||
|
#[sea_orm(column_name = "firstName")]
|
||||||
|
pub first_name: String,
|
||||||
|
#[sea_orm(column_name = "lastName")]
|
||||||
|
pub last_name: String,
|
||||||
|
pub email: String,
|
||||||
|
pub balance: i32,
|
||||||
|
#[sea_orm(column_name = "imagePreference")]
|
||||||
|
pub image_preference: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
|
||||||
|
pub enum Relation {
|
||||||
|
#[sea_orm(has_many = "super::rfid_cards::Entity")]
|
||||||
|
RfidCards,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Related<super::rfid_cards::Entity> for Entity {
|
||||||
|
fn to() -> RelationDef {
|
||||||
|
Relation::RfidCards.def()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ActiveModelBehavior for ActiveModel {}
|
||||||
6
src/entities/mod.rs
Normal file
6
src/entities/mod.rs
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
//! `SeaORM` Entity. Generated by sea-orm-codegen 0.12.10
|
||||||
|
|
||||||
|
pub mod prelude;
|
||||||
|
|
||||||
|
pub mod members;
|
||||||
|
pub mod rfid_cards;
|
||||||
4
src/entities/prelude.rs
Normal file
4
src/entities/prelude.rs
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
//! `SeaORM` Entity. Generated by sea-orm-codegen 0.12.10
|
||||||
|
|
||||||
|
pub use super::members::Entity as Members;
|
||||||
|
pub use super::rfid_cards::Entity as RfidCards;
|
||||||
33
src/entities/rfid_cards.rs
Normal file
33
src/entities/rfid_cards.rs
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
//! `SeaORM` Entity. Generated by sea-orm-codegen 0.12.10
|
||||||
|
|
||||||
|
use sea_orm::entity::prelude::*;
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq)]
|
||||||
|
#[sea_orm(table_name = "rfid_cards")]
|
||||||
|
pub struct Model {
|
||||||
|
#[sea_orm(column_name = "cardId", primary_key, auto_increment = false)]
|
||||||
|
pub card_id: String,
|
||||||
|
#[sea_orm(column_name = "cardComment")]
|
||||||
|
pub card_comment: String,
|
||||||
|
pub member_id: Option<i32>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
|
||||||
|
pub enum Relation {
|
||||||
|
#[sea_orm(
|
||||||
|
belongs_to = "super::members::Entity",
|
||||||
|
from = "Column::MemberId",
|
||||||
|
to = "super::members::Column::Id",
|
||||||
|
on_update = "NoAction",
|
||||||
|
on_delete = "NoAction"
|
||||||
|
)]
|
||||||
|
Members,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Related<super::members::Entity> for Entity {
|
||||||
|
fn to() -> RelationDef {
|
||||||
|
Relation::Members.def()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ActiveModelBehavior for ActiveModel {}
|
||||||
170
src/main.rs
170
src/main.rs
@ -1,9 +1,18 @@
|
|||||||
// Webserver
|
// Webserver
|
||||||
#[macro_use] extern crate rocket;
|
#[macro_use] extern crate rocket;
|
||||||
|
|
||||||
|
use rocket::{State, Error};
|
||||||
use rocket::response::status;
|
use rocket::response::status;
|
||||||
use rocket::serde::{Serialize, Deserialize, json::Json};
|
use rocket::serde::{Serialize, Deserialize, json::Json};
|
||||||
|
|
||||||
// Database
|
// Database
|
||||||
|
use sea_orm::*;
|
||||||
|
|
||||||
|
mod setup;
|
||||||
|
use setup::set_up_db;
|
||||||
|
|
||||||
|
mod entities;
|
||||||
|
use entities::{prelude::*, *};
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
#[derive(Serialize, Deserialize)]
|
||||||
struct RfidCard {
|
struct RfidCard {
|
||||||
@ -13,40 +22,167 @@ struct RfidCard {
|
|||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
#[derive(Serialize, Deserialize)]
|
||||||
struct Member {
|
struct Member {
|
||||||
id: u64,
|
id: i32,
|
||||||
ntnuUsername: String,
|
ntnuUsername: String,
|
||||||
firstName: String,
|
firstName: String,
|
||||||
lastName: String,
|
lastName: String,
|
||||||
email: String,
|
email: String,
|
||||||
balance: i64,
|
balance: i32,
|
||||||
imagePreference: String,
|
imagePreference: String,
|
||||||
rfidCards: Vec<RfidCard>
|
rfidCards: Vec<RfidCard>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
struct MinimalMember {
|
||||||
|
id: i32,
|
||||||
|
ntnuUsername: String,
|
||||||
|
firstName: String,
|
||||||
|
lastName: String,
|
||||||
|
email: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
struct MinimalMemberWithoutId {
|
||||||
|
ntnuUsername: String,
|
||||||
|
firstName: String,
|
||||||
|
lastName: String,
|
||||||
|
email: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Responder)]
|
||||||
|
#[response(status = 500, content_type = "text/plain")]
|
||||||
|
struct ErrorResponder {
|
||||||
|
message: String
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<DbErr> for ErrorResponder {
|
||||||
|
fn from(err: DbErr) -> ErrorResponder {
|
||||||
|
ErrorResponder { message: err.to_string() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl From<String> for ErrorResponder {
|
||||||
|
fn from(string: String) -> ErrorResponder {
|
||||||
|
ErrorResponder { message: string }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl From<&str> for ErrorResponder {
|
||||||
|
fn from(str: &str) -> ErrorResponder {
|
||||||
|
ErrorResponder { message: str.to_owned().into() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[get("/")]
|
#[get("/")]
|
||||||
fn index() -> &'static str {
|
fn index() -> &'static str {
|
||||||
"Hello, world!\nNothing useful is served here."
|
"Hello, world!\nNothing useful is served here."
|
||||||
}
|
}
|
||||||
|
|
||||||
#[get("/member/<memberId>")]
|
#[post("/member", data = "<minimalMemberWithoutId>")]
|
||||||
fn get_member(memberId: u64) -> Json<Member> {
|
async fn add_member(db: &State<DatabaseConnection>, minimalMemberWithoutId: Json<MinimalMemberWithoutId>) -> Result<Json<MinimalMember>, ErrorResponder> {
|
||||||
let sebasthg = Member {
|
// Grab the database connection
|
||||||
id: 1,
|
let db = db as &DatabaseConnection;
|
||||||
ntnuUsername: "sebasthg".to_string(),
|
|
||||||
firstName: "Sebastian".to_string(),
|
// Check if a member with the same NTNU username already exists
|
||||||
lastName: "Gabrielli".to_string(),
|
let matching_member: Option<members::Model> = Members::find() // Find a member in the "Members" table
|
||||||
email: "sebastian@fastmail.mx".to_string(),
|
.filter(members::Column::NtnuUsername.eq(minimalMemberWithoutId.ntnuUsername.to_owned())) // Filter by the provided username in the NtnuUsername column
|
||||||
balance: 50,
|
.one(db) // We only care about one result
|
||||||
imagePreference: "Money".to_string(),
|
.await?; // Wait for the result
|
||||||
rfidCards: vec![ RfidCard { cardId: "0364249683".to_string(), cardComment: "Studentkort".to_string() } ]
|
|
||||||
|
// If a member exists return an error
|
||||||
|
if matching_member.is_some() {
|
||||||
|
return Err(
|
||||||
|
ErrorResponder {
|
||||||
|
message: "A member with this NTNU username already exists".to_string(),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the new member info from the provided JSON
|
||||||
|
let new_member = members::ActiveModel {
|
||||||
|
ntnu_username: ActiveValue::Set(minimalMemberWithoutId.ntnuUsername.to_owned()),
|
||||||
|
first_name: ActiveValue::Set(minimalMemberWithoutId.firstName.to_owned()),
|
||||||
|
last_name: ActiveValue::Set(minimalMemberWithoutId.lastName.to_owned()),
|
||||||
|
email: ActiveValue::Set(minimalMemberWithoutId.email.to_owned()),
|
||||||
|
balance: ActiveValue::Set(0),
|
||||||
|
image_preference: ActiveValue::Set("Money".to_string()),
|
||||||
|
..Default::default()
|
||||||
};
|
};
|
||||||
|
|
||||||
Json(sebasthg)
|
// Add the new member to the database
|
||||||
|
let res = Members::insert(new_member).exec(db).await?;
|
||||||
|
|
||||||
|
// Fetch the member's info back from the DB to verify
|
||||||
|
let new_member;
|
||||||
|
match Members::find_by_id(res.last_insert_id).one(db).await? {
|
||||||
|
Some(model) => new_member = model,
|
||||||
|
None => {
|
||||||
|
return Err(
|
||||||
|
ErrorResponder {
|
||||||
|
message: format!("Failed to fetch member for verification of creating new member with ID. memberId: {}", res.last_insert_id)
|
||||||
|
}
|
||||||
|
);
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// Put the fetched info into a minimal member for returning
|
||||||
|
let member = MinimalMember {
|
||||||
|
id: new_member.id,
|
||||||
|
ntnuUsername: new_member.ntnu_username,
|
||||||
|
firstName: new_member.first_name,
|
||||||
|
lastName: new_member.last_name,
|
||||||
|
email: new_member.email,
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(Json(member))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[get("/member/<memberId>")]
|
||||||
|
async fn get_member_by_id(db: &State<DatabaseConnection>, memberId: i32) -> Result<Json<Member>, ErrorResponder> {
|
||||||
|
let db = db as &DatabaseConnection;
|
||||||
|
|
||||||
|
let database_member;
|
||||||
|
match Members::find_by_id(memberId).one(db).await? {
|
||||||
|
Some(model) => database_member = model,
|
||||||
|
None => {
|
||||||
|
return Err(
|
||||||
|
ErrorResponder {
|
||||||
|
message: format!("Failed to fetch member by ID. memberId: {}", memberId)
|
||||||
|
}
|
||||||
|
);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
let member_rfid_cards: Vec<rfid_cards::Model> = database_member.find_related(RfidCards).all(db).await?;
|
||||||
|
let rfids: Vec<RfidCard> = member_rfid_cards.iter().map(|model| RfidCard {
|
||||||
|
cardId: model.card_id.clone(),
|
||||||
|
cardComment: model.card_comment.clone()
|
||||||
|
}).collect();
|
||||||
|
|
||||||
|
let member = Member {
|
||||||
|
id: database_member.id,
|
||||||
|
ntnuUsername: database_member.ntnu_username,
|
||||||
|
firstName: database_member.first_name,
|
||||||
|
lastName: database_member.last_name,
|
||||||
|
email: database_member.email,
|
||||||
|
balance: database_member.balance,
|
||||||
|
imagePreference: database_member.image_preference,
|
||||||
|
rfidCards: rfids
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(Json(member))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[launch]
|
#[launch]
|
||||||
fn rocket() -> _ {
|
async fn rocket() -> _ {
|
||||||
|
let db = match set_up_db().await {
|
||||||
|
Ok(db) => db,
|
||||||
|
Err(err) => panic!("{}", err)
|
||||||
|
};
|
||||||
|
|
||||||
rocket::build()
|
rocket::build()
|
||||||
.mount("/", routes![index])
|
.manage(db)
|
||||||
.mount("/", routes![get_member])
|
.mount("/", routes![
|
||||||
|
index,
|
||||||
|
get_member_by_id,
|
||||||
|
add_member
|
||||||
|
])
|
||||||
}
|
}
|
||||||
|
|||||||
39
src/setup.rs
Normal file
39
src/setup.rs
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
use sea_orm::*;
|
||||||
|
|
||||||
|
const DATABASE_URL: &str = "sqlite:./test.db";
|
||||||
|
const DB_NAME: &str = "omegav";
|
||||||
|
|
||||||
|
pub(super) async fn set_up_db() -> Result<DatabaseConnection, DbErr> {
|
||||||
|
let db = Database::connect(DATABASE_URL).await?;
|
||||||
|
|
||||||
|
let db = match db.get_database_backend() {
|
||||||
|
DbBackend::MySql => {
|
||||||
|
db.execute(Statement::from_string(
|
||||||
|
db.get_database_backend(),
|
||||||
|
format!("CREATE DATABASE IF NOT EXISTS `{}`;", DB_NAME),
|
||||||
|
))
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let url = format!("{}/{}", DATABASE_URL, DB_NAME);
|
||||||
|
Database::connect(&url).await?
|
||||||
|
}
|
||||||
|
DbBackend::Postgres => {
|
||||||
|
db.execute(Statement::from_string(
|
||||||
|
db.get_database_backend(),
|
||||||
|
format!("DROP DATABASE IF EXISTS \"{}\";", DB_NAME),
|
||||||
|
))
|
||||||
|
.await?;
|
||||||
|
db.execute(Statement::from_string(
|
||||||
|
db.get_database_backend(),
|
||||||
|
format!("CREATE DATABASE \"{}\";", DB_NAME),
|
||||||
|
))
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let url = format!("{}/{}", DATABASE_URL, DB_NAME);
|
||||||
|
Database::connect(&url).await?
|
||||||
|
}
|
||||||
|
DbBackend::Sqlite => db,
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(db)
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue
Block a user