Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 31903e8ff9 | |||
| 9b7d07cdae | |||
| 05a839044c | |||
| 5ba96e86d9 | |||
| f3a107b2a6 | |||
| aa056419c6 | |||
| 3f14c0d572 | |||
| d0a9583d22 |
13
.env.example
Normal file
13
.env.example
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# Backup Configuration
|
||||||
|
BACKUP_FOLDER=/data/volumes
|
||||||
|
BACKUP_CRON=*/5 * * * *
|
||||||
|
|
||||||
|
# FTP Configuration
|
||||||
|
FTP_HOST=ftp.example.com
|
||||||
|
FTP_PORT=21
|
||||||
|
FTP_USER=backup_user
|
||||||
|
FTP_PASSWORD=secret123
|
||||||
|
FTP_PATH=/backups
|
||||||
|
|
||||||
|
# Docker Configuration
|
||||||
|
DOCKER_HOST=unix:///var/run/docker.sock
|
||||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -1 +1,2 @@
|
|||||||
/target
|
/target
|
||||||
|
.env
|
||||||
13
.woodpecker/code-quality.yml
Normal file
13
.woodpecker/code-quality.yml
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
steps:
|
||||||
|
- name: clippy
|
||||||
|
image: rust:${RUST_VERSION}
|
||||||
|
commands:
|
||||||
|
- rustup component add clippy
|
||||||
|
- cargo clippy --all-targets --all-features -- -D warnings
|
||||||
|
when:
|
||||||
|
- event: [push, pull_request]
|
||||||
|
|
||||||
|
matrix:
|
||||||
|
RUST_VERSION:
|
||||||
|
- "1.91.1"
|
||||||
|
- "1.91.0"
|
||||||
20
.woodpecker/test.yml
Normal file
20
.woodpecker/test.yml
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
services:
|
||||||
|
- name: docker
|
||||||
|
image: docker:dind
|
||||||
|
privileged: true
|
||||||
|
commands:
|
||||||
|
- dockerd-entrypoint.sh &
|
||||||
|
- sleep 5
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: test
|
||||||
|
image: rust:${RUST_VERSION}
|
||||||
|
environment:
|
||||||
|
DOCKER_HOST: tcp://docker:2375
|
||||||
|
commands:
|
||||||
|
- cargo test --verbose
|
||||||
|
|
||||||
|
matrix:
|
||||||
|
RUST_VERSION:
|
||||||
|
- "1.91.1"
|
||||||
|
- "1.91.0"
|
||||||
12
Cargo.lock
generated
12
Cargo.lock
generated
@@ -30,6 +30,17 @@ dependencies = [
|
|||||||
"tokio",
|
"tokio",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "async-trait"
|
||||||
|
version = "0.1.89"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "9035ad2d096bed7955a320ee7e2230574d28fd3c3a0f186cbea1ff3c7eed5dbb"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"syn",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "atomic-waker"
|
name = "atomic-waker"
|
||||||
version = "1.1.2"
|
version = "1.1.2"
|
||||||
@@ -54,6 +65,7 @@ version = "0.1.0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-compression",
|
"async-compression",
|
||||||
|
"async-trait",
|
||||||
"bollard",
|
"bollard",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-tar",
|
"tokio-tar",
|
||||||
|
|||||||
@@ -9,3 +9,4 @@ async-compression = { version = "0.4", features = ["tokio", "xz"] }
|
|||||||
bollard = "0.19"
|
bollard = "0.19"
|
||||||
anyhow = "1.0"
|
anyhow = "1.0"
|
||||||
tokio-tar = "0.3"
|
tokio-tar = "0.3"
|
||||||
|
async-trait = "0.1"
|
||||||
11
Dockerfile
Normal file
11
Dockerfile
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
FROM rust:1.91.1
|
||||||
|
|
||||||
|
WORKDIR /beekeper
|
||||||
|
|
||||||
|
COPY src/ src/
|
||||||
|
COPY Cargo.toml .
|
||||||
|
COPY Cargo.lock .
|
||||||
|
|
||||||
|
RUN cargo build --release
|
||||||
|
|
||||||
|
CMD ["./target/release/beekeper"]
|
||||||
@@ -8,6 +8,9 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
- caddy_data:/data
|
- caddy_data:/data
|
||||||
- caddy_config:/config
|
- caddy_config:/config
|
||||||
|
labels:
|
||||||
|
- "beekeper.enable"
|
||||||
|
- "beekeper.pre_command=caddy stop"
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
caddy_data:
|
caddy_data:
|
||||||
38
scripts/dev.sh
Executable file
38
scripts/dev.sh
Executable file
@@ -0,0 +1,38 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# Couleurs pour les messages
|
||||||
|
GREEN='\033[0;32m'
|
||||||
|
RED='\033[0;31m'
|
||||||
|
YELLOW='\033[1;33m'
|
||||||
|
NC='\033[0m' # No Color
|
||||||
|
|
||||||
|
echo -e "${GREEN}🐝 Beekeeper Development Script${NC}"
|
||||||
|
|
||||||
|
# Vérifier si le fichier .env existe
|
||||||
|
if [ ! -f .env ]; then
|
||||||
|
echo -e "${RED}❌ Fichier .env introuvable${NC}"
|
||||||
|
echo -e "${YELLOW}Créez un fichier .env avec les variables nécessaires${NC}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Charger les variables d'environnement
|
||||||
|
echo -e "${GREEN}📋 Chargement des variables d'environnement depuis .env${NC}"
|
||||||
|
set -a
|
||||||
|
source .env
|
||||||
|
set +a
|
||||||
|
|
||||||
|
# Vérifier si cargo-watch est installé
|
||||||
|
if ! command -v cargo-watch &> /dev/null; then
|
||||||
|
echo -e "${YELLOW}⚠️ cargo-watch n'est pas installé${NC}"
|
||||||
|
echo -e "${YELLOW}Installation de cargo-watch...${NC}"
|
||||||
|
cargo install cargo-watch
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Lancer cargo watch avec hot reload
|
||||||
|
echo -e "${GREEN}🚀 Démarrage du programme avec hot reload${NC}"
|
||||||
|
echo -e "${YELLOW}Le programme redémarrera automatiquement lors de modifications${NC}"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
cargo watch -x run
|
||||||
72
src/docker/bollard.rs
Normal file
72
src/docker/bollard.rs
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
use bollard::{Docker, query_parameters::ListServicesOptions};
|
||||||
|
use std::sync::Arc;
|
||||||
|
use tokio::sync::Mutex;
|
||||||
|
|
||||||
|
pub struct BollardAdapter {
|
||||||
|
client: Arc<Mutex<Option<Docker>>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl BollardAdapter {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
client: Arc::new(Mutex::new(None)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl crate::docker::DockerAdapter for BollardAdapter {
|
||||||
|
async fn connect_docker(&self, docker_host: Option<String>) -> anyhow::Result<()> {
|
||||||
|
let docker = match docker_host {
|
||||||
|
Some(host) => Docker::connect_with_http(&host, 60, bollard::API_DEFAULT_VERSION)?,
|
||||||
|
None => Docker::connect_with_socket_defaults()?,
|
||||||
|
};
|
||||||
|
|
||||||
|
docker.ping().await?;
|
||||||
|
|
||||||
|
let mut client = self.client.lock().await;
|
||||||
|
*client = Some(docker);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn version(&self) -> anyhow::Result<String> {
|
||||||
|
let client = self.client.lock().await;
|
||||||
|
if let Some(docker) = &*client {
|
||||||
|
let version_info = docker.version().await?;
|
||||||
|
let version = version_info.version;
|
||||||
|
|
||||||
|
version.ok_or_else(|| anyhow::anyhow!("Version not available"))
|
||||||
|
} else {
|
||||||
|
Err(anyhow::anyhow!("Docker client is not connected"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn find_enabled_services(&self) -> anyhow::Result<Vec<String>> {
|
||||||
|
let client = self.client.lock().await;
|
||||||
|
let docker = client.as_ref().ok_or_else(|| anyhow::anyhow!("Docker client is not connected"))?;
|
||||||
|
|
||||||
|
let services = docker.list_services(None::<ListServicesOptions>).await?;
|
||||||
|
let mut enabled_services = Vec::new();
|
||||||
|
|
||||||
|
for service in services {
|
||||||
|
if let Some(spec) = service.spec {
|
||||||
|
let enabled = spec.labels
|
||||||
|
.map(|hash| hash.contains_key("beekeper.enable"))
|
||||||
|
.is_some();
|
||||||
|
|
||||||
|
if enabled && let Some(name) = spec.name {
|
||||||
|
enabled_services.push(name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(enabled_services)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for BollardAdapter {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::new()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
31
src/docker/mod.rs
Normal file
31
src/docker/mod.rs
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
mod bollard;
|
||||||
|
|
||||||
|
pub use bollard::BollardAdapter;
|
||||||
|
|
||||||
|
pub struct Docker<T> where T: DockerAdapter {
|
||||||
|
adapter: T,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Docker<T> where T: DockerAdapter {
|
||||||
|
pub fn new(adapter: T) -> Self {
|
||||||
|
Self { adapter }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn connect_docker(&self, docker_host: Option<String>) -> anyhow::Result<()> {
|
||||||
|
self.adapter.connect_docker(docker_host).await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn version(&self) -> anyhow::Result<String> {
|
||||||
|
self.adapter.version().await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn find_enabled_services(&self) -> anyhow::Result<Vec<String>> {
|
||||||
|
self.adapter.find_enabled_services().await
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait DockerAdapter {
|
||||||
|
async fn connect_docker(&self, docker_host: Option<String>) -> anyhow::Result<()>;
|
||||||
|
async fn version(&self) -> anyhow::Result<String>;
|
||||||
|
async fn find_enabled_services(&self) -> anyhow::Result<Vec<String>>;
|
||||||
|
}
|
||||||
43
src/main.rs
43
src/main.rs
@@ -1,19 +1,36 @@
|
|||||||
use bollard::Docker;
|
mod docker;
|
||||||
|
|
||||||
|
use docker::{BollardAdapter, Docker};
|
||||||
|
|
||||||
|
struct AppEnvVars {
|
||||||
|
backup_folder: String,
|
||||||
|
docker_host: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() -> anyhow::Result<()> {
|
async fn main() -> anyhow::Result<()> {
|
||||||
let backup_root_folder = std::env::var("BACKUP_FOLDER")?;
|
let envs = load_envs()?;
|
||||||
let backup_root_path = std::path::Path::new(&backup_root_folder);
|
|
||||||
|
|
||||||
let folders = list_folders(backup_root_path).await?;
|
let backup_root_path = std::path::Path::new(&envs.backup_folder);
|
||||||
|
let _folders = list_folders(backup_root_path).await?;
|
||||||
|
|
||||||
let docker_conn = Docker::connect_with_socket_defaults()?;
|
let docker_conn = connect_docker(envs.docker_host).await?;
|
||||||
let docker_version = docker_conn.version().await?;
|
println!("Docker version: {}", docker_conn.version().await?);
|
||||||
println!("Docker version: {:?}", docker_version);
|
|
||||||
|
let enabled_services = docker_conn.find_enabled_services().await?;
|
||||||
|
println!("Enabled services: {:?}", enabled_services);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn connect_docker(docker_host: Option<String>) -> anyhow::Result<Docker<BollardAdapter>> {
|
||||||
|
let adapter = BollardAdapter::new();
|
||||||
|
let docker = Docker::new(adapter);
|
||||||
|
docker.connect_docker(docker_host).await?;
|
||||||
|
|
||||||
|
Ok(docker)
|
||||||
|
}
|
||||||
|
|
||||||
async fn list_folders(path: &std::path::Path) -> anyhow::Result<Vec<std::path::PathBuf>> {
|
async fn list_folders(path: &std::path::Path) -> anyhow::Result<Vec<std::path::PathBuf>> {
|
||||||
let mut folders = Vec::new();
|
let mut folders = Vec::new();
|
||||||
let mut dir_entries = tokio::fs::read_dir(path).await?;
|
let mut dir_entries = tokio::fs::read_dir(path).await?;
|
||||||
@@ -26,3 +43,15 @@ async fn list_folders(path: &std::path::Path) -> anyhow::Result<Vec<std::path::P
|
|||||||
|
|
||||||
Ok(folders)
|
Ok(folders)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn load_envs() -> anyhow::Result<AppEnvVars> {
|
||||||
|
let backup_folder = std::env::var("BACKUP_FOLDER")?;
|
||||||
|
let docker_host = std::env::var("DOCKER_HOST").ok();
|
||||||
|
|
||||||
|
Ok(
|
||||||
|
AppEnvVars {
|
||||||
|
backup_folder,
|
||||||
|
docker_host,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user