2024-04-09 15:31:59 +01:00
|
|
|
mod branch;
|
2024-04-09 10:44:01 +01:00
|
|
|
mod config;
|
2024-04-10 09:16:42 +01:00
|
|
|
pub mod status;
|
2024-04-09 19:30:05 +01:00
|
|
|
mod webhook;
|
2024-04-09 10:44:01 +01:00
|
|
|
|
|
|
|
use actix::prelude::*;
|
|
|
|
use kxio::network::Network;
|
|
|
|
use tracing::info;
|
|
|
|
|
2024-04-09 18:18:19 +01:00
|
|
|
use crate::server::{
|
|
|
|
config::{RepoConfig, RepoDetails},
|
|
|
|
forge,
|
|
|
|
};
|
2024-04-09 10:44:01 +01:00
|
|
|
|
2024-04-09 19:30:05 +01:00
|
|
|
use self::webhook::WebhookId;
|
|
|
|
|
2024-04-09 10:44:01 +01:00
|
|
|
pub struct RepoActor {
|
|
|
|
details: RepoDetails,
|
|
|
|
config: Option<RepoConfig>, // INFO: if [None] then send [StartRepo] to populate it
|
2024-04-09 19:30:05 +01:00
|
|
|
webhook_id: Option<WebhookId>, // INFO: if [None] then no webhook is configured
|
2024-04-09 10:44:01 +01:00
|
|
|
net: Network,
|
|
|
|
}
|
|
|
|
impl RepoActor {
|
|
|
|
pub(crate) const fn new(details: RepoDetails, net: Network) -> Self {
|
|
|
|
Self {
|
|
|
|
details,
|
|
|
|
config: None,
|
2024-04-09 19:30:05 +01:00
|
|
|
webhook_id: None,
|
2024-04-09 10:44:01 +01:00
|
|
|
net,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl Actor for RepoActor {
|
|
|
|
type Context = Context<Self>;
|
2024-04-09 19:30:05 +01:00
|
|
|
fn stopping(&mut self, ctx: &mut Self::Context) -> Running {
|
|
|
|
match self.webhook_id.take() {
|
|
|
|
Some(webhook_id) => {
|
|
|
|
let repo_details = self.details.clone();
|
|
|
|
let addr = ctx.address();
|
|
|
|
let net = self.net.clone();
|
|
|
|
webhook::unregister(webhook_id, repo_details, addr, net)
|
|
|
|
.into_actor(self)
|
|
|
|
.wait(ctx);
|
|
|
|
Running::Continue
|
|
|
|
}
|
|
|
|
None => Running::Stop,
|
|
|
|
}
|
|
|
|
}
|
2024-04-09 10:44:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Message)]
|
|
|
|
#[rtype(result = "()")]
|
|
|
|
pub struct StartRepo;
|
|
|
|
impl Handler<StartRepo> for RepoActor {
|
|
|
|
type Result = ();
|
|
|
|
fn handle(&mut self, _msg: StartRepo, ctx: &mut Self::Context) -> Self::Result {
|
|
|
|
info!(%self.details, "Starting Repo");
|
|
|
|
let details = self.details.clone();
|
|
|
|
let addr = ctx.address();
|
|
|
|
let net = self.net.clone();
|
|
|
|
config::load(details, addr, net).into_actor(self).wait(ctx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Message)]
|
|
|
|
#[rtype(result = "()")]
|
|
|
|
struct LoadedConfig(pub RepoConfig);
|
|
|
|
impl Handler<LoadedConfig> for RepoActor {
|
|
|
|
type Result = ();
|
2024-04-09 15:31:59 +01:00
|
|
|
fn handle(&mut self, msg: LoadedConfig, ctx: &mut Self::Context) -> Self::Result {
|
2024-04-09 10:44:01 +01:00
|
|
|
let config = msg.0;
|
|
|
|
info!(%self.details, %config, "Config loaded");
|
2024-04-09 15:31:59 +01:00
|
|
|
self.config.replace(config.clone());
|
2024-04-09 16:16:01 +01:00
|
|
|
let repo_details = self.details.clone();
|
2024-04-09 15:31:59 +01:00
|
|
|
let addr = ctx.address();
|
|
|
|
let net = self.net.clone();
|
2024-04-09 16:16:01 +01:00
|
|
|
branch::validate_positions(repo_details, config, addr, net)
|
2024-04-09 15:31:59 +01:00
|
|
|
.into_actor(self)
|
|
|
|
.wait(ctx);
|
2024-04-09 10:44:01 +01:00
|
|
|
}
|
|
|
|
}
|
2024-04-09 18:18:19 +01:00
|
|
|
|
|
|
|
#[derive(Message)]
|
|
|
|
#[rtype(result = "()")]
|
|
|
|
pub struct StartMonitoring {
|
|
|
|
pub main: forge::Commit,
|
|
|
|
pub next: forge::Commit,
|
|
|
|
pub dev: forge::Commit,
|
|
|
|
}
|
|
|
|
impl Handler<StartMonitoring> for RepoActor {
|
|
|
|
type Result = ();
|
2024-04-09 19:30:05 +01:00
|
|
|
fn handle(&mut self, msg: StartMonitoring, ctx: &mut Self::Context) -> Self::Result {
|
|
|
|
info!("Monitoring started");
|
|
|
|
let next_ahead_of_main = msg.main != msg.next;
|
|
|
|
let dev_ahead_of_next = msg.next != msg.dev;
|
|
|
|
let repo_details = self.details.clone();
|
|
|
|
let addr = ctx.address();
|
|
|
|
let net = self.net.clone();
|
|
|
|
if next_ahead_of_main {
|
|
|
|
status::check_next(msg.next, repo_details, addr, net)
|
|
|
|
.into_actor(self)
|
|
|
|
.wait(ctx);
|
|
|
|
} else if dev_ahead_of_next {
|
|
|
|
branch::advance_next(msg.next, repo_details, addr, net)
|
|
|
|
.into_actor(self)
|
|
|
|
.wait(ctx);
|
|
|
|
} else if self.webhook_id.is_none() {
|
|
|
|
webhook::register(repo_details, addr, net)
|
|
|
|
.into_actor(self)
|
|
|
|
.wait(ctx)
|
|
|
|
// TODO: (#18) watch for changes on dev
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Message)]
|
|
|
|
#[rtype(result = "()")]
|
|
|
|
pub struct WebhookRegistered(pub WebhookId);
|
|
|
|
impl Handler<WebhookRegistered> for RepoActor {
|
|
|
|
type Result = ();
|
|
|
|
fn handle(&mut self, msg: WebhookRegistered, _ctx: &mut Self::Context) -> Self::Result {
|
|
|
|
self.webhook_id.replace(msg.0);
|
2024-04-09 18:18:19 +01:00
|
|
|
}
|
|
|
|
}
|
2024-04-09 22:43:54 +01:00
|
|
|
|
|
|
|
#[derive(Message)]
|
|
|
|
#[rtype(result = "()")]
|
|
|
|
pub struct AdvanceMainTo(pub forge::Commit);
|
|
|
|
impl Handler<AdvanceMainTo> for RepoActor {
|
|
|
|
type Result = ();
|
|
|
|
fn handle(&mut self, msg: AdvanceMainTo, ctx: &mut Self::Context) -> Self::Result {
|
|
|
|
let repo_details = self.details.clone();
|
|
|
|
let addr = ctx.address();
|
|
|
|
let net = self.net.clone();
|
|
|
|
branch::advance_main(msg.0, repo_details, addr, net)
|
|
|
|
.into_actor(self)
|
|
|
|
.wait(ctx);
|
|
|
|
}
|
|
|
|
}
|