feat: add initial structures and routing logic
This commit is contained in:
parent
3883e0f46b
commit
0736f3851b
19
.gitea/issue_template/bug_report.md
Normal file
19
.gitea/issue_template/bug_report.md
Normal file
@ -0,0 +1,19 @@
|
||||
---
|
||||
name: "🐛 Bug report"
|
||||
about: "For reporting bugs"
|
||||
title: ""
|
||||
labels:
|
||||
- "type::bug"
|
||||
- "status::review_needed"
|
||||
|
||||
---
|
||||
|
||||
**Description**
|
||||
*write a concise bug description*
|
||||
|
||||
**Steps to reproduce**
|
||||
1.
|
||||
2.
|
||||
|
||||
**Expected behavior**
|
||||
*describe what you expected to happen*
|
15
.gitea/issue_template/design_discussion.md
Normal file
15
.gitea/issue_template/design_discussion.md
Normal file
@ -0,0 +1,15 @@
|
||||
---
|
||||
name: "🗣 Discussion"
|
||||
about: "For discussion about the software, when you want to discuss about several conception possibilities"
|
||||
title: ""
|
||||
labels:
|
||||
- "type::discussion"
|
||||
- "status::review_needed"
|
||||
|
||||
---
|
||||
|
||||
*describe the problem *
|
||||
|
||||
## Propositions
|
||||
|
||||
*(optionnal) explain the different implementation that you would propose*
|
14
.gitea/issue_template/feature_request.md
Normal file
14
.gitea/issue_template/feature_request.md
Normal file
@ -0,0 +1,14 @@
|
||||
---
|
||||
name: "💡 Feature request"
|
||||
about: "For requesting a new feature, with an implementation plan"
|
||||
title: ""
|
||||
labels:
|
||||
- "type::enhancement"
|
||||
- "status::review_needed"
|
||||
---
|
||||
|
||||
*(if applicable) describe what problem or frustration you have currently*
|
||||
|
||||
*describe what you would like to be able to do, or what solution you would like*
|
||||
|
||||
*(optional) additional context, comments
|
15
.gitea/issue_template/question.md
Normal file
15
.gitea/issue_template/question.md
Normal file
@ -0,0 +1,15 @@
|
||||
---
|
||||
name: "❓ Ask a question"
|
||||
about: "If you have a question about the usage of the libraries or the tool"
|
||||
title: ""
|
||||
labels:
|
||||
- "type::question"
|
||||
- "status::review_needed"
|
||||
|
||||
---
|
||||
|
||||
*ask your question*
|
||||
|
||||
*describe what you have you read so far to try to answer this question ?*
|
||||
|
||||
*(optional) would you think the is a in documentation ?*
|
11
.gitea/issue_template/refactor.md
Normal file
11
.gitea/issue_template/refactor.md
Normal file
@ -0,0 +1,11 @@
|
||||
---
|
||||
name: "🚧 Refactor"
|
||||
about: "For refactoring propositions"
|
||||
title: ""
|
||||
labels:
|
||||
- "type::refactor"
|
||||
- "status::review_needed"
|
||||
|
||||
---
|
||||
|
||||
*explain why and what you want to refactor*
|
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
||||
/target
|
1479
Cargo.lock
generated
Normal file
1479
Cargo.lock
generated
Normal file
File diff suppressed because it is too large
Load Diff
23
Cargo.toml
Normal file
23
Cargo.toml
Normal file
@ -0,0 +1,23 @@
|
||||
[package]
|
||||
name = "feuille"
|
||||
version = "0.1.0"
|
||||
authors = ["koalp <koalp@alpaga.dev>"]
|
||||
edition = "2018"
|
||||
|
||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||
|
||||
[dependencies]
|
||||
anyhow = "1.0.40"
|
||||
dotenv = "0.15.0"
|
||||
url = "2.2.1"
|
||||
thiserror = "1.0.24"
|
||||
|
||||
[dependencies.rocket]
|
||||
git = "https://github.com/SergioBenitez/Rocket"
|
||||
rev = "3a7559edcec7c443e68e22e038aaa2d90ef27c23"
|
||||
version = "0.5.0-dev"
|
||||
|
||||
[dependencies.rocket_contrib]
|
||||
git = "https://github.com/SergioBenitez/Rocket"
|
||||
rev = "3a7559edcec7c443e68e22e038aaa2d90ef27c23"
|
||||
version = "0.5.0-dev"
|
32
README.md
Normal file
32
README.md
Normal file
@ -0,0 +1,32 @@
|
||||
A static website hosting service for use in CD.
|
||||
|
||||
It aim at providing a service similar to github or gitlab pages for self-hosted
|
||||
software forges and CI/CD services.
|
||||
|
||||
# TODO
|
||||
|
||||
## Phase I
|
||||
- [ ] upload /subdomain/<subdomain>/upload -> untar
|
||||
- [ ] serve for any subdomain
|
||||
|
||||
## Phase II
|
||||
|
||||
- [ ] store subdomains linked to each upload (which database ? sled ?)
|
||||
- [ ] serve for specific subdomain (rewrite sub.pages.domain.net to pages.domain.net/sub ?)
|
||||
|
||||
## Phase III
|
||||
|
||||
- [ ] config (ex: single page app, custom 404, et c)
|
||||
- config in static file, or in api ? (or both ?)
|
||||
- [ ] simple login (and prepare fail2ban)
|
||||
|
||||
Yeah ! A minimal working service !
|
||||
|
||||
## Phase IV
|
||||
|
||||
Highly hypothetical
|
||||
|
||||
- [ ] upload from web interface
|
||||
- [ ] ability to set login backend and api OIDC auth
|
||||
- [ ] subdomain reservation mechanism (usecase: forge reserve subdomains for
|
||||
users/orgs)
|
19
src/api.rs
Normal file
19
src/api.rs
Normal file
@ -0,0 +1,19 @@
|
||||
use rocket::{
|
||||
data::TempFile,
|
||||
post,
|
||||
routes,
|
||||
Route,
|
||||
};
|
||||
|
||||
pub fn routes() -> Vec<Route> {
|
||||
routes![subdomain]
|
||||
}
|
||||
|
||||
#[post(
|
||||
"/subdomain/<subdomain>/upload",
|
||||
format = "application/x-gtar",
|
||||
data = "<file>"
|
||||
)]
|
||||
pub fn subdomain(subdomain: String, mut file: TempFile<'_>) {
|
||||
println!("upload to subdomain {}", subdomain);
|
||||
}
|
26
src/main.rs
Normal file
26
src/main.rs
Normal file
@ -0,0 +1,26 @@
|
||||
use std::env;
|
||||
use std::sync::{Arc, Mutex};
|
||||
|
||||
use dotenv::dotenv;
|
||||
use rocket::{get, launch, routes};
|
||||
use url::Host;
|
||||
|
||||
mod subdomains;
|
||||
use subdomains::Hosts;
|
||||
mod api;
|
||||
|
||||
#[launch]
|
||||
fn rocket() -> _ {
|
||||
dotenv().ok();
|
||||
env::var("FEUILLE_DATA_DIR").expect("You have to set the FEUILLE_DATA_DIR configuration");
|
||||
|
||||
// TODO: replace by builder
|
||||
let subdomain_hosts =
|
||||
Hosts::new(Host::parse("example.com").expect("invalid host")).add_subdomain("test");
|
||||
|
||||
rocket::build()
|
||||
.mount("/", api::routes())
|
||||
.mount("/", subdomains::routes())
|
||||
.manage(subdomain_hosts)
|
||||
//.mount("/", StaticFiles::from("tmp.data/"))
|
||||
}
|
110
src/subdomains.rs
Normal file
110
src/subdomains.rs
Normal file
@ -0,0 +1,110 @@
|
||||
use std::path::PathBuf;
|
||||
use std::sync::{self, Arc, Mutex};
|
||||
use std::convert::TryInto;
|
||||
|
||||
use thiserror::Error;
|
||||
use url::Host;
|
||||
use rocket::{get, Route, routes, State};
|
||||
use rocket::outcome::try_outcome;
|
||||
use rocket::request::{FromRequest, Request, Outcome};
|
||||
|
||||
pub fn routes() -> Vec<Route> {
|
||||
routes![static_website]
|
||||
}
|
||||
|
||||
#[derive(PartialEq, Clone, Debug, Error)]
|
||||
pub enum Error {
|
||||
#[error("Can't lock an underling mutex")]
|
||||
Lock,
|
||||
#[error("Malformed subdomain, did you put '.' in it ?")]
|
||||
MalformedSubdomain,
|
||||
#[error("IP host are not supported")]
|
||||
IPHost,
|
||||
}
|
||||
|
||||
type Result<T, E = Error> = core::result::Result<T, E>;
|
||||
|
||||
#[derive(Debug, Clone, PartialEq)]
|
||||
pub(crate) struct Website {
|
||||
path: PathBuf,
|
||||
host: Host,
|
||||
}
|
||||
|
||||
impl Website {
|
||||
/// Creates a website from a given subdomain and a base_host
|
||||
///
|
||||
/// # Errors
|
||||
///
|
||||
/// If your subdomain contains dots, an Error::MalformedUrl is returned
|
||||
/// Error::IPHost is returned if the given host is an IP
|
||||
fn from_subdomain<S: AsRef<str>>(subdomain: S, base_host: &Host) -> Result<Self> {
|
||||
// TODO: implement the path creation
|
||||
let path = "".into();
|
||||
if subdomain.as_ref().contains(".") {
|
||||
return Err(Error::MalformedSubdomain);
|
||||
}
|
||||
let host = match base_host {
|
||||
Host::Domain(base_host) => format!("{}.{}", subdomain.as_ref(), base_host.to_string()),
|
||||
_ => return Err(Error::IPHost),
|
||||
};
|
||||
let host = Host::parse(&host).map_err(|_| Error::MalformedSubdomain)?;
|
||||
Ok(Self {
|
||||
path,
|
||||
host,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone)]
|
||||
pub(crate) struct Hosts {
|
||||
hosts: Arc<Mutex<Vec<Website>>>,
|
||||
base_host: Host,
|
||||
}
|
||||
|
||||
impl Hosts {
|
||||
|
||||
pub fn new(base_host: Host) -> Self {
|
||||
Self{
|
||||
hosts: Arc::new(Mutex::new(vec![])),
|
||||
base_host,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn contains_host(&self, host: &Host) -> Result<bool>{
|
||||
match self.hosts.lock() {
|
||||
Ok(hosts) => Ok(hosts.iter().any(|website| &website.host == host)),
|
||||
Err(_) => Err(Error::Lock),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn add_subdomain<S: AsRef<str>>(&self, subdomain: S) -> Result<()> {
|
||||
match &mut self.hosts.lock() {
|
||||
Ok(hosts) => {
|
||||
hosts.push(Website::from_subdomain(subdomain, &self.base_host)?);
|
||||
Ok(())
|
||||
},
|
||||
Err(_) => Err(Error::Lock),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub struct Header<'a>(pub &'a str);
|
||||
|
||||
#[rocket::async_trait]
|
||||
impl<'a> FromRequest<'a> for Header<'a> {
|
||||
type Error = ();
|
||||
|
||||
async fn from_request(req: &'a Request<'_>) -> Outcome<Self, Self::Error> {
|
||||
let subdomain_hosts = try_outcome!(req.guard::<State<Hosts>>().await);
|
||||
match (req.headers().get_one("Host"), &subdomain_hosts) {
|
||||
// TODO: those unwrap should return Outcome Errors
|
||||
(Some(h), sh) if sh.contains_host(&Host::parse(h).unwrap()).unwrap() => Outcome::Success(Header(h)),
|
||||
(_, _) => return Outcome::Forward(()),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[get("/<path..>")]
|
||||
fn static_website(path: PathBuf, subdomain: Header) {
|
||||
println!("go go go to the subdomain {} and path {:?} !", subdomain.0, path);
|
||||
}
|
Loading…
Reference in New Issue
Block a user