This commit is contained in:
Kasper Juul Hermansen 2023-04-02 23:23:50 +02:00
commit 79ef3ecc43
Signed by: kjuulh
GPG Key ID: 57B6E1465221F912
8 changed files with 3501 additions and 0 deletions

3
.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
/target
.env
target/

1067
Cargo.lock generated Normal file

File diff suppressed because it is too large Load Diff

23
Cargo.toml Normal file
View File

@ -0,0 +1,23 @@
[package]
name = "update-deployment"
version = "0.1.0"
edition = "2021"
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
[dependencies]
clap = "4.2.1"
color-eyre = "0.6.2"
dotenv = "0.15.0"
eyre = "0.6.8"
git2 = { version = "0.16.1", features = [
"vendored-libgit2",
"vendored-openssl",
] }
serde = { version = "1.0.159", features = ["derive"] }
serde_json = "1.0.95"
serde_yaml = "0.9.19"
tempdir = "0.3.7"
tokio = { version = "1.27.0", features = ["full"] }
tracing = { version = "0.1.37", features = ["log"] }
tracing-subscriber = "0.3.16"

9
build.sh Executable file
View File

@ -0,0 +1,9 @@
#!/bin/bash
pushd ci || exit
cargo build
popd || exit
./ci/target/debug/ci

2157
ci/Cargo.lock generated Normal file

File diff suppressed because it is too large Load Diff

14
ci/Cargo.toml Normal file
View File

@ -0,0 +1,14 @@
[package]
name = "ci"
version = "0.1.0"
edition = "2021"
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
[dependencies]
chrono = "0.4.24"
color-eyre = "0.6.2"
dagger-sdk = "0.2.19"
eyre = "0.6.8"
tokio = { version = "1.27.0", features = ["full"] }
tokio-scoped = "0.2.0"

84
ci/src/main.rs Normal file
View File

@ -0,0 +1,84 @@
use std::sync::Arc;
use dagger_sdk::{HostDirectoryOptsBuilder, QueryContainerOptsBuilder};
use tokio::sync::Mutex;
#[tokio::main]
async fn main() -> eyre::Result<()> {
color_eyre::install().unwrap();
let client = dagger_sdk::connect().await?;
let src = client.host().directory_opts(
".",
HostDirectoryOptsBuilder::default()
.exclude(vec!["target/", ".git/"])
.build()?,
);
let variants = vec!["linux/amd64", "linux/arm64"];
let platform_variants = Arc::new(Mutex::new(Vec::new()));
tokio_scoped::scope(|s| {
for platform in variants {
let client = client.clone();
let platform_variants = platform_variants.clone();
let src = src.clone();
s.spawn(async move {
let rust_dep_image = client
.container_opts(
QueryContainerOptsBuilder::default()
.platform(platform)
.build()
.unwrap(),
)
.from("rustlang/rust:nightly")
.with_workdir("/app")
.with_directory(".", src.id().await.unwrap())
.with_exec(vec!["cargo", "build", "--release"]);
let dep_image = client
.container_opts(
QueryContainerOptsBuilder::default()
.platform(platform)
.build()
.unwrap(),
)
.from("debian:bullseye")
.with_file(
"/usr/bin/update-deployment",
rust_dep_image
.file("target/release/update-deployment")
.id()
.await
.unwrap(),
);
let mut platform_variants = platform_variants.lock().await;
platform_variants.push(dep_image.id().await.unwrap())
});
}
});
let variants = platform_variants
.lock()
.await
.iter()
.map(|c| c.clone())
.collect::<Vec<_>>();
let tag = chrono::Utc::now().timestamp();
let _ = client
.container()
.publish_opts(
format!("kasperhermansen/update-deployment:{tag}"),
dagger_sdk::ContainerPublishOptsBuilder::default()
.platform_variants(variants)
.build()?,
)
.await?;
Ok(())
}

144
src/main.rs Normal file
View File

@ -0,0 +1,144 @@
use git2::build::{CheckoutBuilder, RepoBuilder};
use git2::{Cred, FetchOptions, PushOptions, RemoteCallbacks};
use serde::{Deserialize, Serialize};
use serde_yaml::Value;
use tempdir::TempDir;
#[tokio::main]
async fn main() -> eyre::Result<()> {
dotenv::dotenv()?;
color_eyre::install().unwrap();
tracing_subscriber::fmt().pretty().init();
let matches = clap::Command::new("update-deployment")
.arg(clap::Arg::new("repo").long("repo").required(true))
.arg(clap::Arg::new("service").long("service").required(true))
.arg(clap::Arg::new("image").long("image").required(true))
.arg(clap::Arg::new("path").long("path"))
.get_matches();
let repo = matches.get_one::<String>("repo").unwrap();
let service = matches.get_one::<String>("service").unwrap();
let image = matches.get_one::<String>("image").unwrap();
let docker_compose_path = matches.get_one::<String>("path");
tracing::info!(
repo = repo,
service = service,
image = image,
path = docker_compose_path,
"updating deployment"
);
let tmpdir = TempDir::new("update-deployment")?;
let tmpdir = tmpdir.path();
tracing::info!(
repo = repo,
dest_dir = tmpdir.display().to_string(),
"clone repo"
);
let mut cb = RemoteCallbacks::new();
cb.credentials(|_, _, _| {
let username = std::env::var("GIT_USERNAME").expect("GIT_USERNAME to be set");
let password = std::env::var("GIT_PASSWORD").expect("GIT_PASSWORD to be set");
Cred::userpass_plaintext(&username, &password)
});
let co = CheckoutBuilder::new();
let mut fo = FetchOptions::new();
fo.remote_callbacks(cb);
RepoBuilder::new()
.fetch_options(fo)
.with_checkout(co)
.clone(repo, tmpdir)?;
let mut repo_dir = tmpdir.to_path_buf();
repo_dir.push("repo");
let repo = git2::Repository::clone(repo, &repo_dir)?;
let dir = std::fs::read_dir(&repo_dir)?;
for file in dir {
println!("file: {}", file?.file_name().to_str().unwrap());
}
let (docker_compose_raw, docker_compose_path) =
if let Some(docker_compose_path) = docker_compose_path {
let mut path = repo_dir.clone();
path.push(docker_compose_path);
(tokio::fs::read(&path).await?, path)
} else {
let mut path = repo_dir.clone();
path.push("docker-compose.yml");
(tokio::fs::read(&path).await?, path)
};
let mut docker_compose: DockerCompose = serde_yaml::from_slice(docker_compose_raw.as_slice())?;
tracing::info!(
docker_compose = serde_json::to_string(&docker_compose)?,
"found docker-compose.yml"
);
let service = docker_compose.services.get_mut(service);
if let Some(service) = service {
if let Some(img) = service.image.as_mut() {
*img = image.clone();
}
}
let dest_docker_compose = serde_yaml::to_string(&docker_compose)?;
tokio::fs::write(docker_compose_path, dest_docker_compose).await?;
let tree_id = {
let mut index = repo.index()?;
index.add_all(&["."], git2::IndexAddOption::DEFAULT, None)?;
index.write()?;
index.write_tree()?
};
let sig = repo.signature()?;
let tree = repo.find_tree(tree_id)?;
let head = repo.head()?;
let commit = head.peel_to_commit()?;
let _ = repo.commit(
Some("HEAD"),
&sig,
&sig,
"update: docker-compose",
&tree,
&[&commit],
)?;
let mut remote = repo.find_remote("origin")?;
let mut cb = RemoteCallbacks::new();
cb.credentials(|_, _, _| {
let username = std::env::var("GIT_USERNAME").expect("GIT_USERNAME to be set");
let password = std::env::var("GIT_PASSWORD").expect("GIT_PASSWORD to be set");
Cred::userpass_plaintext(&username, &password)
});
remote.push(
&["HEAD:refs/heads/main"],
Some(PushOptions::new().remote_callbacks(cb)),
)?;
Ok(())
}
use std::collections::BTreeMap as Map;
#[derive(Clone, Debug, Serialize, Deserialize)]
struct DockerCompose {
#[serde(flatten)]
pub other: Map<String, Value>,
pub services: Map<String, Service>,
}
#[derive(Clone, Debug, Serialize, Deserialize)]
struct Service {
pub image: Option<String>,
#[serde(flatten)]
pub other: Map<String, Value>,
}