add: littleweb full

This commit is contained in:
trisua 2025-07-08 13:35:23 -04:00
parent 3fc0872867
commit d67e7c9c33
32 changed files with 1699 additions and 71 deletions

View file

@ -3,15 +3,12 @@ use crate::{
routes::api::v1::{CreateDomain, UpdateDomainData},
State,
};
use axum::{
extract::{Path, Query},
response::IntoResponse,
http::StatusCode,
Extension, Json,
};
use axum::{extract::Path, response::IntoResponse, http::StatusCode, Extension, Json};
use axum_extra::extract::CookieJar;
use tetratto_core::model::{littleweb::Domain, oauth, ApiReturn, Error};
use serde::Deserialize;
use tetratto_core::model::{
littleweb::{Domain, ServiceFsMime},
oauth, ApiReturn, Error,
};
pub async fn get_request(
Path(id): Path<usize>,
@ -112,17 +109,12 @@ pub async fn delete_request(
}
}
#[derive(Deserialize)]
pub struct GetFileQuery {
pub addr: String,
}
pub async fn get_file_request(
Path(addr): Path<String>,
Extension(data): Extension<State>,
Query(props): Query<GetFileQuery>,
) -> impl IntoResponse {
let data = &(data.read().await).0;
let (subdomain, domain, tld, path) = Domain::from_str(&props.addr);
let (subdomain, domain, tld, path) = Domain::from_str(&addr);
// resolve domain
let domain = match data.get_domain_by_name_tld(&domain, &tld).await {
@ -150,9 +142,19 @@ pub async fn get_file_request(
// resolve file
match service.file(&path) {
Some(f) => Ok((
Some((f, _)) => Ok((
[("Content-Type".to_string(), f.mime.to_string())],
f.content,
if f.mime == ServiceFsMime::Html {
f.content.replace(
"</body>",
&format!(
"<script src=\"{}/js/proto_links.js\" defer></script></body>",
data.0.0.host
),
)
} else {
f.content
},
)),
None => {
return Err((

View file

@ -641,7 +641,12 @@ pub fn routes() -> Router {
.route("/services", post(services::create_request))
.route("/services/{id}", get(services::get_request))
.route("/services/{id}", delete(services::delete_request))
.route("/services/{id}/name", post(services::update_name_request))
.route("/services/{id}/files", post(services::update_files_request))
.route(
"/services/{id}/content",
post(services::update_content_request),
)
// domains
.route("/domains", get(domains::list_request))
.route("/domains", post(domains::create_request))
@ -651,7 +656,7 @@ pub fn routes() -> Router {
}
pub fn lw_routes() -> Router {
Router::new().route("/file", get(domains::get_file_request))
Router::new().route("/net/{*addr}", get(domains::get_file_request))
}
#[derive(Deserialize)]
@ -1076,9 +1081,21 @@ pub struct CreateService {
pub name: String,
}
#[derive(Deserialize)]
pub struct UpdateServiceName {
pub name: String,
}
#[derive(Deserialize)]
pub struct UpdateServiceFiles {
pub files: Vec<ServiceFsEntry>,
pub id_path: Vec<String>,
}
#[derive(Deserialize)]
pub struct UpdateServiceFileContent {
pub content: String,
pub id_path: Vec<String>,
}
#[derive(Deserialize)]

View file

@ -1,6 +1,8 @@
use crate::{
get_user_from_token,
routes::api::v1::{UpdateServiceFiles, CreateService},
routes::api::v1::{
CreateService, UpdateServiceFileContent, UpdateServiceFiles, UpdateServiceName,
},
State,
};
use axum::{extract::Path, response::IntoResponse, Extension, Json};
@ -60,6 +62,28 @@ pub async fn create_request(
}
}
pub async fn update_name_request(
jar: CookieJar,
Extension(data): Extension<State>,
Path(id): Path<usize>,
Json(req): Json<UpdateServiceName>,
) -> impl IntoResponse {
let data = &(data.read().await).0;
let user = match get_user_from_token!(jar, data, oauth::AppScope::UserManageServices) {
Some(ua) => ua,
None => return Json(Error::NotAllowed.into()),
};
match data.update_service_name(id, &user, &req.name).await {
Ok(_) => Json(ApiReturn {
ok: true,
message: "Service updated".to_string(),
payload: (),
}),
Err(e) => Json(e.into()),
}
}
pub async fn update_files_request(
jar: CookieJar,
Extension(data): Extension<State>,
@ -72,7 +96,57 @@ pub async fn update_files_request(
None => return Json(Error::NotAllowed.into()),
};
match data.update_service_files(id, &user, req.files).await {
let mut service = match data.get_service_by_id(id).await {
Ok(x) => x,
Err(e) => return Json(e.into()),
};
if req.id_path.is_empty() {
service.files = req.files;
} else {
match service.file_mut(req.id_path) {
Some(f) => f.children = req.files,
None => return Json(Error::GeneralNotFound("file".to_string()).into()),
}
}
match data.update_service_files(id, &user, service.files).await {
Ok(_) => Json(ApiReturn {
ok: true,
message: "Service updated".to_string(),
payload: (),
}),
Err(e) => Json(e.into()),
}
}
pub async fn update_content_request(
jar: CookieJar,
Extension(data): Extension<State>,
Path(id): Path<usize>,
Json(req): Json<UpdateServiceFileContent>,
) -> impl IntoResponse {
let data = &(data.read().await).0;
let user = match get_user_from_token!(jar, data, oauth::AppScope::UserManageServices) {
Some(ua) => ua,
None => return Json(Error::NotAllowed.into()),
};
let mut service = match data.get_service_by_id(id).await {
Ok(x) => x,
Err(e) => return Json(e.into()),
};
// update
let file = match service.file_mut(req.id_path) {
Some(f) => f,
None => return Json(Error::GeneralNotFound("file".to_string()).into()),
};
file.content = req.content;
// ...
match data.update_service_files(id, &user, service.files).await {
Ok(_) => Json(ApiReturn {
ok: true,
message: "Service updated".to_string(),