2024-05-22 08:41:30 +01:00
|
|
|
//
|
|
|
|
|
|
|
|
use std::{
|
2024-06-19 07:03:08 +01:00
|
|
|
collections::BTreeMap,
|
2024-05-22 08:41:30 +01:00
|
|
|
net::SocketAddr,
|
|
|
|
path::{Path, PathBuf},
|
|
|
|
str::FromStr,
|
|
|
|
};
|
|
|
|
|
|
|
|
use kxio::fs::FileSystem;
|
2024-07-18 20:51:47 +01:00
|
|
|
use secrecy::Secret;
|
2024-05-22 08:41:30 +01:00
|
|
|
use tracing::info;
|
|
|
|
|
2024-06-19 07:02:18 +01:00
|
|
|
use crate::{newtype, 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
|
2024-06-19 07:03:08 +01:00
|
|
|
#[derive(
|
|
|
|
Clone,
|
|
|
|
Debug,
|
|
|
|
derive_more::From,
|
|
|
|
PartialEq,
|
|
|
|
Eq,
|
|
|
|
PartialOrd,
|
|
|
|
Ord,
|
|
|
|
derive_more::AsRef,
|
|
|
|
serde::Deserialize,
|
|
|
|
derive_more::Constructor,
|
|
|
|
)]
|
2024-05-22 08:41:30 +01:00
|
|
|
pub struct ServerConfig {
|
|
|
|
http: Http,
|
2024-07-18 20:51:47 +01:00
|
|
|
webhook: InboundWebhook,
|
|
|
|
notification: Notification,
|
2024-05-22 08:41:30 +01:00
|
|
|
storage: ServerStorage,
|
2024-06-19 07:03:08 +01:00
|
|
|
pub forge: BTreeMap<String, ForgeConfig>,
|
2024-05-22 08:41:30 +01:00
|
|
|
}
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2024-07-18 20:51:47 +01:00
|
|
|
pub const fn notification(&self) -> &Notification {
|
|
|
|
&self.notification
|
|
|
|
}
|
|
|
|
|
|
|
|
pub const fn inbound_webhook(&self) -> &InboundWebhook {
|
2024-05-22 08:41:30 +01:00
|
|
|
&self.webhook
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn http(&self) -> Result<SocketAddr> {
|
|
|
|
self.http.socket_addr()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Defines the port the server will listen to for incoming webhooks messages
|
2024-06-19 07:03:08 +01:00
|
|
|
#[derive(
|
|
|
|
Clone,
|
|
|
|
Debug,
|
|
|
|
derive_more::From,
|
|
|
|
PartialEq,
|
|
|
|
Eq,
|
|
|
|
PartialOrd,
|
|
|
|
Ord,
|
|
|
|
derive_more::AsRef,
|
|
|
|
serde::Deserialize,
|
|
|
|
derive_more::Constructor,
|
|
|
|
)]
|
2024-05-22 08:41:30 +01:00
|
|
|
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
|
2024-06-19 07:03:08 +01:00
|
|
|
#[derive(
|
|
|
|
Clone,
|
|
|
|
Debug,
|
|
|
|
derive_more::From,
|
|
|
|
PartialEq,
|
|
|
|
Eq,
|
|
|
|
PartialOrd,
|
|
|
|
Ord,
|
|
|
|
derive_more::AsRef,
|
|
|
|
serde::Deserialize,
|
|
|
|
derive_more::Constructor,
|
|
|
|
)]
|
2024-07-18 20:51:47 +01:00
|
|
|
pub struct InboundWebhook {
|
2024-05-22 08:41:30 +01:00
|
|
|
url: String,
|
|
|
|
}
|
2024-07-18 20:51:47 +01:00
|
|
|
impl InboundWebhook {
|
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
|
|
|
}
|
2024-06-08 15:06:34 +01:00
|
|
|
pub fn base_url(&self) -> &str {
|
|
|
|
&self.url
|
|
|
|
}
|
2024-05-22 08:41:30 +01:00
|
|
|
}
|
|
|
|
|
2024-06-29 14:56:20 +01:00
|
|
|
newtype!(WebhookUrl: String, serde::Serialize: "The URL for the webhook where forges should send their updates");
|
2024-05-22 08:41:30 +01:00
|
|
|
|
|
|
|
/// The directory to store server data, such as cloned repos
|
2024-06-19 07:03:08 +01:00
|
|
|
#[derive(
|
|
|
|
Clone,
|
|
|
|
Debug,
|
|
|
|
derive_more::From,
|
|
|
|
PartialEq,
|
|
|
|
Eq,
|
|
|
|
PartialOrd,
|
|
|
|
Ord,
|
|
|
|
derive_more::AsRef,
|
|
|
|
serde::Deserialize,
|
|
|
|
derive_more::Constructor,
|
|
|
|
)]
|
2024-05-22 08:41:30 +01:00
|
|
|
pub struct ServerStorage {
|
|
|
|
path: PathBuf,
|
|
|
|
}
|
|
|
|
impl ServerStorage {
|
|
|
|
pub fn path(&self) -> &Path {
|
|
|
|
self.path.as_path()
|
|
|
|
}
|
|
|
|
}
|
2024-07-18 20:51:47 +01:00
|
|
|
|
|
|
|
/// Identifier for the type of Notification
|
|
|
|
#[derive(
|
|
|
|
Clone,
|
|
|
|
Default,
|
|
|
|
Debug,
|
|
|
|
derive_more::From,
|
|
|
|
PartialEq,
|
|
|
|
Eq,
|
|
|
|
PartialOrd,
|
|
|
|
Ord,
|
|
|
|
serde::Deserialize,
|
|
|
|
Copy,
|
|
|
|
)]
|
|
|
|
pub enum NotificationType {
|
|
|
|
#[default]
|
|
|
|
None,
|
|
|
|
Webhook,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Defines the Webhook Forges should send updates to
|
|
|
|
/// Must be an address that is accessible from the remote forge
|
|
|
|
#[derive(
|
|
|
|
Clone,
|
|
|
|
Debug,
|
|
|
|
derive_more::From,
|
|
|
|
PartialEq,
|
|
|
|
Eq,
|
|
|
|
PartialOrd,
|
|
|
|
Ord,
|
|
|
|
derive_more::AsRef,
|
|
|
|
serde::Deserialize,
|
|
|
|
)]
|
|
|
|
pub struct Notification {
|
|
|
|
r#type: NotificationType,
|
|
|
|
webhook: Option<OutboundWebhook>,
|
|
|
|
}
|
|
|
|
impl Notification {
|
|
|
|
pub const fn none() -> Self {
|
|
|
|
Self {
|
|
|
|
r#type: NotificationType::None,
|
|
|
|
webhook: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pub const fn webhook(webhook: OutboundWebhook) -> Self {
|
|
|
|
Self {
|
|
|
|
r#type: NotificationType::Webhook,
|
|
|
|
webhook: Some(webhook),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub const fn r#type(&self) -> NotificationType {
|
|
|
|
self.r#type
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn webhook_url(&self) -> Option<String> {
|
|
|
|
self.webhook.clone().map(|x| x.url)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn webhook_secret(&self) -> Option<Secret<String>> {
|
|
|
|
self.webhook.clone().map(|x| x.secret).map(Secret::new)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl Default for Notification {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::none()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(
|
|
|
|
Clone,
|
|
|
|
Debug,
|
|
|
|
derive_more::From,
|
|
|
|
PartialEq,
|
|
|
|
Eq,
|
|
|
|
PartialOrd,
|
|
|
|
Ord,
|
|
|
|
serde::Deserialize,
|
|
|
|
derive_more::Constructor,
|
|
|
|
)]
|
|
|
|
pub struct OutboundWebhook {
|
|
|
|
url: String,
|
|
|
|
secret: String,
|
|
|
|
}
|
|
|
|
impl OutboundWebhook {
|
|
|
|
pub fn url(&self) -> &str {
|
|
|
|
self.url.as_ref()
|
|
|
|
}
|
|
|
|
pub fn secret(&self) -> Secret<String> {
|
|
|
|
Secret::new(self.secret.clone())
|
|
|
|
}
|
|
|
|
}
|