simple bot uploads
This commit is contained in:
parent
2dbb085008
commit
52242b03f1
15 changed files with 328 additions and 32 deletions
|
@ -7,7 +7,7 @@ edition = "2021"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
rocket = { version= "0.5.0-rc.1", features = ["json"] }
|
rocket = { version= "0.5.0-rc.1", features = ["json"] }
|
||||||
diesel = { version = "1.4.4", features = ["postgres", "r2d2"] }
|
diesel = { version = "1.4.4", features = ["postgres", "r2d2", "chrono"] }
|
||||||
dotenv = "0.15.0"
|
dotenv = "0.15.0"
|
||||||
rust-argon2 = "0.8"
|
rust-argon2 = "0.8"
|
||||||
rand = "0.8.4"
|
rand = "0.8.4"
|
||||||
|
@ -16,6 +16,7 @@ serde_bytes = "0.11"
|
||||||
chrono = { version = "0.4", features = ["serde"] }
|
chrono = { version = "0.4", features = ["serde"] }
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
base64 = "0.13.0"
|
base64 = "0.13.0"
|
||||||
|
zip = "0.5"
|
||||||
|
|
||||||
|
|
||||||
[dependencies.rocket_sync_db_pools]
|
[dependencies.rocket_sync_db_pools]
|
||||||
|
|
3
backend/migrations/2021-12-18-130837_bots/down.sql
Normal file
3
backend/migrations/2021-12-18-130837_bots/down.sql
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
DROP TABLE code_bundles;
|
||||||
|
DROP INDEX bots_index;
|
||||||
|
DROP TABLE bots;
|
14
backend/migrations/2021-12-18-130837_bots/up.sql
Normal file
14
backend/migrations/2021-12-18-130837_bots/up.sql
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
CREATE TABLE bots (
|
||||||
|
id serial PRIMARY KEY,
|
||||||
|
owner_id integer REFERENCES users(id) NOT NULL,
|
||||||
|
name text NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE UNIQUE INDEX bots_index ON bots(owner_id, name);
|
||||||
|
|
||||||
|
CREATE TABLE code_bundles (
|
||||||
|
id serial PRIMARY KEY,
|
||||||
|
bot_id integer REFERENCES bots(id) NOT NULL,
|
||||||
|
path text NOT NULL,
|
||||||
|
created_at timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP
|
||||||
|
);
|
54
backend/src/db/bots.rs
Normal file
54
backend/src/db/bots.rs
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
use diesel::prelude::*;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
use crate::schema::{bots, code_bundles};
|
||||||
|
use crate::DbConn;
|
||||||
|
use chrono;
|
||||||
|
|
||||||
|
#[derive(Insertable)]
|
||||||
|
#[table_name = "bots"]
|
||||||
|
pub struct NewBot<'a> {
|
||||||
|
pub owner_id: i32,
|
||||||
|
pub name: &'a str,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Queryable, Debug, PartialEq, Serialize, Deserialize)]
|
||||||
|
pub struct Bot {
|
||||||
|
pub id: i32,
|
||||||
|
pub owner_id: i32,
|
||||||
|
pub name: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn create_bot(new_bot: &NewBot, conn: &PgConnection) -> QueryResult<Bot> {
|
||||||
|
diesel::insert_into(bots::table)
|
||||||
|
.values(new_bot)
|
||||||
|
.get_result(conn)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn find_bot(id: i32, conn: &PgConnection) -> QueryResult<Bot> {
|
||||||
|
bots::table.find(id).first(conn)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Insertable)]
|
||||||
|
#[table_name = "code_bundles"]
|
||||||
|
pub struct NewCodeBundle<'a> {
|
||||||
|
pub bot_id: i32,
|
||||||
|
pub path: &'a str,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Queryable, Serialize, Deserialize, Debug)]
|
||||||
|
pub struct CodeBundle {
|
||||||
|
pub id: i32,
|
||||||
|
pub bot_id: i32,
|
||||||
|
pub path: String,
|
||||||
|
pub created_at: chrono::NaiveDateTime,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn create_code_bundle(
|
||||||
|
new_code_bundle: &NewCodeBundle,
|
||||||
|
conn: &PgConnection,
|
||||||
|
) -> QueryResult<CodeBundle> {
|
||||||
|
diesel::insert_into(code_bundles::table)
|
||||||
|
.values(new_code_bundle)
|
||||||
|
.get_result(conn)
|
||||||
|
}
|
|
@ -1,2 +1,3 @@
|
||||||
|
pub mod bots;
|
||||||
pub mod sessions;
|
pub mod sessions;
|
||||||
pub mod users;
|
pub mod users;
|
||||||
|
|
|
@ -22,7 +22,7 @@ pub struct Session {
|
||||||
pub fn create_session(user: &User, conn: &PgConnection) -> Session {
|
pub fn create_session(user: &User, conn: &PgConnection) -> Session {
|
||||||
let new_session = NewSession {
|
let new_session = NewSession {
|
||||||
token: gen_session_token(),
|
token: gen_session_token(),
|
||||||
user_id: user.user_id,
|
user_id: user.id,
|
||||||
};
|
};
|
||||||
let session = insert_into(sessions::table)
|
let session = insert_into(sessions::table)
|
||||||
.values(&new_session)
|
.values(&new_session)
|
||||||
|
|
|
@ -20,7 +20,7 @@ pub struct NewUser<'a> {
|
||||||
|
|
||||||
#[derive(Queryable, Debug)]
|
#[derive(Queryable, Debug)]
|
||||||
pub struct User {
|
pub struct User {
|
||||||
pub user_id: i32,
|
pub id: i32,
|
||||||
pub username: String,
|
pub username: String,
|
||||||
pub password_salt: Vec<u8>,
|
pub password_salt: Vec<u8>,
|
||||||
pub password_hash: Vec<u8>,
|
pub password_hash: Vec<u8>,
|
||||||
|
|
|
@ -8,9 +8,9 @@ extern crate rocket;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
extern crate diesel;
|
extern crate diesel;
|
||||||
|
|
||||||
mod db;
|
pub mod db;
|
||||||
mod routes;
|
pub mod routes;
|
||||||
mod schema;
|
pub mod schema;
|
||||||
|
|
||||||
#[database("postgresql_database")]
|
#[database("postgresql_database")]
|
||||||
pub struct DbConn(diesel::PgConnection);
|
pub struct DbConn(diesel::PgConnection);
|
||||||
|
@ -29,6 +29,9 @@ pub fn rocket() -> Rocket<Build> {
|
||||||
routes::users::register,
|
routes::users::register,
|
||||||
routes::users::login,
|
routes::users::login,
|
||||||
routes::users::current_user,
|
routes::users::current_user,
|
||||||
|
routes::bots::create_bot,
|
||||||
|
routes::bots::get_bot,
|
||||||
|
routes::bots::upload_bot_code,
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
.attach(DbConn::fairing())
|
.attach(DbConn::fairing())
|
||||||
|
|
96
backend/src/routes/bots.rs
Normal file
96
backend/src/routes/bots.rs
Normal file
|
@ -0,0 +1,96 @@
|
||||||
|
use rand::Rng;
|
||||||
|
use rocket::data::ToByteUnit;
|
||||||
|
use rocket::fs::TempFile;
|
||||||
|
use rocket::Data;
|
||||||
|
use rocket::{response::status, serde::json::Json};
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
use std::io::Cursor;
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
use crate::DbConn;
|
||||||
|
|
||||||
|
use crate::db::bots::{self, CodeBundle};
|
||||||
|
use crate::db::users::User;
|
||||||
|
use bots::Bot;
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize, Debug)]
|
||||||
|
pub struct BotParams {
|
||||||
|
name: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: handle errors
|
||||||
|
#[post("/bots", data = "<params>")]
|
||||||
|
pub async fn create_bot(
|
||||||
|
db_conn: DbConn,
|
||||||
|
user: User,
|
||||||
|
params: Json<BotParams>,
|
||||||
|
) -> status::Created<Json<Bot>> {
|
||||||
|
db_conn
|
||||||
|
.run(move |conn| {
|
||||||
|
let bot_params = bots::NewBot {
|
||||||
|
owner_id: user.id,
|
||||||
|
name: ¶ms.name,
|
||||||
|
};
|
||||||
|
let bot = bots::create_bot(&bot_params, conn).unwrap();
|
||||||
|
let bot_url = uri!(get_bot(bot.id)).to_string();
|
||||||
|
status::Created::new(bot_url).body(Json(bot))
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: handle errors
|
||||||
|
#[get("/bots/<bot_id>")]
|
||||||
|
pub async fn get_bot(db_conn: DbConn, bot_id: i32) -> Json<Bot> {
|
||||||
|
db_conn
|
||||||
|
.run(move |conn| {
|
||||||
|
let bot = bots::find_bot(bot_id, conn).unwrap();
|
||||||
|
Json(bot)
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: proper error handling
|
||||||
|
#[post("/bots/<bot_id>/upload", data = "<data>")]
|
||||||
|
pub async fn upload_bot_code(
|
||||||
|
db_conn: DbConn,
|
||||||
|
user: User,
|
||||||
|
bot_id: i32,
|
||||||
|
data: Data<'_>,
|
||||||
|
) -> status::Created<Json<CodeBundle>> {
|
||||||
|
// TODO: put in config somewhere
|
||||||
|
let data_path = "./data/bots";
|
||||||
|
|
||||||
|
let bot = db_conn
|
||||||
|
.run(move |conn| bots::find_bot(bot_id, conn))
|
||||||
|
.await
|
||||||
|
.expect("Bot not found");
|
||||||
|
|
||||||
|
assert_eq!(user.id, bot.owner_id);
|
||||||
|
|
||||||
|
// generate a random filename
|
||||||
|
let token: [u8; 16] = rand::thread_rng().gen();
|
||||||
|
let name = base64::encode(&token);
|
||||||
|
|
||||||
|
let path = Path::new(data_path).join(name);
|
||||||
|
let capped_buf = data.open(10usize.megabytes()).into_bytes().await.unwrap();
|
||||||
|
assert!(capped_buf.is_complete());
|
||||||
|
let buf = capped_buf.into_inner();
|
||||||
|
|
||||||
|
zip::ZipArchive::new(Cursor::new(buf))
|
||||||
|
.unwrap()
|
||||||
|
.extract(&path)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let code_bundle = db_conn
|
||||||
|
.run(move |conn| {
|
||||||
|
let bundle = bots::NewCodeBundle {
|
||||||
|
bot_id: bot.id,
|
||||||
|
path: path.to_str().unwrap(),
|
||||||
|
};
|
||||||
|
bots::create_code_bundle(&bundle, conn).expect("Failed to create code bundle")
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
|
||||||
|
// TODO: proper location
|
||||||
|
status::Created::new("").body(Json(code_bundle))
|
||||||
|
}
|
|
@ -1 +1,2 @@
|
||||||
|
pub mod bots;
|
||||||
pub mod users;
|
pub mod users;
|
||||||
|
|
|
@ -55,7 +55,7 @@ pub struct UserData {
|
||||||
impl From<User> for UserData {
|
impl From<User> for UserData {
|
||||||
fn from(user: User) -> Self {
|
fn from(user: User) -> Self {
|
||||||
UserData {
|
UserData {
|
||||||
user_id: user.user_id,
|
user_id: user.id,
|
||||||
username: user.username,
|
username: user.username,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,3 +1,20 @@
|
||||||
|
table! {
|
||||||
|
bots (id) {
|
||||||
|
id -> Int4,
|
||||||
|
owner_id -> Int4,
|
||||||
|
name -> Text,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
table! {
|
||||||
|
code_bundles (id) {
|
||||||
|
id -> Int4,
|
||||||
|
bot_id -> Int4,
|
||||||
|
path -> Text,
|
||||||
|
created_at -> Timestamp,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
table! {
|
table! {
|
||||||
sessions (id) {
|
sessions (id) {
|
||||||
id -> Int4,
|
id -> Int4,
|
||||||
|
@ -15,6 +32,8 @@ table! {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
joinable!(bots -> users (owner_id));
|
||||||
|
joinable!(code_bundles -> bots (bot_id));
|
||||||
joinable!(sessions -> users (user_id));
|
joinable!(sessions -> users (user_id));
|
||||||
|
|
||||||
allow_tables_to_appear_in_same_query!(sessions, users,);
|
allow_tables_to_appear_in_same_query!(bots, code_bundles, sessions, users,);
|
||||||
|
|
98
backend/tests/bots.rs
Normal file
98
backend/tests/bots.rs
Normal file
|
@ -0,0 +1,98 @@
|
||||||
|
#![feature(async_closure)]
|
||||||
|
extern crate mozaic4_backend;
|
||||||
|
extern crate zip;
|
||||||
|
|
||||||
|
use rocket::http::{ContentType, Status};
|
||||||
|
|
||||||
|
mod util;
|
||||||
|
use mozaic4_backend::db::{bots, sessions, users};
|
||||||
|
use mozaic4_backend::DbConn;
|
||||||
|
use sessions::Session;
|
||||||
|
use users::{Credentials, User};
|
||||||
|
use util::{run_test, BearerAuth};
|
||||||
|
|
||||||
|
async fn user_with_session(conn: &DbConn) -> (User, Session) {
|
||||||
|
conn.run(|conn| {
|
||||||
|
let credentials = Credentials {
|
||||||
|
username: "piepkonijn",
|
||||||
|
password: "geheim123",
|
||||||
|
};
|
||||||
|
let user = users::create_user(&credentials, conn).unwrap();
|
||||||
|
let session = sessions::create_session(&user, conn);
|
||||||
|
(user, session)
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
#[rocket::async_test]
|
||||||
|
async fn test_bot_create() {
|
||||||
|
run_test(async move |client, conn| {
|
||||||
|
let (user, session) = user_with_session(&conn).await;
|
||||||
|
|
||||||
|
let response = client
|
||||||
|
.post("/bots")
|
||||||
|
.header(BearerAuth::new(session.token.clone()))
|
||||||
|
.header(ContentType::JSON)
|
||||||
|
.body(
|
||||||
|
r#"{
|
||||||
|
"name": "testbot"
|
||||||
|
}"#,
|
||||||
|
)
|
||||||
|
.dispatch()
|
||||||
|
.await;
|
||||||
|
|
||||||
|
assert_eq!(response.status(), Status::Created);
|
||||||
|
assert_eq!(response.content_type(), Some(ContentType::JSON));
|
||||||
|
|
||||||
|
let resp_text = response.into_string().await.unwrap();
|
||||||
|
let json: serde_json::Value = serde_json::from_str(&resp_text).unwrap();
|
||||||
|
assert_eq!(json["name"], "testbot");
|
||||||
|
assert_eq!(json["owner_id"], user.id);
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
// create an example zipfile for bot upload
|
||||||
|
fn create_zip() -> std::io::Result<Vec<u8>> {
|
||||||
|
use std::io::Write;
|
||||||
|
use zip::write::FileOptions;
|
||||||
|
|
||||||
|
let cursor = std::io::Cursor::new(Vec::new());
|
||||||
|
let mut zip = zip::ZipWriter::new(cursor);
|
||||||
|
|
||||||
|
zip.start_file("test.txt", FileOptions::default())?;
|
||||||
|
zip.write_all(b"sup brudi")?;
|
||||||
|
let buf = zip.finish()?;
|
||||||
|
Ok(buf.into_inner())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[rocket::async_test]
|
||||||
|
async fn test_bot_upload() {
|
||||||
|
run_test(async move |client, conn| {
|
||||||
|
let (user, session) = user_with_session(&conn).await;
|
||||||
|
|
||||||
|
let owner_id = user.id;
|
||||||
|
let bot = conn
|
||||||
|
.run(move |conn| {
|
||||||
|
let new_bot = bots::NewBot {
|
||||||
|
name: "testbot",
|
||||||
|
owner_id: owner_id,
|
||||||
|
};
|
||||||
|
bots::create_bot(&new_bot, conn).unwrap()
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
|
||||||
|
let zip_file = create_zip().unwrap();
|
||||||
|
|
||||||
|
let response = client
|
||||||
|
.post(format!("/bots/{}/upload", bot.id))
|
||||||
|
.header(BearerAuth::new(session.token.clone()))
|
||||||
|
.header(ContentType::JSON)
|
||||||
|
.body(zip_file)
|
||||||
|
.dispatch()
|
||||||
|
.await;
|
||||||
|
|
||||||
|
assert_eq!(response.status(), Status::Created);
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
}
|
|
@ -1,27 +1,11 @@
|
||||||
#![feature(async_closure)]
|
#![feature(async_closure)]
|
||||||
extern crate mozaic4_backend;
|
extern crate mozaic4_backend;
|
||||||
|
|
||||||
use rocket::http::{ContentType, Header, Status};
|
use rocket::http::{ContentType, Status};
|
||||||
|
|
||||||
mod util;
|
mod util;
|
||||||
use util::run_test;
|
use util::run_test;
|
||||||
|
|
||||||
pub struct BearerAuth {
|
|
||||||
token: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl BearerAuth {
|
|
||||||
pub fn new(token: String) -> Self {
|
|
||||||
Self { token }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a> Into<Header<'a>> for BearerAuth {
|
|
||||||
fn into(self) -> Header<'a> {
|
|
||||||
Header::new("Authorization", format!("Bearer {}", self.token))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[rocket::async_test]
|
#[rocket::async_test]
|
||||||
async fn test_registration() {
|
async fn test_registration() {
|
||||||
run_test(async move |client, _conn| {
|
run_test(async move |client, _conn| {
|
||||||
|
@ -47,7 +31,7 @@ async fn test_registration() {
|
||||||
|
|
||||||
let response = client
|
let response = client
|
||||||
.get("/users/me")
|
.get("/users/me")
|
||||||
.header(BearerAuth::new(token))
|
.header(util::BearerAuth::new(token))
|
||||||
.dispatch()
|
.dispatch()
|
||||||
.await;
|
.await;
|
||||||
|
|
||||||
|
@ -56,7 +40,8 @@ async fn test_registration() {
|
||||||
let resp = response.into_string().await.unwrap();
|
let resp = response.into_string().await.unwrap();
|
||||||
let json: serde_json::Value = serde_json::from_str(&resp).unwrap();
|
let json: serde_json::Value = serde_json::from_str(&resp).unwrap();
|
||||||
assert_eq!(json["username"], "piepkonijn");
|
assert_eq!(json["username"], "piepkonijn");
|
||||||
}).await
|
})
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
|
|
||||||
#[rocket::async_test]
|
#[rocket::async_test]
|
||||||
|
@ -71,5 +56,6 @@ async fn test_reject_invalid_credentials() {
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::Forbidden);
|
assert_eq!(response.status(), Status::Forbidden);
|
||||||
// assert_eq!(response.content_type(), Some(ContentType::JSON));
|
// assert_eq!(response.content_type(), Some(ContentType::JSON));
|
||||||
}).await
|
})
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ use std::future::Future;
|
||||||
|
|
||||||
use diesel::RunQueryDsl;
|
use diesel::RunQueryDsl;
|
||||||
use mozaic4_backend::DbConn;
|
use mozaic4_backend::DbConn;
|
||||||
use rocket::local::asynchronous::Client;
|
use rocket::{http::Header, local::asynchronous::Client};
|
||||||
|
|
||||||
// We use a lock to synchronize between tests so DB operations don't collide.
|
// We use a lock to synchronize between tests so DB operations don't collide.
|
||||||
// For now. In the future, we'll have a nice way to run each test in a DB
|
// For now. In the future, we'll have a nice way to run each test in a DB
|
||||||
|
@ -11,9 +11,13 @@ static DB_LOCK: parking_lot::Mutex<()> = parking_lot::const_mutex(());
|
||||||
|
|
||||||
async fn reset_db(db: &DbConn) {
|
async fn reset_db(db: &DbConn) {
|
||||||
db.run(|conn| {
|
db.run(|conn| {
|
||||||
diesel::sql_query("TRUNCATE TABLE users, sessions")
|
diesel::sql_query(
|
||||||
.execute(conn)
|
r#"
|
||||||
.expect("drop all tables");
|
TRUNCATE TABLE users, sessions,
|
||||||
|
bots, code_bundles"#,
|
||||||
|
)
|
||||||
|
.execute(conn)
|
||||||
|
.expect("drop all tables");
|
||||||
})
|
})
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
|
@ -37,3 +41,19 @@ where
|
||||||
|
|
||||||
test_closure(client, db).await;
|
test_closure(client, db).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub struct BearerAuth {
|
||||||
|
token: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl BearerAuth {
|
||||||
|
pub fn new(token: String) -> Self {
|
||||||
|
Self { token }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> Into<Header<'a>> for BearerAuth {
|
||||||
|
fn into(self) -> Header<'a> {
|
||||||
|
Header::new("Authorization", format!("Bearer {}", self.token))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue