feat: bootstarp loading templates
This commit is contained in:
parent
b884d361b0
commit
a8b58da3eb
10 changed files with 290 additions and 49 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -4,3 +4,4 @@
|
|||
tmp
|
||||
tarpaulin-report.html
|
||||
src/cache_buster_data.json
|
||||
assets
|
||||
|
|
12
Cargo.lock
generated
12
Cargo.lock
generated
|
@ -197,6 +197,17 @@ dependencies = [
|
|||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "actix-web-codegen-const-routes"
|
||||
version = "4.0.0"
|
||||
source = "git+https://github.com/realaravinth/actix-web-codegen-const-routes#0de7af46d50f14ab2c79b79252c03ddb3270a5e9"
|
||||
dependencies = [
|
||||
"actix-router",
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "adler"
|
||||
version = "1.0.2"
|
||||
|
@ -2736,6 +2747,7 @@ dependencies = [
|
|||
"actix-identity",
|
||||
"actix-rt",
|
||||
"actix-web",
|
||||
"actix-web-codegen-const-routes",
|
||||
"async-trait",
|
||||
"cache-buster",
|
||||
"config",
|
||||
|
|
|
@ -44,6 +44,9 @@ urlencoding = "2.1.0"
|
|||
[dependencies.cache-buster]
|
||||
git = "https://github.com/realaravinth/cache-buster"
|
||||
|
||||
[dependencies.actix-web-codegen-const-routes]
|
||||
git = "https://github.com/realaravinth/actix-web-codegen-const-routes"
|
||||
|
||||
[dependencies.reqwest]
|
||||
features = ["rustls-tls-native-roots", "gzip", "deflate", "brotli", "json"]
|
||||
version = "0.11.10"
|
||||
|
|
|
@ -26,6 +26,7 @@ pub mod errors;
|
|||
pub mod federate;
|
||||
pub mod forge;
|
||||
pub mod pages;
|
||||
pub mod routes;
|
||||
pub mod settings;
|
||||
pub mod spider;
|
||||
pub mod static_assets;
|
||||
|
@ -50,7 +51,7 @@ pub const DOMAIN: &str = "developer-starchart.forgeflux.org";
|
|||
|
||||
pub type ArcCtx = Arc<Ctx>;
|
||||
pub type WebCtx = Data<ArcCtx>;
|
||||
pub type WebData = Data<BoxDB>;
|
||||
pub type WebDB = Data<BoxDB>;
|
||||
pub type WebFederate = Data<ArcFederate>;
|
||||
|
||||
lazy_static! {
|
||||
|
@ -79,7 +80,7 @@ async fn main() {
|
|||
.wrap(
|
||||
middleware::DefaultHeaders::new().add(("Permissions-Policy", "interest-cohort=()")),
|
||||
)
|
||||
.configure(services)
|
||||
.configure(routes::services)
|
||||
})
|
||||
.bind(&socket_addr)
|
||||
.unwrap()
|
||||
|
@ -87,7 +88,3 @@ async fn main() {
|
|||
.await
|
||||
.unwrap();
|
||||
}
|
||||
|
||||
fn services(cfg: &mut actix_web::web::ServiceConfig) {
|
||||
// cfg.service();
|
||||
}
|
||||
|
|
|
@ -146,30 +146,30 @@ mod tests {
|
|||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod http_page_tests {
|
||||
use actix_web::http::StatusCode;
|
||||
use actix_web::test;
|
||||
|
||||
use crate::ctx::Ctx;
|
||||
use crate::db::BoxDB;
|
||||
use crate::tests::*;
|
||||
use crate::*;
|
||||
|
||||
use super::PAGES;
|
||||
|
||||
#[actix_rt::test]
|
||||
async fn sqlite_templates_work() {
|
||||
let (db, data) = sqlx_sqlite::get_ctx().await;
|
||||
templates_work(data, db).await;
|
||||
}
|
||||
|
||||
async fn templates_work(data: Arc<Data>, db: BoxDB) {
|
||||
let app = get_app!(data, db).await;
|
||||
|
||||
for file in [PAGES.auth.login, PAGES.auth.register].iter() {
|
||||
let resp = get_request!(&app, file);
|
||||
assert_eq!(resp.status(), StatusCode::OK);
|
||||
}
|
||||
}
|
||||
}
|
||||
//#[cfg(test)]
|
||||
//mod http_page_tests {
|
||||
// use actix_web::http::StatusCode;
|
||||
// use actix_web::test;
|
||||
//
|
||||
// use crate::ctx::Ctx;
|
||||
// use crate::db::BoxDB;
|
||||
// use crate::tests::*;
|
||||
// use crate::*;
|
||||
//
|
||||
// use super::PAGES;
|
||||
//
|
||||
// #[actix_rt::test]
|
||||
// async fn sqlite_templates_work() {
|
||||
// let (db, data, _federate, _tmp_dir) = sqlx_sqlite::get_ctx().await;
|
||||
// templates_work(data, db).await;
|
||||
// }
|
||||
//
|
||||
// async fn templates_work(data: ArcCtx, db: BoxDB) {
|
||||
// let app = get_app!(data, db).await;
|
||||
//
|
||||
// for file in [PAGES.auth.login, PAGES.auth.register].iter() {
|
||||
// let resp = get_request!(&app, file);
|
||||
// assert_eq!(resp.status(), StatusCode::OK);
|
||||
// }
|
||||
// }
|
||||
//}
|
||||
|
|
|
@ -42,6 +42,9 @@ pub struct Auth {
|
|||
pub logout: &'static str,
|
||||
/// login route
|
||||
pub login: &'static str,
|
||||
|
||||
/// verify route
|
||||
pub verify: &'static str,
|
||||
}
|
||||
|
||||
impl Auth {
|
||||
|
@ -49,7 +52,12 @@ impl Auth {
|
|||
pub const fn new() -> Auth {
|
||||
let login = "/login";
|
||||
let logout = "/logout";
|
||||
Auth { login, logout }
|
||||
let verify = "/verify";
|
||||
Auth {
|
||||
login,
|
||||
logout,
|
||||
verify,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
22
src/routes.rs
Normal file
22
src/routes.rs
Normal file
|
@ -0,0 +1,22 @@
|
|||
/*
|
||||
* ForgeFlux StarChart - A federated software forge spider
|
||||
* Copyright © 2022 Aravinth Manivannan <realaravinth@batsense.net>
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU Affero General Public License as
|
||||
* published by the Free Software Foundation, either version 3 of the
|
||||
* License, or (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU Affero General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Affero General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
pub fn services(cfg: &mut actix_web::web::ServiceConfig) {
|
||||
crate::pages::services(cfg);
|
||||
crate::static_assets::services(cfg);
|
||||
}
|
|
@ -67,27 +67,17 @@ mod tests {
|
|||
use actix_web::http::StatusCode;
|
||||
use actix_web::test;
|
||||
|
||||
use crate::data::Data;
|
||||
use crate::db::BoxDB;
|
||||
use crate::tests::*;
|
||||
use crate::ArcCtx;
|
||||
use crate::*;
|
||||
|
||||
use super::assets::CSS;
|
||||
|
||||
#[actix_rt::test]
|
||||
async fn postgrest_static_files_works() {
|
||||
let (db, data) = sqlx_postgres::get_data().await;
|
||||
static_assets_work(data, db).await;
|
||||
}
|
||||
|
||||
#[actix_rt::test]
|
||||
async fn sqlite_static_files_works() {
|
||||
let (db, data) = sqlx_sqlite::get_data().await;
|
||||
static_assets_work(data, db).await;
|
||||
}
|
||||
|
||||
async fn static_assets_work(data: Arc<Data>, db: BoxDB) {
|
||||
let app = get_app!(data, db).await;
|
||||
async fn static_assets_work() {
|
||||
let (db, ctx, federate, _tmpdir) = sqlx_sqlite::get_ctx().await;
|
||||
let app = get_app!(ctx, db, federate).await;
|
||||
|
||||
let file = *CSS;
|
||||
let resp = get_request!(&app, file);
|
||||
|
|
210
src/tests.rs
210
src/tests.rs
|
@ -24,6 +24,19 @@ pub use crate::db::BoxDB;
|
|||
pub use crate::federate::{get_federate, ArcFederate};
|
||||
use crate::settings::{DBType, Settings};
|
||||
|
||||
|
||||
//use actix_web::cookie::Cookie;
|
||||
use actix_web::test;
|
||||
use actix_web::{
|
||||
body::{BoxBody, EitherBody},
|
||||
dev::ServiceResponse,
|
||||
error::ResponseError,
|
||||
http::StatusCode,
|
||||
};
|
||||
use serde::Serialize;
|
||||
use crate::errors::*;
|
||||
use crate::*;
|
||||
|
||||
//pub mod sqlx_postgres {
|
||||
// use super::*;
|
||||
//
|
||||
|
@ -41,7 +54,7 @@ pub mod sqlx_sqlite {
|
|||
use crate::db::sqlite;
|
||||
use mktemp::Temp;
|
||||
|
||||
pub async fn get_ctx() -> (BoxDB, Arc<Ctx>, ArcFederate, Temp) {
|
||||
pub async fn get_ctx() -> (BoxDB, ArcCtx, ArcFederate, Temp) {
|
||||
let url = env::var("SQLITE_DATABASE_URL").unwrap();
|
||||
env::set_var("DATABASE_URL", &url);
|
||||
println!("found db url: {url}");
|
||||
|
@ -57,3 +70,198 @@ pub mod sqlx_sqlite {
|
|||
(db, Ctx::new(settings).await, federate, tmp_dir)
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! get_cookie {
|
||||
($resp:expr) => {
|
||||
$resp.response().cookies().next().unwrap().to_owned()
|
||||
};
|
||||
}
|
||||
|
||||
#[allow(dead_code, clippy::upper_case_acronyms)]
|
||||
pub struct FORM;
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! post_request {
|
||||
($uri:expr) => {
|
||||
test::TestRequest::post().uri($uri)
|
||||
};
|
||||
|
||||
($serializable:expr, $uri:expr) => {
|
||||
test::TestRequest::post()
|
||||
.uri($uri)
|
||||
.insert_header((actix_web::http::header::CONTENT_TYPE, "application/json"))
|
||||
.set_payload(serde_json::to_string($serializable).unwrap())
|
||||
};
|
||||
|
||||
($serializable:expr, $uri:expr, FORM) => {
|
||||
test::TestRequest::post().uri($uri).set_form($serializable)
|
||||
};
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! get_request {
|
||||
($app:expr,$route:expr ) => {
|
||||
test::call_service(&$app, test::TestRequest::get().uri($route).to_request()).await
|
||||
};
|
||||
|
||||
($app:expr, $route:expr, $cookies:expr) => {
|
||||
test::call_service(
|
||||
&$app,
|
||||
test::TestRequest::get()
|
||||
.uri($route)
|
||||
.cookie($cookies)
|
||||
.to_request(),
|
||||
)
|
||||
.await
|
||||
};
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! delete_request {
|
||||
($app:expr,$route:expr ) => {
|
||||
test::call_service(&$app, test::TestRequest::delete().uri($route).to_request()).await
|
||||
};
|
||||
|
||||
($app:expr, $route:expr, $cookies:expr) => {
|
||||
test::call_service(
|
||||
&$app,
|
||||
test::TestRequest::delete()
|
||||
.uri($route)
|
||||
.cookie($cookies)
|
||||
.to_request(),
|
||||
)
|
||||
.await
|
||||
};
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! get_app {
|
||||
("APP", $settings:expr) => {
|
||||
actix_web::App::new()
|
||||
.wrap(actix_web::middleware::NormalizePath::new(
|
||||
actix_web::middleware::TrailingSlash::Trim,
|
||||
))
|
||||
.configure(crate::routes::services)
|
||||
};
|
||||
|
||||
($settings:ident) => {
|
||||
test::init_service(get_app!("APP", $settings))
|
||||
};
|
||||
($ctx:expr, $db:expr, $federate:expr) => {
|
||||
test::init_service(
|
||||
get_app!("APP", &$ctx.settings)
|
||||
.app_data(crate::WebDB::new($db.clone()))
|
||||
.app_data(crate::WebCtx::new($ctx.clone()))
|
||||
.app_data(crate::WebFederate::new($federate.clone())),
|
||||
)
|
||||
};
|
||||
}
|
||||
|
||||
//impl Data {
|
||||
// /// register and signin utility
|
||||
// pub async fn register_and_signin(
|
||||
// &self,
|
||||
// db: &BoxDB,
|
||||
// name: &str,
|
||||
// email: &str,
|
||||
// password: &str,
|
||||
// ) -> (Login, ServiceResponse<EitherBody<BoxBody>>) {
|
||||
// self.register_test(db, name, email, password).await;
|
||||
// self.signin_test(db, name, password).await
|
||||
// }
|
||||
//
|
||||
// pub fn to_arc(&self) -> Arc<Self> {
|
||||
// Arc::new(self.clone())
|
||||
// }
|
||||
//
|
||||
// /// register utility
|
||||
// pub async fn register_test(&self, db: &BoxDB, name: &str, email: &str, password: &str) {
|
||||
// let app = get_app!(self.to_arc(), db.clone()).await;
|
||||
//
|
||||
// // 1. Register
|
||||
// let msg = Register {
|
||||
// username: name.into(),
|
||||
// password: password.into(),
|
||||
// confirm_password: password.into(),
|
||||
// email: Some(email.into()),
|
||||
// };
|
||||
// let resp =
|
||||
// test::call_service(&app, post_request!(&msg, ROUTES.auth.register).to_request()).await;
|
||||
// // let resp_err: ErrorToResponse = test::read_body_json(resp).await;
|
||||
// // panic!("{}", resp_err.error);
|
||||
// assert_eq!(resp.status(), StatusCode::OK);
|
||||
// }
|
||||
//
|
||||
// /// signin util
|
||||
// pub async fn signin_test(
|
||||
// &self,
|
||||
// db: &BoxDB,
|
||||
// name: &str,
|
||||
// password: &str,
|
||||
// ) -> (Login, ServiceResponse<EitherBody<BoxBody>>) {
|
||||
// let app = get_app!(self.to_arc(), db.clone()).await;
|
||||
//
|
||||
// // 2. signin
|
||||
// let creds = Login {
|
||||
// login: name.into(),
|
||||
// password: password.into(),
|
||||
// };
|
||||
// let signin_resp =
|
||||
// test::call_service(&app, post_request!(&creds, ROUTES.auth.login).to_request()).await;
|
||||
// assert_eq!(signin_resp.status(), StatusCode::OK);
|
||||
// (creds, signin_resp)
|
||||
// }
|
||||
//
|
||||
// /// pub duplicate test
|
||||
// pub async fn bad_post_req_test<T: Serialize>(
|
||||
// &self,
|
||||
// db: &BoxDB,
|
||||
// name: &str,
|
||||
// password: &str,
|
||||
// url: &str,
|
||||
// payload: &T,
|
||||
// err: ServiceError,
|
||||
// ) {
|
||||
// let (_, signin_resp) = self.signin_test(db, name, password).await;
|
||||
// let cookies = get_cookie!(signin_resp);
|
||||
// let app = get_app!(self.to_arc(), db.clone()).await;
|
||||
//
|
||||
// let resp = test::call_service(
|
||||
// &app,
|
||||
// post_request!(&payload, url)
|
||||
// .cookie(cookies.clone())
|
||||
// .to_request(),
|
||||
// )
|
||||
// .await;
|
||||
// assert_eq!(resp.status(), err.status_code());
|
||||
// let resp_err: ErrorToResponse = test::read_body_json(resp).await;
|
||||
// //println!("{}", txt.error);
|
||||
// assert_eq!(resp_err.error, format!("{}", err));
|
||||
// }
|
||||
//
|
||||
// /// bad post req test without payload
|
||||
// pub async fn bad_post_req_test_witout_payload(
|
||||
// &self,
|
||||
// db: &BoxDB,
|
||||
// name: &str,
|
||||
// password: &str,
|
||||
// url: &str,
|
||||
// err: ServiceError,
|
||||
// ) {
|
||||
// let (_, signin_resp) = self.signin_test(db, name, password).await;
|
||||
// let app = get_app!(self.to_arc(), db.clone()).await;
|
||||
// let cookies = get_cookie!(signin_resp);
|
||||
//
|
||||
// let resp = test::call_service(
|
||||
// &app,
|
||||
// post_request!(url).cookie(cookies.clone()).to_request(),
|
||||
// )
|
||||
// .await;
|
||||
// assert_eq!(resp.status(), err.status_code());
|
||||
// let resp_err: ErrorToResponse = test::read_body_json(resp).await;
|
||||
// //println!("{}", resp_err.error);
|
||||
// assert_eq!(resp_err.error, format!("{}", err));
|
||||
// }
|
||||
//}
|
||||
|
|
0
static/cache/css/main.css
vendored
Normal file
0
static/cache/css/main.css
vendored
Normal file
Loading…
Add table
Reference in a new issue