2024-05-22 08:41:30 +01:00
|
|
|
//
|
|
|
|
use actix::prelude::*;
|
|
|
|
|
|
|
|
use std::{
|
|
|
|
collections::HashMap,
|
|
|
|
net::SocketAddr,
|
|
|
|
path::{Path, PathBuf},
|
|
|
|
str::FromStr,
|
|
|
|
};
|
|
|
|
|
|
|
|
use kxio::fs::FileSystem;
|
|
|
|
use tracing::info;
|
|
|
|
|
2024-05-29 19:22:05 +01:00
|
|
|
use crate::{ForgeAlias, ForgeConfig, RepoAlias};
|
2024-05-22 08:41:30 +01:00
|
|
|
|
2024-06-03 07:38:59 +01:00
|
|
|
#[derive(Debug, thiserror::Error)]
|
2024-05-22 08:41:30 +01:00
|
|
|
pub enum Error {
|
2024-06-03 07:38:59 +01:00
|
|
|
#[error("fs: {0}")]
|
|
|
|
KxioFs(#[from] kxio::fs::Error),
|
|
|
|
|
|
|
|
#[error("deserialise toml: {0}")]
|
|
|
|
TomlDe(#[from] toml::de::Error),
|
|
|
|
|
|
|
|
#[error("parse IP addres/port: {0}")]
|
|
|
|
AddressParse(#[from] std::net::AddrParseError),
|
2024-05-22 08:41:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type Result<T> = core::result::Result<T, Error>;
|
|
|
|
|
|
|
|
/// Mapped from the `git-next-server.toml` file
|
|
|
|
#[derive(Debug, PartialEq, Eq, serde::Deserialize, Message, derive_more::Constructor)]
|
|
|
|
#[rtype(result = "()")]
|
|
|
|
pub struct ServerConfig {
|
|
|
|
http: Http,
|
|
|
|
webhook: Webhook,
|
|
|
|
storage: ServerStorage,
|
|
|
|
pub forge: HashMap<String, ForgeConfig>,
|
|
|
|
}
|
|
|
|
impl ServerConfig {
|
|
|
|
#[tracing::instrument(skip_all)]
|
|
|
|
pub fn load(fs: &FileSystem) -> Result<Self> {
|
|
|
|
let file = fs.base().join("git-next-server.toml");
|
|
|
|
info!(?file, "");
|
|
|
|
let str = fs.file_read_to_string(&file)?;
|
2024-06-03 07:38:59 +01:00
|
|
|
Ok(toml::from_str(&str)?)
|
2024-05-22 08:41:30 +01:00
|
|
|
}
|
|
|
|
|
2024-05-29 19:22:05 +01:00
|
|
|
pub fn forges(&self) -> impl Iterator<Item = (ForgeAlias, &ForgeConfig)> {
|
2024-05-22 08:41:30 +01:00
|
|
|
self.forge
|
|
|
|
.iter()
|
2024-05-29 19:22:05 +01:00
|
|
|
.map(|(alias, forge)| (ForgeAlias::new(alias.clone()), forge))
|
2024-05-22 08:41:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pub const fn storage(&self) -> &ServerStorage {
|
|
|
|
&self.storage
|
|
|
|
}
|
|
|
|
|
|
|
|
pub const fn webhook(&self) -> &Webhook {
|
|
|
|
&self.webhook
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn http(&self) -> Result<SocketAddr> {
|
|
|
|
self.http.socket_addr()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Defines the port the server will listen to for incoming webhooks messages
|
|
|
|
#[derive(Clone, Debug, PartialEq, Eq, serde::Deserialize, derive_more::Constructor)]
|
|
|
|
pub struct Http {
|
|
|
|
addr: String,
|
|
|
|
port: u16,
|
|
|
|
}
|
|
|
|
impl Http {
|
|
|
|
fn socket_addr(&self) -> Result<SocketAddr> {
|
|
|
|
Ok(SocketAddr::from_str(&format!(
|
|
|
|
"{}:{}",
|
|
|
|
self.addr, self.port
|
|
|
|
))?)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Defines the Webhook Forges should send updates to
|
|
|
|
/// Must be an address that is accessible from the remote forge
|
|
|
|
#[derive(Clone, Debug, PartialEq, Eq, serde::Deserialize, derive_more::Constructor)]
|
|
|
|
pub struct Webhook {
|
|
|
|
url: String,
|
|
|
|
}
|
|
|
|
impl Webhook {
|
2024-05-29 19:22:05 +01:00
|
|
|
pub fn url(&self, forge_alias: &ForgeAlias, repo_alias: &RepoAlias) -> WebhookUrl {
|
|
|
|
let base_url = &self.url;
|
|
|
|
WebhookUrl(format!("{base_url}/{forge_alias}/{repo_alias}"))
|
2024-05-22 08:41:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The URL for the webhook where forges should send their updates
|
2024-06-01 14:49:28 +01:00
|
|
|
#[derive(Clone, Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize, derive_more::AsRef)]
|
2024-05-22 08:41:30 +01:00
|
|
|
pub struct WebhookUrl(String);
|
|
|
|
|
|
|
|
/// The directory to store server data, such as cloned repos
|
|
|
|
#[derive(Clone, Debug, PartialEq, Eq, serde::Deserialize, derive_more::Constructor)]
|
|
|
|
pub struct ServerStorage {
|
|
|
|
path: PathBuf,
|
|
|
|
}
|
|
|
|
impl ServerStorage {
|
|
|
|
pub fn path(&self) -> &Path {
|
|
|
|
self.path.as_path()
|
|
|
|
}
|
|
|
|
}
|