Compare commits
54 Commits
Author | SHA1 | Date | |
---|---|---|---|
545e8c5476 | |||
42f23fdfac | |||
249a40f1aa | |||
c1187022f2 | |||
0fc1438a4a | |||
166be0c289 | |||
|
ff9f5e25d2 | ||
9ac744b39d | |||
9c967a0f31 | |||
3878e6bc0a | |||
|
9999fca9b0 | ||
dad8fc472e | |||
05ecdf5251 | |||
12063f7c23 | |||
490130126b | |||
9201ff9294 | |||
477d82af55 | |||
d94b9cbe86 | |||
2b277ec61f | |||
3f2642aed0 | |||
44bc26de93 | |||
013bf3b3dc | |||
497ae0f19d | |||
2dcd9d0cd1 | |||
a0634a542b | |||
036998b0b9 | |||
e92284002a | |||
69f64bdab2 | |||
f624109643 | |||
2404a14a32 | |||
9a5396a81e | |||
ae0a54db69 | |||
8e36f2a3ca | |||
|
384e575758 | ||
850ada11c2 | |||
56d33e2ca5 | |||
d64a1d15dc | |||
c2e0b548f6 | |||
51ca73a53b | |||
675947ed1e | |||
bf3593eee4 | |||
19d748702a | |||
4276f4529c | |||
d287a54cdf | |||
7baf51c1f2 | |||
56b44cf2e2 | |||
2919ca9a04 | |||
ff2b59dd02 | |||
19dd0ff636 | |||
c08918ad6f | |||
19e7adfedb | |||
27cb31f433 | |||
113e5282ef | |||
fa67dfeee3 |
172
.drone.yml
172
.drone.yml
@ -1,170 +1,2 @@
|
|||||||
kind: pipeline
|
kind: template
|
||||||
name: default
|
load: cuddle-rust-cli-plan.yaml
|
||||||
type: docker
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: build ci
|
|
||||||
image: rustlang/rust:nightly
|
|
||||||
volumes:
|
|
||||||
- name: ci
|
|
||||||
path: /mnt/ci
|
|
||||||
environment:
|
|
||||||
PKG_CONFIG_SYSROOT_DIR: "/"
|
|
||||||
CI_PREFIX: "/mnt/ci"
|
|
||||||
commands:
|
|
||||||
- set -e
|
|
||||||
- apt update
|
|
||||||
- apt install musl-tools pkg-config libssl-dev openssl build-essential musl-dev -y
|
|
||||||
- rustup target add x86_64-unknown-linux-musl
|
|
||||||
- cargo build --target=x86_64-unknown-linux-musl -p ci
|
|
||||||
#- cargo build -p ci
|
|
||||||
- mv target/x86_64-unknown-linux-musl/debug/ci "$CI_PREFIX/ci"
|
|
||||||
#- mv target/debug/ci $CI_PREFIX/ci
|
|
||||||
|
|
||||||
- name: load_secret
|
|
||||||
image: debian:buster-slim
|
|
||||||
volumes:
|
|
||||||
- name: ssh
|
|
||||||
path: /root/.ssh/
|
|
||||||
environment:
|
|
||||||
SSH_KEY:
|
|
||||||
from_secret: gitea_id_ed25519
|
|
||||||
commands:
|
|
||||||
- mkdir -p $HOME/.ssh/
|
|
||||||
- echo "$SSH_KEY" | base64 -d > $HOME/.ssh/id_ed25519
|
|
||||||
|
|
||||||
- name: build pr
|
|
||||||
image: kasperhermansen/cuddle:latest
|
|
||||||
pull: always
|
|
||||||
volumes:
|
|
||||||
- name: ssh
|
|
||||||
path: /root/.ssh/
|
|
||||||
- name: dockersock
|
|
||||||
path: /var/run
|
|
||||||
- name: ci
|
|
||||||
path: /mnt/ci
|
|
||||||
commands:
|
|
||||||
- eval `ssh-agent`
|
|
||||||
- chmod -R 600 ~/.ssh
|
|
||||||
- ssh-add
|
|
||||||
- echo "$DOCKER_PASSWORD" | docker login --password-stdin --username="$DOCKER_USERNAME" docker.io
|
|
||||||
- ldd $CI_PREFIX
|
|
||||||
- apk add git
|
|
||||||
- cuddle x ci:pr
|
|
||||||
environment:
|
|
||||||
DOCKER_BUILDKIT: 1
|
|
||||||
DOCKER_PASSWORD:
|
|
||||||
from_secret: docker_password
|
|
||||||
DOCKER_USERNAME:
|
|
||||||
from_secret: docker_username
|
|
||||||
CUDDLE_SECRETS_PROVIDER: 1password
|
|
||||||
CUDDLE_ONE_PASSWORD_DOT_ENV: ".env.ci"
|
|
||||||
CUDDLE_SSH_AGENT: "true"
|
|
||||||
CI_PREFIX: "/mnt/ci/ci"
|
|
||||||
CUDDLE_PLEASE_TOKEN:
|
|
||||||
from_secret: cuddle_please_token
|
|
||||||
OP_SERVICE_ACCOUNT_TOKEN:
|
|
||||||
from_secret: op_service_account_token
|
|
||||||
when:
|
|
||||||
event:
|
|
||||||
- pull_request
|
|
||||||
exclude:
|
|
||||||
- main
|
|
||||||
- master
|
|
||||||
depends_on:
|
|
||||||
- "load_secret"
|
|
||||||
- "build ci"
|
|
||||||
|
|
||||||
- name: build main
|
|
||||||
image: kasperhermansen/cuddle:latest
|
|
||||||
pull: always
|
|
||||||
volumes:
|
|
||||||
- name: ssh
|
|
||||||
path: /root/.ssh/
|
|
||||||
- name: dockersock
|
|
||||||
path: /var/run
|
|
||||||
- name: ci
|
|
||||||
path: /mnt/ci
|
|
||||||
commands:
|
|
||||||
- eval `ssh-agent`
|
|
||||||
- chmod -R 600 ~/.ssh
|
|
||||||
- ssh-add
|
|
||||||
- echo "$DOCKER_PASSWORD" | docker login --password-stdin --username="$DOCKER_USERNAME" docker.io
|
|
||||||
- ldd $CI_PREFIX
|
|
||||||
- apk add git
|
|
||||||
- cuddle x ci:main
|
|
||||||
environment:
|
|
||||||
DOCKER_BUILDKIT: 1
|
|
||||||
DOCKER_PASSWORD:
|
|
||||||
from_secret: docker_password
|
|
||||||
DOCKER_USERNAME:
|
|
||||||
from_secret: docker_username
|
|
||||||
CUDDLE_SECRETS_PROVIDER: 1password
|
|
||||||
CUDDLE_ONE_PASSWORD_DOT_ENV: ".env.ci"
|
|
||||||
CUDDLE_SSH_AGENT: "true"
|
|
||||||
CI_PREFIX: "/mnt/ci/ci"
|
|
||||||
CUDDLE_PLEASE_TOKEN:
|
|
||||||
from_secret: cuddle_please_token
|
|
||||||
OP_SERVICE_ACCOUNT_TOKEN:
|
|
||||||
from_secret: op_service_account_token
|
|
||||||
when:
|
|
||||||
event:
|
|
||||||
- push
|
|
||||||
branch:
|
|
||||||
- main
|
|
||||||
- master
|
|
||||||
exclude:
|
|
||||||
- pull_request
|
|
||||||
depends_on:
|
|
||||||
- "load_secret"
|
|
||||||
- "build ci"
|
|
||||||
|
|
||||||
- name: deploy release
|
|
||||||
image: kasperhermansen/cuddle:latest
|
|
||||||
pull: always
|
|
||||||
volumes:
|
|
||||||
- name: ssh
|
|
||||||
path: /root/.ssh/
|
|
||||||
- name: dockersock
|
|
||||||
path: /var/run
|
|
||||||
commands:
|
|
||||||
- eval `ssh-agent`
|
|
||||||
- chmod -R 600 ~/.ssh
|
|
||||||
- ssh-add
|
|
||||||
- cuddle x build:release:all
|
|
||||||
- cuddle x deploy:docs:preview
|
|
||||||
environment:
|
|
||||||
DOCKER_BUILDKIT: 1
|
|
||||||
CUDDLE_SECRETS_PROVIDER: 1password
|
|
||||||
CUDDLE_ONE_PASSWORD_DOT_ENV: ".env.ci"
|
|
||||||
CUDDLE_SSH_AGENT: "true"
|
|
||||||
CUDDLE_CI: "true"
|
|
||||||
CUDDLE_PLEASE_TOKEN:
|
|
||||||
from_secret: cuddle_please_token
|
|
||||||
OP_SERVICE_ACCOUNT_TOKEN:
|
|
||||||
from_secret: op_service_account_token
|
|
||||||
when:
|
|
||||||
event:
|
|
||||||
- tag
|
|
||||||
ref:
|
|
||||||
include:
|
|
||||||
- refs/tags/v*
|
|
||||||
depends_on:
|
|
||||||
- "load_secret"
|
|
||||||
- "build ci"
|
|
||||||
|
|
||||||
services:
|
|
||||||
- name: docker
|
|
||||||
image: docker:dind
|
|
||||||
privileged: true
|
|
||||||
volumes:
|
|
||||||
- name: dockersock
|
|
||||||
path: /var/run
|
|
||||||
|
|
||||||
volumes:
|
|
||||||
- name: ssh
|
|
||||||
temp: {}
|
|
||||||
- name: dockersock
|
|
||||||
temp: {}
|
|
||||||
- name: ci
|
|
||||||
temp: {}
|
|
||||||
|
66
CHANGELOG.md
66
CHANGELOG.md
@ -6,6 +6,72 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
## [0.5.0] - 2024-04-09
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- stuff
|
||||||
|
- with rust version
|
||||||
|
- add rust actions
|
||||||
|
|
||||||
|
## [0.4.0] - 2024-04-08
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- remove comment
|
||||||
|
- add jq
|
||||||
|
- fix tests
|
||||||
|
- update deps
|
||||||
|
- update chrono
|
||||||
|
- remove deps
|
||||||
|
- without dagger
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
- *(deps)* update rust crate futures to 0.3.30
|
||||||
|
- *(deps)* update rust crate futures to 0.3.29
|
||||||
|
|
||||||
|
### Other
|
||||||
|
- *(deps)* update all dependencies
|
||||||
|
- something
|
||||||
|
|
||||||
|
- *(deps)* update rust crate chrono to 0.4.28
|
||||||
|
- *(deps)* update rust crate chrono to 0.4.27
|
||||||
|
- *(deps)* update rust crate clap to 4.4.1
|
||||||
|
- *(deps)* update rust crate url to 2.4.1
|
||||||
|
- *(deps)* update rust crate regex to 1.9.4
|
||||||
|
- *(deps)* update rust crate clap to 4.4.0
|
||||||
|
- *(deps)* update rust crate reqwest to 0.11.20
|
||||||
|
- *(deps)* update rust crate clap to 4.3.24
|
||||||
|
- *(deps)* update rust crate reqwest to 0.11.19
|
||||||
|
- *(deps)* update rust crate clap to 4.3.23
|
||||||
|
- *(deps)* update all dependencies
|
||||||
|
|
||||||
|
## [0.3.0] - 2023-08-13
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- *(ci)* with dagger-rust components
|
||||||
|
- allow v in start of versions
|
||||||
|
- *(json-edit)* added json-edit to update some json content with next global version
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
- *(git)* make sure we always fail on exit code != 0
|
||||||
|
- *(json-edit)* with actual arg instead of stupid str replace
|
||||||
|
- *(ci)* without token
|
||||||
|
- *(docs)* check fix version
|
||||||
|
- *(crate)* initial pr always included the entire changelog
|
||||||
|
- *(crate)* always prefix with 'v' when creating prs, or releases (#9)
|
||||||
|
|
||||||
|
### Other
|
||||||
|
- remove unnused arguments
|
||||||
|
- *(ci)* fix release step
|
||||||
|
- add dagger-rust and dagger-cuddle-please
|
||||||
|
- *(deps)* update rust crate clap to 4.3.21
|
||||||
|
- *(deps)* update rust crate clap to 4.3.20
|
||||||
|
- *(deps)* update rust crate parse-changelog to 0.6.2
|
||||||
|
- *(deps)* update rust crate regex to 1.9.3
|
||||||
|
- *(deps)* update rust crate regex to 1.9.2
|
||||||
|
- remove cr
|
||||||
|
- *(json-edit)* clarify errors
|
||||||
|
- *(docs)* remove 0.2 checklist
|
||||||
|
|
||||||
## [0.2.1] - 2023-08-04
|
## [0.2.1] - 2023-08-04
|
||||||
|
|
||||||
### Docs
|
### Docs
|
||||||
|
1910
Cargo.lock
generated
1910
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
35
Cargo.toml
35
Cargo.toml
@ -1,10 +1,6 @@
|
|||||||
[workspace]
|
[workspace]
|
||||||
members = [
|
members = [
|
||||||
"crates/cuddle-please",
|
"crates/*"
|
||||||
"crates/cuddle-please-frontend",
|
|
||||||
"crates/cuddle-please-commands",
|
|
||||||
"crates/cuddle-please-misc",
|
|
||||||
"ci"
|
|
||||||
]
|
]
|
||||||
resolver = "2"
|
resolver = "2"
|
||||||
|
|
||||||
@ -13,24 +9,31 @@ cuddle-please = { path = "crates/cuddle-please", version = "0.1.0" }
|
|||||||
cuddle-please-frontend = { path = "crates/cuddle-please-frontend", version = "0.1.0" }
|
cuddle-please-frontend = { path = "crates/cuddle-please-frontend", version = "0.1.0" }
|
||||||
cuddle-please-commands = { path = "crates/cuddle-please-commands", version = "0.1.0" }
|
cuddle-please-commands = { path = "crates/cuddle-please-commands", version = "0.1.0" }
|
||||||
cuddle-please-misc = { path = "crates/cuddle-please-misc", version = "0.1.0" }
|
cuddle-please-misc = { path = "crates/cuddle-please-misc", version = "0.1.0" }
|
||||||
|
cuddle-please-release-strategy = { path = "crates/cuddle-please-release-strategy", version = "0.1.0" }
|
||||||
|
cuddle-please-actions = { path = "crates/cuddle-please-actions", version = "0.1.0" }
|
||||||
|
|
||||||
anyhow = { version = "1.0.72" }
|
anyhow = { version = "1.0.81" }
|
||||||
tracing = { version = "0.1", features = ["log"] }
|
tracing = { version = "0.1", features = ["log"] }
|
||||||
tracing-subscriber = { version = "0.3.17", features = ["env-filter"] }
|
tracing-subscriber = { version = "0.3.18", features = ["env-filter"] }
|
||||||
clap = { version = "4.3.19", features = ["derive", "env"] }
|
clap = { version = "4.5.4", features = ["derive", "env"] }
|
||||||
dotenv = { version = "0.15.0" }
|
dotenv = { version = "0.15.0" }
|
||||||
url = { version = "2.4.0" }
|
url = { version = "2.5.0" }
|
||||||
serde_yaml = { version = "0.9.25" }
|
serde_yaml = { version = "0.9.34+deprecated" }
|
||||||
|
yaml-rust2 = {version = "0.8.0"}
|
||||||
serde = { version = "1", features = ["derive"] }
|
serde = { version = "1", features = ["derive"] }
|
||||||
semver = "1.0.18"
|
semver = "1.0.22"
|
||||||
conventional_commit_parser = "0.9.4"
|
conventional_commit_parser = "0.9.4"
|
||||||
tempdir = "0.3.7"
|
tempdir = "0.3.7"
|
||||||
reqwest = { version = "0.11.18" }
|
reqwest = { version = "0.12.3" }
|
||||||
git-cliff-core = "1.2.0"
|
git-cliff-core = "2.2.0"
|
||||||
regex = "1.9.1"
|
regex = "1.10.4"
|
||||||
chrono = "0.4.26"
|
chrono = "0.4.37"
|
||||||
lazy_static = "1.4.0"
|
lazy_static = "1.4.0"
|
||||||
parse-changelog = "0.6.1"
|
parse-changelog = "0.6.6"
|
||||||
|
toml_edit = "0.22.9"
|
||||||
|
|
||||||
tracing-test = "0.2"
|
tracing-test = "0.2"
|
||||||
pretty_assertions = "1.4"
|
pretty_assertions = "1.4"
|
||||||
|
|
||||||
|
[workspace.package]
|
||||||
|
version = "0.5.0"
|
||||||
|
14
README.md
14
README.md
@ -36,22 +36,14 @@ See docs for more information about installation and some such
|
|||||||
|
|
||||||
## Checklist
|
## Checklist
|
||||||
|
|
||||||
### 0.2 Milestone
|
|
||||||
|
|
||||||
- [x] Add docs
|
|
||||||
- [ ] Add asciinema
|
|
||||||
- [x] Create docker image
|
|
||||||
- [x] Add examples
|
|
||||||
- [x] Fx drone config
|
|
||||||
- [x] Releaser
|
|
||||||
- [x] On main/master
|
|
||||||
- [ ] tbd...
|
|
||||||
|
|
||||||
### 0.3 Milestone
|
### 0.3 Milestone
|
||||||
|
|
||||||
|
- [x] Fix: 0.0.0 -> **v**0.0.0
|
||||||
|
- [ ] Add release strategies
|
||||||
- [ ] Add reporter for PR and Repositories
|
- [ ] Add reporter for PR and Repositories
|
||||||
- [ ] Add inquire for missing values when needed (when not running in ci or have a proper tty)
|
- [ ] Add inquire for missing values when needed (when not running in ci or have a proper tty)
|
||||||
- [ ] Break down cuddle-please-misc
|
- [ ] Break down cuddle-please-misc
|
||||||
|
- [ ] ci(release): Add cuddle-please release artifacts for the different os and so on.
|
||||||
|
|
||||||
### 0.x Milestone
|
### 0.x Milestone
|
||||||
- [ ] Add github support
|
- [ ] Add github support
|
||||||
|
1861
ci/Cargo.lock
generated
1861
ci/Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
@ -1,16 +0,0 @@
|
|||||||
[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]
|
|
||||||
dagger-sdk = "*"
|
|
||||||
eyre = "*"
|
|
||||||
color-eyre = "*"
|
|
||||||
tokio = "1"
|
|
||||||
clap = {version = "4", features = ["derive"]}
|
|
||||||
futures = "0.3.28"
|
|
||||||
async-scoped = { version = "0.7.1", features = ["tokio", "use-tokio"] }
|
|
||||||
dotenv.workspace = true
|
|
770
ci/src/main.rs
770
ci/src/main.rs
@ -1,770 +0,0 @@
|
|||||||
use std::path::Path;
|
|
||||||
use std::path::PathBuf;
|
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use clap::Args;
|
|
||||||
use clap::Parser;
|
|
||||||
use clap::Subcommand;
|
|
||||||
use clap::ValueEnum;
|
|
||||||
|
|
||||||
use dagger_sdk::Platform;
|
|
||||||
use dagger_sdk::QueryContainerOpts;
|
|
||||||
use futures::StreamExt;
|
|
||||||
|
|
||||||
use crate::please_release::run_release_please;
|
|
||||||
|
|
||||||
#[derive(Parser, Clone)]
|
|
||||||
#[command(author, version, about, long_about = None, subcommand_required = true)]
|
|
||||||
pub struct Command {
|
|
||||||
#[command(subcommand)]
|
|
||||||
commands: Commands,
|
|
||||||
|
|
||||||
#[command(flatten)]
|
|
||||||
global: GlobalArgs,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Subcommand, Clone)]
|
|
||||||
pub enum Commands {
|
|
||||||
#[command(subcommand_required = true)]
|
|
||||||
Local {
|
|
||||||
#[command(subcommand)]
|
|
||||||
command: LocalCommands,
|
|
||||||
},
|
|
||||||
PullRequest {
|
|
||||||
#[arg(long)]
|
|
||||||
image: String,
|
|
||||||
#[arg(long)]
|
|
||||||
tag: String,
|
|
||||||
#[arg(long)]
|
|
||||||
bin_name: String,
|
|
||||||
},
|
|
||||||
Main {
|
|
||||||
#[arg(long)]
|
|
||||||
image: String,
|
|
||||||
#[arg(long)]
|
|
||||||
tag: String,
|
|
||||||
#[arg(long)]
|
|
||||||
bin_name: String,
|
|
||||||
},
|
|
||||||
Release,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Subcommand, Clone)]
|
|
||||||
pub enum LocalCommands {
|
|
||||||
Build {
|
|
||||||
#[arg(long, default_value = "debug")]
|
|
||||||
profile: BuildProfile,
|
|
||||||
#[arg(long)]
|
|
||||||
bin_name: String,
|
|
||||||
},
|
|
||||||
Test,
|
|
||||||
DockerImage {
|
|
||||||
#[arg(long)]
|
|
||||||
image: String,
|
|
||||||
#[arg(long)]
|
|
||||||
tag: String,
|
|
||||||
#[arg(long)]
|
|
||||||
bin_name: String,
|
|
||||||
},
|
|
||||||
PleaseRelease,
|
|
||||||
BuildDocs {},
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, ValueEnum)]
|
|
||||||
pub enum BuildProfile {
|
|
||||||
Debug,
|
|
||||||
Release,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, Args)]
|
|
||||||
pub struct GlobalArgs {
|
|
||||||
#[arg(long, global = true, help_heading = "Global")]
|
|
||||||
dry_run: bool,
|
|
||||||
|
|
||||||
#[arg(long, global = true, help_heading = "Global")]
|
|
||||||
rust_builder_image: Option<String>,
|
|
||||||
|
|
||||||
#[arg(long, global = true, help_heading = "Global")]
|
|
||||||
production_image: Option<String>,
|
|
||||||
|
|
||||||
#[arg(long, global = true, help_heading = "Global")]
|
|
||||||
mkdocs_image: Option<String>,
|
|
||||||
|
|
||||||
#[arg(long, global = true, help_heading = "Global")]
|
|
||||||
caddy_image: Option<String>,
|
|
||||||
|
|
||||||
#[arg(long, global = true, help_heading = "Global")]
|
|
||||||
source: Option<PathBuf>,
|
|
||||||
|
|
||||||
#[arg(long, global = true, help_heading = "Global")]
|
|
||||||
docs_image: Option<String>,
|
|
||||||
|
|
||||||
#[arg(long, global = true, help_heading = "Global")]
|
|
||||||
docs_image_tag: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tokio::main]
|
|
||||||
async fn main() -> eyre::Result<()> {
|
|
||||||
let _ = dotenv::dotenv();
|
|
||||||
let _ = color_eyre::install();
|
|
||||||
|
|
||||||
let client = dagger_sdk::connect().await?;
|
|
||||||
|
|
||||||
let cli = Command::parse();
|
|
||||||
|
|
||||||
match &cli.commands {
|
|
||||||
Commands::Local { command } => match command {
|
|
||||||
LocalCommands::Build {
|
|
||||||
profile: _,
|
|
||||||
bin_name,
|
|
||||||
} => {
|
|
||||||
let base_image = base_rust_image(
|
|
||||||
client.clone(),
|
|
||||||
&cli.global,
|
|
||||||
&None,
|
|
||||||
&bin_name.clone(),
|
|
||||||
&"release".into(),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
let prod_image = get_base_debian_image(client.clone(), &cli.global, None).await?;
|
|
||||||
build::execute(
|
|
||||||
client,
|
|
||||||
&cli.global,
|
|
||||||
&base_image,
|
|
||||||
&prod_image,
|
|
||||||
bin_name,
|
|
||||||
&None,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
}
|
|
||||||
LocalCommands::Test => {
|
|
||||||
let base_image = base_rust_image(
|
|
||||||
client.clone(),
|
|
||||||
&cli.global,
|
|
||||||
&None,
|
|
||||||
&"cuddle-please".into(),
|
|
||||||
&"debug".into(),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
test::execute(client, &cli.global, base_image).await?;
|
|
||||||
}
|
|
||||||
LocalCommands::DockerImage {
|
|
||||||
tag,
|
|
||||||
image,
|
|
||||||
bin_name,
|
|
||||||
} => {
|
|
||||||
build::build_and_deploy(client, &cli.global, bin_name, image, tag).await?;
|
|
||||||
}
|
|
||||||
LocalCommands::PleaseRelease => todo!(),
|
|
||||||
LocalCommands::BuildDocs {} => {
|
|
||||||
let _image = docs::execute(
|
|
||||||
client.clone(),
|
|
||||||
&cli.global,
|
|
||||||
&Some("linux/amd64".to_string()),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Commands::PullRequest {
|
|
||||||
image,
|
|
||||||
tag,
|
|
||||||
bin_name,
|
|
||||||
} => {
|
|
||||||
async fn test(client: Arc<dagger_sdk::Query>, cli: &Command, bin_name: &String) {
|
|
||||||
let args = &cli.global;
|
|
||||||
|
|
||||||
let base_image =
|
|
||||||
base_rust_image(client.clone(), args, &None, bin_name, &"debug".into())
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
test::execute(client.clone(), args, base_image)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
async fn build(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
cli: &Command,
|
|
||||||
bin_name: &String,
|
|
||||||
image: &String,
|
|
||||||
tag: &String,
|
|
||||||
) {
|
|
||||||
let args = &cli.global;
|
|
||||||
|
|
||||||
build::build(client.clone(), args, bin_name, image, tag)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
tokio::join!(
|
|
||||||
test(client.clone(), &cli, bin_name),
|
|
||||||
build(client.clone(), &cli, bin_name, image, tag),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
Commands::Main {
|
|
||||||
image,
|
|
||||||
tag,
|
|
||||||
bin_name,
|
|
||||||
} => {
|
|
||||||
async fn test(client: Arc<dagger_sdk::Query>, cli: &Command, bin_name: &String) {
|
|
||||||
let args = &cli.global;
|
|
||||||
|
|
||||||
let base_image =
|
|
||||||
base_rust_image(client.clone(), args, &None, bin_name, &"debug".into())
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
test::execute(client.clone(), args, base_image)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
async fn build(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
cli: &Command,
|
|
||||||
bin_name: &String,
|
|
||||||
image: &String,
|
|
||||||
tag: &String,
|
|
||||||
) {
|
|
||||||
let args = &cli.global;
|
|
||||||
|
|
||||||
build::build_and_deploy(client.clone(), args, bin_name, image, tag)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn cuddle_please(client: Arc<dagger_sdk::Query>, cli: &Command) {
|
|
||||||
run_release_please(client.clone(), &cli.global)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
tokio::join!(
|
|
||||||
test(client.clone(), &cli, bin_name),
|
|
||||||
build(client.clone(), &cli, bin_name, image, tag),
|
|
||||||
cuddle_please(client.clone(), &cli)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
Commands::Release => todo!(),
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
mod please_release {
|
|
||||||
|
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use crate::{base_rust_image, GlobalArgs};
|
|
||||||
|
|
||||||
pub async fn run_release_please(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
args: &GlobalArgs,
|
|
||||||
) -> eyre::Result<()> {
|
|
||||||
let base_image = base_rust_image(
|
|
||||||
client.clone(),
|
|
||||||
args,
|
|
||||||
&Some("linux/amd64".to_string()),
|
|
||||||
&"cuddle-please".to_string(),
|
|
||||||
&"release".into(),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
let build_image = base_image.with_exec(vec![
|
|
||||||
"cargo",
|
|
||||||
"install",
|
|
||||||
"--target",
|
|
||||||
"x86_64-unknown-linux-gnu",
|
|
||||||
"--path=crates/cuddle-please",
|
|
||||||
]);
|
|
||||||
|
|
||||||
let src = client
|
|
||||||
.git_opts(
|
|
||||||
"https://git.front.kjuulh.io/kjuulh/cuddle-please",
|
|
||||||
dagger_sdk::QueryGitOpts {
|
|
||||||
experimental_service_host: None,
|
|
||||||
keep_git_dir: Some(true),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.branch("main")
|
|
||||||
.tree();
|
|
||||||
|
|
||||||
let res = build_image
|
|
||||||
.with_secret_variable(
|
|
||||||
"CUDDLE_PLEASE_TOKEN",
|
|
||||||
client
|
|
||||||
.set_secret("CUDDLE_PLEASE_TOKEN", std::env::var("CUDDLE_PLEASE_TOKEN")?)
|
|
||||||
.id()
|
|
||||||
.await?,
|
|
||||||
)
|
|
||||||
.with_workdir("/mnt/app")
|
|
||||||
.with_directory(".", src.id().await?)
|
|
||||||
.with_exec(vec![
|
|
||||||
"git",
|
|
||||||
"remote",
|
|
||||||
"set-url",
|
|
||||||
"origin",
|
|
||||||
&format!(
|
|
||||||
"https://git:{}@git.front.kjuulh.io/kjuulh/cuddle-please.git",
|
|
||||||
std::env::var("CUDDLE_PLEASE_TOKEN")?
|
|
||||||
),
|
|
||||||
])
|
|
||||||
.with_exec(vec![
|
|
||||||
"git",
|
|
||||||
"config",
|
|
||||||
"http.extraheader",
|
|
||||||
"'Authorization: token b52c18cab8a95d33f34b0d081440f77a2b156886'",
|
|
||||||
])
|
|
||||||
.with_exec(vec![
|
|
||||||
"cuddle-please",
|
|
||||||
"release",
|
|
||||||
"--engine=gitea",
|
|
||||||
"--owner=kjuulh",
|
|
||||||
"--repo=cuddle-please",
|
|
||||||
"--branch=main",
|
|
||||||
"--api-url=https://git.front.kjuulh.io",
|
|
||||||
"--log-level=debug",
|
|
||||||
]);
|
|
||||||
|
|
||||||
let exit_code = res.exit_code().await?;
|
|
||||||
if exit_code != 0 {
|
|
||||||
eyre::bail!("failed to run cuddle-please");
|
|
||||||
}
|
|
||||||
|
|
||||||
let please_out = res.stdout().await?;
|
|
||||||
println!("{please_out}");
|
|
||||||
let please_out = res.stderr().await?;
|
|
||||||
println!("{please_out}");
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mod docs {
|
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use dagger_sdk::Container;
|
|
||||||
|
|
||||||
use crate::GlobalArgs;
|
|
||||||
|
|
||||||
pub fn get_docs_src(client: Arc<dagger_sdk::Query>) -> eyre::Result<dagger_sdk::Directory> {
|
|
||||||
let docs_content = client.host().directory_opts(
|
|
||||||
".",
|
|
||||||
dagger_sdk::HostDirectoryOpts {
|
|
||||||
exclude: None,
|
|
||||||
include: Some(vec!["mkdocs.yml", "docs/"]),
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
Ok(docs_content)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn execute(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
args: &GlobalArgs,
|
|
||||||
_platform: &Option<String>,
|
|
||||||
) -> eyre::Result<Container> {
|
|
||||||
let mkdocs_container = client.container().from(
|
|
||||||
args.mkdocs_image
|
|
||||||
.as_ref()
|
|
||||||
.expect("--mkdocs-image to be set"),
|
|
||||||
);
|
|
||||||
|
|
||||||
let built_mkdocs_container = mkdocs_container
|
|
||||||
.with_directory("/docs", get_docs_src(client.clone())?.id().await?)
|
|
||||||
.with_exec(vec!["build"]);
|
|
||||||
|
|
||||||
let site_output = built_mkdocs_container.directory("/docs/site").id().await?;
|
|
||||||
|
|
||||||
let caddy_file = client.host().directory("templates").file("Caddyfile");
|
|
||||||
|
|
||||||
let dep_image = client
|
|
||||||
.container()
|
|
||||||
.from(args.caddy_image.as_ref().expect("--caddy-image to be set"))
|
|
||||||
.with_directory("/usr/share/caddy", site_output)
|
|
||||||
.with_file("/etc/caddy/Caddyfile", caddy_file.id().await?)
|
|
||||||
.with_exec(vec!["echo", "caddy"]);
|
|
||||||
|
|
||||||
Ok(dep_image)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn publish(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
args: &GlobalArgs,
|
|
||||||
containers: &Vec<dagger_sdk::Container>,
|
|
||||||
) -> eyre::Result<()> {
|
|
||||||
let container_ids =
|
|
||||||
futures::future::join_all(containers.iter().map(|c| c.id()).collect::<Vec<_>>()).await;
|
|
||||||
|
|
||||||
let container_ids = container_ids
|
|
||||||
.into_iter()
|
|
||||||
.collect::<eyre::Result<Vec<dagger_sdk::ContainerId>>>()?;
|
|
||||||
|
|
||||||
client
|
|
||||||
.container()
|
|
||||||
.publish_opts(
|
|
||||||
format!(
|
|
||||||
"{}:{}",
|
|
||||||
args.docs_image.as_ref().expect("--docs-image to be set"),
|
|
||||||
args.docs_image_tag
|
|
||||||
.as_ref()
|
|
||||||
.expect("--docs-image-tag to be set")
|
|
||||||
),
|
|
||||||
dagger_sdk::ContainerPublishOpts {
|
|
||||||
platform_variants: Some(container_ids),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mod build {
|
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use dagger_sdk::Container;
|
|
||||||
|
|
||||||
use crate::{base_rust_image, get_base_debian_image, GlobalArgs};
|
|
||||||
|
|
||||||
pub async fn build_and_deploy(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
args: &GlobalArgs,
|
|
||||||
bin_name: &String,
|
|
||||||
image: &String,
|
|
||||||
tag: &String,
|
|
||||||
) -> eyre::Result<()> {
|
|
||||||
// let containers = vec!["linux/amd64", "linux/arm64"];
|
|
||||||
|
|
||||||
let base_image = get_base_debian_image(
|
|
||||||
client.clone(),
|
|
||||||
&args.clone(),
|
|
||||||
Some("linux/amd64".to_string()),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
let container = base_rust_image(
|
|
||||||
client.clone(),
|
|
||||||
args,
|
|
||||||
&Some("linux/amd64".to_string()),
|
|
||||||
&bin_name.clone(),
|
|
||||||
&"release".into(),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
let build_image = execute(
|
|
||||||
client.clone(),
|
|
||||||
args,
|
|
||||||
&container,
|
|
||||||
&base_image,
|
|
||||||
bin_name,
|
|
||||||
&Some("linux/amd64".to_string()),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
let build_id = build_image.id().await?;
|
|
||||||
|
|
||||||
let _container = client
|
|
||||||
.clone()
|
|
||||||
.container()
|
|
||||||
.publish_opts(
|
|
||||||
format!("{image}:{tag}"),
|
|
||||||
dagger_sdk::ContainerPublishOpts {
|
|
||||||
platform_variants: Some(vec![build_id]),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
pub async fn build(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
args: &GlobalArgs,
|
|
||||||
bin_name: &String,
|
|
||||||
_image: &String,
|
|
||||||
_tag: &String,
|
|
||||||
) -> eyre::Result<()> {
|
|
||||||
// let containers = vec!["linux/amd64", "linux/arm64"];
|
|
||||||
|
|
||||||
let base_image = get_base_debian_image(
|
|
||||||
client.clone(),
|
|
||||||
&args.clone(),
|
|
||||||
Some("linux/amd64".to_string()),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
let container = base_rust_image(
|
|
||||||
client.clone(),
|
|
||||||
args,
|
|
||||||
&Some("linux/amd64".to_string()),
|
|
||||||
&bin_name.clone(),
|
|
||||||
&"release".into(),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
let build_image = execute(
|
|
||||||
client.clone(),
|
|
||||||
args,
|
|
||||||
&container,
|
|
||||||
&base_image,
|
|
||||||
bin_name,
|
|
||||||
&Some("linux/amd64".to_string()),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
build_image.exit_code().await?;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
pub async fn execute(
|
|
||||||
_client: Arc<dagger_sdk::Query>,
|
|
||||||
_args: &GlobalArgs,
|
|
||||||
container: &dagger_sdk::Container,
|
|
||||||
base_image: &dagger_sdk::Container,
|
|
||||||
bin_name: &String,
|
|
||||||
platform: &Option<String>,
|
|
||||||
) -> eyre::Result<Container> {
|
|
||||||
let rust_target = match platform
|
|
||||||
.clone()
|
|
||||||
.unwrap_or("linux/amd64".to_string())
|
|
||||||
.as_str()
|
|
||||||
{
|
|
||||||
"linux/amd64" => "x86_64-unknown-linux-gnu",
|
|
||||||
"linux/arm64" => "aarch64-unknown-linux-gnu",
|
|
||||||
_ => eyre::bail!("architecture not supported"),
|
|
||||||
};
|
|
||||||
let build_image = container.with_exec(vec![
|
|
||||||
"cargo",
|
|
||||||
"build",
|
|
||||||
"--target",
|
|
||||||
rust_target,
|
|
||||||
"--release",
|
|
||||||
"-p",
|
|
||||||
bin_name,
|
|
||||||
]);
|
|
||||||
|
|
||||||
let final_image = base_image
|
|
||||||
.with_file(
|
|
||||||
format!("/usr/local/bin/{}", &bin_name),
|
|
||||||
build_image
|
|
||||||
.file(format!("target/{}/release/{}", rust_target, &bin_name))
|
|
||||||
.id()
|
|
||||||
.await?,
|
|
||||||
)
|
|
||||||
.with_exec(vec![bin_name, "--help"]);
|
|
||||||
|
|
||||||
let output = final_image.stdout().await?;
|
|
||||||
println!("{output}");
|
|
||||||
|
|
||||||
//.with_entrypoint(vec![&bin_name, "--log-level=debug"]);
|
|
||||||
|
|
||||||
Ok(final_image)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mod test {
|
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use crate::GlobalArgs;
|
|
||||||
|
|
||||||
pub async fn execute(
|
|
||||||
_client: Arc<dagger_sdk::Query>,
|
|
||||||
_args: &GlobalArgs,
|
|
||||||
container: dagger_sdk::Container,
|
|
||||||
) -> eyre::Result<()> {
|
|
||||||
let test_image = container
|
|
||||||
.pipeline("rust:test")
|
|
||||||
.with_exec(vec!["apt", "update"])
|
|
||||||
.with_exec(vec!["apt", "install", "-y", "git"])
|
|
||||||
.with_exec(vec!["cargo", "test"]);
|
|
||||||
|
|
||||||
test_image.exit_code().await?;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn get_base_debian_image(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
args: &GlobalArgs,
|
|
||||||
platform: Option<String>,
|
|
||||||
) -> eyre::Result<dagger_sdk::Container> {
|
|
||||||
let default_platform = client.default_platform().await?;
|
|
||||||
let platform = platform.map(Platform).unwrap_or(default_platform);
|
|
||||||
|
|
||||||
let image = client
|
|
||||||
.container_opts(QueryContainerOpts {
|
|
||||||
id: None,
|
|
||||||
platform: Some(platform),
|
|
||||||
})
|
|
||||||
.from(
|
|
||||||
args.production_image
|
|
||||||
.clone()
|
|
||||||
.unwrap_or("debian:bullseye".to_string()),
|
|
||||||
);
|
|
||||||
|
|
||||||
let base_image = image.with_exec(vec!["apt", "update"]).with_exec(vec![
|
|
||||||
"apt",
|
|
||||||
"install",
|
|
||||||
"-y",
|
|
||||||
"libssl-dev",
|
|
||||||
"pkg-config",
|
|
||||||
"openssl",
|
|
||||||
"git",
|
|
||||||
]);
|
|
||||||
|
|
||||||
Ok(base_image)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn get_src(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
args: &GlobalArgs,
|
|
||||||
) -> eyre::Result<dagger_sdk::Directory> {
|
|
||||||
let directory = client.host().directory_opts(
|
|
||||||
args.source
|
|
||||||
.clone()
|
|
||||||
.unwrap_or(PathBuf::from("."))
|
|
||||||
.display()
|
|
||||||
.to_string(),
|
|
||||||
dagger_sdk::HostDirectoryOptsBuilder::default()
|
|
||||||
.exclude(vec!["node_modules/", ".git/", "target/"])
|
|
||||||
.build()?,
|
|
||||||
);
|
|
||||||
|
|
||||||
Ok(directory)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn get_rust_dep_src(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
args: &GlobalArgs,
|
|
||||||
) -> eyre::Result<dagger_sdk::Directory> {
|
|
||||||
let directory = client.host().directory_opts(
|
|
||||||
args.source
|
|
||||||
.clone()
|
|
||||||
.unwrap_or(PathBuf::from("."))
|
|
||||||
.display()
|
|
||||||
.to_string(),
|
|
||||||
dagger_sdk::HostDirectoryOptsBuilder::default()
|
|
||||||
.include(vec!["**/Cargo.toml", "**/Cargo.lock"])
|
|
||||||
.build()?,
|
|
||||||
);
|
|
||||||
|
|
||||||
Ok(directory)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn get_rust_skeleton_files(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
_args: &GlobalArgs,
|
|
||||||
) -> eyre::Result<(dagger_sdk::Directory, Vec<String>)> {
|
|
||||||
let mut rust_crates = vec![PathBuf::from("ci")];
|
|
||||||
let mut dirs = tokio::fs::read_dir("crates").await?;
|
|
||||||
|
|
||||||
while let Some(entry) = dirs.next_entry().await? {
|
|
||||||
if entry.metadata().await?.is_dir() {
|
|
||||||
rust_crates.push(entry.path())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn create_skeleton_files(
|
|
||||||
directory: dagger_sdk::Directory,
|
|
||||||
path: &Path,
|
|
||||||
) -> eyre::Result<dagger_sdk::Directory> {
|
|
||||||
println!("found crates: {}", path.display());
|
|
||||||
let main_content = r#"
|
|
||||||
#[allow(dead_code)]
|
|
||||||
fn main() { panic!("should never be executed"); }"#;
|
|
||||||
let lib_content = r#"
|
|
||||||
#[allow(dead_code)]
|
|
||||||
fn some() { panic!("should never be executed"); }"#;
|
|
||||||
|
|
||||||
let directory = directory.with_new_file(
|
|
||||||
path.join("src").join("main.rs").display().to_string(),
|
|
||||||
main_content,
|
|
||||||
);
|
|
||||||
let directory = directory.with_new_file(
|
|
||||||
path.join("src").join("lib.rs").display().to_string(),
|
|
||||||
lib_content,
|
|
||||||
);
|
|
||||||
|
|
||||||
Ok(directory)
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut directory = client.directory();
|
|
||||||
let mut crate_names = Vec::new();
|
|
||||||
|
|
||||||
for rust_crate in rust_crates.iter() {
|
|
||||||
if let Some(file_name) = rust_crate.file_name() {
|
|
||||||
crate_names.push(file_name.to_str().unwrap().to_string());
|
|
||||||
}
|
|
||||||
directory = create_skeleton_files(directory, rust_crate)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok((directory, crate_names))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn base_rust_image(
|
|
||||||
client: Arc<dagger_sdk::Query>,
|
|
||||||
args: &GlobalArgs,
|
|
||||||
platform: &Option<String>,
|
|
||||||
bin_name: &String,
|
|
||||||
profile: &String,
|
|
||||||
) -> eyre::Result<dagger_sdk::Container> {
|
|
||||||
let dep_src = get_rust_dep_src(client.clone(), args).await?;
|
|
||||||
let (skeleton_files, crates) = get_rust_skeleton_files(client.clone(), args).await?;
|
|
||||||
let src = get_src(client.clone(), args)?;
|
|
||||||
|
|
||||||
let client = client.pipeline("rust_base_image");
|
|
||||||
|
|
||||||
let rust_target = match platform
|
|
||||||
.clone()
|
|
||||||
.unwrap_or("linux/amd64".to_string())
|
|
||||||
.as_str()
|
|
||||||
{
|
|
||||||
"linux/amd64" => "x86_64-unknown-linux-gnu",
|
|
||||||
"linux/arm64" => "aarch64-unknown-linux-gnu",
|
|
||||||
_ => eyre::bail!("architecture not supported"),
|
|
||||||
};
|
|
||||||
let rust_build_image = client
|
|
||||||
.container()
|
|
||||||
.from(
|
|
||||||
args.rust_builder_image
|
|
||||||
.as_ref()
|
|
||||||
.unwrap_or(&"rustlang/rust:nightly".into()),
|
|
||||||
)
|
|
||||||
.with_exec(vec!["rustup", "target", "add", rust_target]);
|
|
||||||
|
|
||||||
let target_cache = client.cache_volume(format!("rust_target_{}", profile));
|
|
||||||
let mut build_options = vec!["cargo", "build", "--target", rust_target, "-p", bin_name];
|
|
||||||
|
|
||||||
if profile == "release" {
|
|
||||||
build_options.push("--release");
|
|
||||||
}
|
|
||||||
let rust_prebuild = rust_build_image
|
|
||||||
.with_workdir("/mnt/src")
|
|
||||||
.with_directory("/mnt/src", dep_src.id().await?)
|
|
||||||
.with_directory("/mnt/src/", skeleton_files.id().await?)
|
|
||||||
.with_exec(build_options)
|
|
||||||
.with_mounted_cache("/mnt/src/target/", target_cache.id().await?);
|
|
||||||
|
|
||||||
let exclude = crates
|
|
||||||
.iter()
|
|
||||||
.filter(|c| **c != "ci")
|
|
||||||
.map(|c| format!("**/*{}*", c.replace('-', "_")))
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
let exclude = exclude.iter().map(|c| c.as_str()).collect();
|
|
||||||
|
|
||||||
let incremental_dir = client.directory().with_directory_opts(
|
|
||||||
".",
|
|
||||||
rust_prebuild.directory("target").id().await?,
|
|
||||||
dagger_sdk::DirectoryWithDirectoryOpts {
|
|
||||||
exclude: Some(exclude),
|
|
||||||
include: None,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
let rust_with_src = rust_build_image
|
|
||||||
.with_workdir("/mnt/src")
|
|
||||||
.with_directory(
|
|
||||||
"/usr/local/cargo",
|
|
||||||
rust_prebuild.directory("/usr/local/cargo").id().await?,
|
|
||||||
)
|
|
||||||
.with_directory("target", incremental_dir.id().await?)
|
|
||||||
.with_directory("/mnt/src/", src.id().await?);
|
|
||||||
|
|
||||||
Ok(rust_with_src)
|
|
||||||
}
|
|
19
crates/cuddle-please-actions/Cargo.toml
Normal file
19
crates/cuddle-please-actions/Cargo.toml
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
[package]
|
||||||
|
name = "cuddle-please-actions"
|
||||||
|
description = "A release-please inspired release manager tool, built on top of cuddle, but also useful standalone, cuddle-please supports, your ci of choice, as well as gitea, github"
|
||||||
|
repository = "https://git.front.kjuulh.io/kjuulh/cuddle-please"
|
||||||
|
readme = "../../README.md"
|
||||||
|
license-file = "../../LICENSE"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
publishable = true
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
anyhow.workspace = true
|
||||||
|
tracing.workspace = true
|
||||||
|
semver.workspace = true
|
||||||
|
toml_edit.workspace = true
|
||||||
|
yaml-rust2.workspace = true
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
pretty_assertions.workspace = true
|
9
crates/cuddle-please-actions/src/actions.rs
Normal file
9
crates/cuddle-please-actions/src/actions.rs
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
use semver::Version;
|
||||||
|
|
||||||
|
use crate::ActionConfig;
|
||||||
|
|
||||||
|
pub trait Action {
|
||||||
|
fn enabled(&self, config: &ActionConfig) -> anyhow::Result<bool>;
|
||||||
|
fn name(&self) -> String;
|
||||||
|
fn execute(&self, version: &Version) -> anyhow::Result<()>;
|
||||||
|
}
|
49
crates/cuddle-please-actions/src/config.rs
Normal file
49
crates/cuddle-please-actions/src/config.rs
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
use std::ops::Deref;
|
||||||
|
|
||||||
|
use anyhow::Context;
|
||||||
|
|
||||||
|
pub enum ActionConfig {
|
||||||
|
Actual { doc: yaml_rust2::Yaml },
|
||||||
|
None,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deref for ActionConfig {
|
||||||
|
type Target = yaml_rust2::Yaml;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
match &self {
|
||||||
|
ActionConfig::Actual { doc } => doc,
|
||||||
|
ActionConfig::None => &yaml_rust2::Yaml::BadValue,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryFrom<&str> for ActionConfig {
|
||||||
|
type Error = anyhow::Error;
|
||||||
|
|
||||||
|
fn try_from(value: &str) -> Result<Self, Self::Error> {
|
||||||
|
let mut cuddle = yaml_rust2::YamlLoader::load_from_str(value)?;
|
||||||
|
|
||||||
|
if cuddle.len() != 1 {
|
||||||
|
anyhow::bail!("cuddle.yaml can only be 1 document wide");
|
||||||
|
}
|
||||||
|
|
||||||
|
let doc = cuddle.pop().unwrap();
|
||||||
|
let doc = doc["please"]["actions"].clone();
|
||||||
|
|
||||||
|
if doc.is_badvalue() {
|
||||||
|
return Ok(Self::None);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Self::Actual { doc })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ActionConfig {
|
||||||
|
pub fn parse() -> anyhow::Result<Self> {
|
||||||
|
let cuddle_yaml =
|
||||||
|
std::fs::read_to_string("cuddle.yaml").context("failed to read cuddle.yaml")?;
|
||||||
|
|
||||||
|
Self::try_from(cuddle_yaml.as_str())
|
||||||
|
}
|
||||||
|
}
|
45
crates/cuddle-please-actions/src/lib.rs
Normal file
45
crates/cuddle-please-actions/src/lib.rs
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
pub(crate) mod actions;
|
||||||
|
mod config;
|
||||||
|
mod rust_action;
|
||||||
|
|
||||||
|
use std::{ops::Deref, sync::Arc};
|
||||||
|
|
||||||
|
use catalog::RustAction;
|
||||||
|
pub use config::ActionConfig;
|
||||||
|
use semver::Version;
|
||||||
|
pub mod catalog {
|
||||||
|
pub use crate::rust_action::*;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Action(Arc<dyn actions::Action>);
|
||||||
|
|
||||||
|
impl Deref for Action {
|
||||||
|
type Target = Arc<dyn actions::Action>;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Actions(Vec<Action>);
|
||||||
|
|
||||||
|
impl Actions {
|
||||||
|
pub fn from_cuddle() -> anyhow::Result<Self> {
|
||||||
|
let config = ActionConfig::parse()?;
|
||||||
|
|
||||||
|
Ok(Self(
|
||||||
|
vec![Action(Arc::new(RustAction::new()))]
|
||||||
|
.into_iter()
|
||||||
|
.filter(|a| a.enabled(&config).unwrap_or_default())
|
||||||
|
.collect(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn execute(&self, version: &Version) -> anyhow::Result<()> {
|
||||||
|
for action in &self.0 {
|
||||||
|
action.execute(version)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
237
crates/cuddle-please-actions/src/rust_action.rs
Normal file
237
crates/cuddle-please-actions/src/rust_action.rs
Normal file
@ -0,0 +1,237 @@
|
|||||||
|
use std::io::Write;
|
||||||
|
|
||||||
|
use anyhow::Context;
|
||||||
|
|
||||||
|
use crate::{actions::Action, ActionConfig};
|
||||||
|
|
||||||
|
#[derive(Default, Clone)]
|
||||||
|
pub struct RustAction {}
|
||||||
|
|
||||||
|
impl RustAction {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn execute_content(
|
||||||
|
&self,
|
||||||
|
version: &semver::Version,
|
||||||
|
cargo_content: &str,
|
||||||
|
) -> anyhow::Result<String> {
|
||||||
|
tracing::trace!("parsing Cargo.toml file as tolm");
|
||||||
|
let mut cargo_doc = cargo_content.parse::<toml_edit::DocumentMut>()?;
|
||||||
|
|
||||||
|
tracing::debug!(
|
||||||
|
"updating cargo workspace package version to {}",
|
||||||
|
version.to_string()
|
||||||
|
);
|
||||||
|
|
||||||
|
let workspace = if cargo_doc.contains_table("workspace") {
|
||||||
|
cargo_doc["workspace"].as_table_mut().unwrap()
|
||||||
|
} else {
|
||||||
|
let mut t = toml_edit::Table::new();
|
||||||
|
t.set_implicit(true);
|
||||||
|
cargo_doc["workspace"] = toml_edit::Item::Table(t);
|
||||||
|
cargo_doc["workspace"].as_table_mut().unwrap()
|
||||||
|
};
|
||||||
|
let package = workspace["package"].or_insert(toml_edit::table());
|
||||||
|
package["version"] = toml_edit::value(version.to_string());
|
||||||
|
|
||||||
|
Ok(cargo_doc.to_string())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Action for RustAction {
|
||||||
|
fn enabled(&self, config: &ActionConfig) -> anyhow::Result<bool> {
|
||||||
|
if let Ok(v) = std::env::var("CUDDLE_PLEASE_RUST_ACTION") {
|
||||||
|
if let Ok(true) = v.parse::<bool>() {
|
||||||
|
return Ok(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let val = &config[self.name().as_str()];
|
||||||
|
|
||||||
|
if val.is_badvalue() {
|
||||||
|
return Ok(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(val.as_bool().unwrap_or(true))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn name(&self) -> String {
|
||||||
|
"rust".into()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn execute(&self, version: &semver::Version) -> anyhow::Result<()> {
|
||||||
|
tracing::info!(
|
||||||
|
"running rust action for version: {} and file: Cargo.toml",
|
||||||
|
version.to_string()
|
||||||
|
);
|
||||||
|
|
||||||
|
let path = std::path::PathBuf::from("Cargo.toml");
|
||||||
|
|
||||||
|
tracing::trace!("reading Cargo.toml");
|
||||||
|
|
||||||
|
let file = match std::fs::read_to_string(&path) {
|
||||||
|
Ok(file) => file,
|
||||||
|
Err(e) => match e.kind() {
|
||||||
|
std::io::ErrorKind::NotFound => {
|
||||||
|
anyhow::bail!("err: Cargo.toml was not found in dir")
|
||||||
|
}
|
||||||
|
_ => Err(e)?,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
let cargo_doc = self.execute_content(version, &file)?;
|
||||||
|
|
||||||
|
let mut cargo_file = std::fs::File::create(&path)?;
|
||||||
|
cargo_file.write_all(cargo_doc.as_bytes())?;
|
||||||
|
cargo_file.sync_all()?;
|
||||||
|
|
||||||
|
tracing::debug!("finished writing cargo file");
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use semver::{BuildMetadata, Prerelease};
|
||||||
|
|
||||||
|
use crate::{actions::Action, ActionConfig};
|
||||||
|
|
||||||
|
use super::RustAction;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_is_enabled() {
|
||||||
|
let config = ActionConfig::try_from(
|
||||||
|
r#"
|
||||||
|
please:
|
||||||
|
actions:
|
||||||
|
rust: true
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let enabled = RustAction::new().enabled(&config).unwrap();
|
||||||
|
|
||||||
|
assert!(enabled)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_is_disabled_by_default() {
|
||||||
|
let config = ActionConfig::try_from(
|
||||||
|
r#"
|
||||||
|
please:
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let enabled = RustAction::new().enabled(&config).unwrap();
|
||||||
|
|
||||||
|
assert!(!enabled)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_is_disabled() {
|
||||||
|
let config = ActionConfig::try_from(
|
||||||
|
r#"
|
||||||
|
please:
|
||||||
|
actions:
|
||||||
|
rust: false
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let enabled = RustAction::new().enabled(&config).unwrap();
|
||||||
|
|
||||||
|
assert!(!enabled)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_missing_value_is_enabled() {
|
||||||
|
let config = ActionConfig::try_from(
|
||||||
|
r#"
|
||||||
|
please:
|
||||||
|
actions:
|
||||||
|
rust:
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let enabled = RustAction::new().enabled(&config).unwrap();
|
||||||
|
|
||||||
|
assert!(enabled)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_can_edit_empty_file() {
|
||||||
|
let output = RustAction::default()
|
||||||
|
.execute_content(
|
||||||
|
&semver::Version {
|
||||||
|
major: 0,
|
||||||
|
minor: 1,
|
||||||
|
patch: 0,
|
||||||
|
pre: Prerelease::default(),
|
||||||
|
build: BuildMetadata::default(),
|
||||||
|
},
|
||||||
|
"",
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
pretty_assertions::assert_eq!(
|
||||||
|
r#"[workspace.package]
|
||||||
|
version = "0.1.0"
|
||||||
|
"#,
|
||||||
|
&output,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_only_edits_stuff() {
|
||||||
|
let input = r#"
|
||||||
|
[workspace]
|
||||||
|
members = ["."]
|
||||||
|
|
||||||
|
|
||||||
|
[package]
|
||||||
|
something = {some = "something"}
|
||||||
|
|
||||||
|
# Some comment
|
||||||
|
|
||||||
|
[workspace.package]
|
||||||
|
version = "0.0.0" # some comment
|
||||||
|
readme = "../../"
|
||||||
|
"#;
|
||||||
|
|
||||||
|
let output = RustAction::default()
|
||||||
|
.execute_content(
|
||||||
|
&semver::Version {
|
||||||
|
major: 0,
|
||||||
|
minor: 1,
|
||||||
|
patch: 0,
|
||||||
|
pre: Prerelease::default(),
|
||||||
|
build: BuildMetadata::default(),
|
||||||
|
},
|
||||||
|
input,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
pretty_assertions::assert_eq!(
|
||||||
|
r#"
|
||||||
|
[workspace]
|
||||||
|
members = ["."]
|
||||||
|
|
||||||
|
|
||||||
|
[package]
|
||||||
|
something = {some = "something"}
|
||||||
|
|
||||||
|
# Some comment
|
||||||
|
|
||||||
|
[workspace.package]
|
||||||
|
version = "0.1.0"
|
||||||
|
readme = "../../"
|
||||||
|
"#,
|
||||||
|
&output,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
@ -12,6 +12,7 @@ publishable = true
|
|||||||
[dependencies]
|
[dependencies]
|
||||||
cuddle-please-frontend.workspace = true
|
cuddle-please-frontend.workspace = true
|
||||||
cuddle-please-misc.workspace = true
|
cuddle-please-misc.workspace = true
|
||||||
|
cuddle-please-actions.workspace = true
|
||||||
|
|
||||||
anyhow.workspace = true
|
anyhow.workspace = true
|
||||||
tracing.workspace = true
|
tracing.workspace = true
|
||||||
|
@ -6,6 +6,7 @@ use std::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use clap::{Parser, Subcommand};
|
use clap::{Parser, Subcommand};
|
||||||
|
use cuddle_please_actions::Actions;
|
||||||
use cuddle_please_frontend::{gatheres::ConfigArgs, PleaseConfig, PleaseConfigBuilder};
|
use cuddle_please_frontend::{gatheres::ConfigArgs, PleaseConfig, PleaseConfigBuilder};
|
||||||
use cuddle_please_misc::{
|
use cuddle_please_misc::{
|
||||||
ConsoleUi, DynRemoteGitClient, DynUi, GiteaClient, GlobalArgs, LocalGitClient, StdinFn,
|
ConsoleUi, DynRemoteGitClient, DynUi, GiteaClient, GlobalArgs, LocalGitClient, StdinFn,
|
||||||
@ -90,26 +91,27 @@ impl Command {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn execute(self, current_dir: Option<&Path>) -> anyhow::Result<()> {
|
pub fn execute(self, current_dir: Option<&Path>) -> anyhow::Result<()> {
|
||||||
match &self.commands {
|
if let Some(c) = &self.commands {
|
||||||
Some(Commands::Release {}) => {
|
match c {
|
||||||
let (config, git_client, gitea_client) = self.get_deps(current_dir)?;
|
Commands::Release {} => {
|
||||||
ReleaseCommandHandler::new(self.ui, config, git_client, gitea_client)
|
let (config, git_client, gitea_client, actions) = self.get_deps(current_dir)?;
|
||||||
.execute(self.global.dry_run)?;
|
ReleaseCommandHandler::new(self.ui, config, git_client, gitea_client, actions)
|
||||||
}
|
.execute(self.global.dry_run)?;
|
||||||
Some(Commands::Config { command }) => {
|
}
|
||||||
let (config, _, _) = self.get_deps(current_dir)?;
|
Commands::Config { command } => {
|
||||||
ConfigCommandHandler::new(self.ui, config).execute(command)?;
|
let (config, _, _, _) = self.get_deps(current_dir)?;
|
||||||
}
|
ConfigCommandHandler::new(self.ui, config).execute(command)?;
|
||||||
Some(Commands::Gitea { command }) => {
|
}
|
||||||
let (config, _, gitea_client) = self.get_deps(current_dir)?;
|
Commands::Gitea { command } => {
|
||||||
|
let (config, _, gitea_client, _) = self.get_deps(current_dir)?;
|
||||||
|
|
||||||
GiteaCommandHandler::new(self.ui, config, gitea_client)
|
GiteaCommandHandler::new(self.ui, config, gitea_client)
|
||||||
.execute(command, self.global.token.expect("token to be set").deref())?;
|
.execute(command, self.global.token.expect("token to be set").deref())?;
|
||||||
|
}
|
||||||
|
Commands::Doctor {} => {
|
||||||
|
DoctorCommandHandler::new(self.ui).execute()?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Some(Commands::Doctor {}) => {
|
|
||||||
DoctorCommandHandler::new(self.ui).execute()?;
|
|
||||||
}
|
|
||||||
None => {}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -118,7 +120,7 @@ impl Command {
|
|||||||
fn get_deps(
|
fn get_deps(
|
||||||
&self,
|
&self,
|
||||||
current_dir: Option<&Path>,
|
current_dir: Option<&Path>,
|
||||||
) -> anyhow::Result<(PleaseConfig, VcsClient, DynRemoteGitClient)> {
|
) -> anyhow::Result<(PleaseConfig, VcsClient, DynRemoteGitClient, Actions)> {
|
||||||
let config = self.build_config(current_dir)?;
|
let config = self.build_config(current_dir)?;
|
||||||
let git_client =
|
let git_client =
|
||||||
self.get_git(&config, self.global.token.clone().expect("token to be set"))?;
|
self.get_git(&config, self.global.token.clone().expect("token to be set"))?;
|
||||||
@ -140,7 +142,9 @@ impl Command {
|
|||||||
tracing_subscriber::fmt().with_env_filter(env_filter).init();
|
tracing_subscriber::fmt().with_env_filter(env_filter).init();
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok((config, git_client, gitea_client))
|
let actions = self.get_actions()?;
|
||||||
|
|
||||||
|
Ok((config, git_client, gitea_client, actions))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn build_config(&self, current_dir: Option<&Path>) -> Result<PleaseConfig, anyhow::Error> {
|
fn build_config(&self, current_dir: Option<&Path>) -> Result<PleaseConfig, anyhow::Error> {
|
||||||
@ -183,6 +187,10 @@ impl Command {
|
|||||||
)),
|
)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn get_actions(&self) -> anyhow::Result<Actions> {
|
||||||
|
Actions::from_cuddle()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Subcommand)]
|
#[derive(Debug, Clone, Subcommand)]
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
use cuddle_please_actions::Actions;
|
||||||
use cuddle_please_frontend::PleaseConfig;
|
use cuddle_please_frontend::PleaseConfig;
|
||||||
|
|
||||||
use ::semver::Version;
|
use ::semver::Version;
|
||||||
@ -13,6 +14,7 @@ pub struct ReleaseCommandHandler {
|
|||||||
config: PleaseConfig,
|
config: PleaseConfig,
|
||||||
git_client: VcsClient,
|
git_client: VcsClient,
|
||||||
gitea_client: DynRemoteGitClient,
|
gitea_client: DynRemoteGitClient,
|
||||||
|
actions: Actions,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ReleaseCommandHandler {
|
impl ReleaseCommandHandler {
|
||||||
@ -21,12 +23,14 @@ impl ReleaseCommandHandler {
|
|||||||
config: PleaseConfig,
|
config: PleaseConfig,
|
||||||
git_client: VcsClient,
|
git_client: VcsClient,
|
||||||
gitea_client: DynRemoteGitClient,
|
gitea_client: DynRemoteGitClient,
|
||||||
|
actions: Actions,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
ui,
|
ui,
|
||||||
config,
|
config,
|
||||||
git_client,
|
git_client,
|
||||||
gitea_client,
|
gitea_client,
|
||||||
|
actions,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -69,6 +73,8 @@ impl ReleaseCommandHandler {
|
|||||||
let (changelog_placement, changelog, changelog_last_changes) =
|
let (changelog_placement, changelog, changelog_last_changes) =
|
||||||
compose_changelog(&commit_strs, &next_version, source)?;
|
compose_changelog(&commit_strs, &next_version, source)?;
|
||||||
|
|
||||||
|
self.actions.execute(&next_version)?;
|
||||||
|
|
||||||
if let Some(first_commit) = commit_strs.first() {
|
if let Some(first_commit) = commit_strs.first() {
|
||||||
if first_commit.contains("chore(release): ") {
|
if first_commit.contains("chore(release): ") {
|
||||||
tracing::trace!("creating release");
|
tracing::trace!("creating release");
|
||||||
@ -149,7 +155,8 @@ impl ReleaseCommandHandler {
|
|||||||
owner,
|
owner,
|
||||||
repository,
|
repository,
|
||||||
&next_version.to_string(),
|
&next_version.to_string(),
|
||||||
&changelog_last_changes.unwrap(),
|
&changelog_last_changes
|
||||||
|
.ok_or(anyhow::anyhow!("could not get the latest changes"))?,
|
||||||
existing_pr,
|
existing_pr,
|
||||||
)?
|
)?
|
||||||
} else {
|
} else {
|
||||||
@ -164,7 +171,8 @@ impl ReleaseCommandHandler {
|
|||||||
owner,
|
owner,
|
||||||
repository,
|
repository,
|
||||||
&next_version.to_string(),
|
&next_version.to_string(),
|
||||||
&changelog,
|
&changelog_last_changes
|
||||||
|
.ok_or(anyhow::anyhow!("could not get the latest changes"))?,
|
||||||
branch,
|
branch,
|
||||||
)?
|
)?
|
||||||
} else {
|
} else {
|
||||||
|
@ -3,7 +3,7 @@ use chrono::{DateTime, NaiveDate, Utc};
|
|||||||
use git_cliff_core::{
|
use git_cliff_core::{
|
||||||
changelog::Changelog,
|
changelog::Changelog,
|
||||||
commit::Commit,
|
commit::Commit,
|
||||||
config::{ChangelogConfig, CommitParser, Config, GitConfig},
|
config::{Bump, ChangelogConfig, CommitParser, Config, GitConfig, Remote, RemoteConfig},
|
||||||
release::Release,
|
release::Release,
|
||||||
};
|
};
|
||||||
use regex::Regex;
|
use regex::Regex;
|
||||||
@ -77,6 +77,8 @@ impl ChangeLogBuilder {
|
|||||||
commit_id: None,
|
commit_id: None,
|
||||||
timestamp,
|
timestamp,
|
||||||
previous: None,
|
previous: None,
|
||||||
|
message: None,
|
||||||
|
repository: None,
|
||||||
},
|
},
|
||||||
config: self.config,
|
config: self.config,
|
||||||
release_link: self.release_link,
|
release_link: self.release_link,
|
||||||
@ -86,7 +88,7 @@ impl ChangeLogBuilder {
|
|||||||
fn release_timestamp(&self) -> i64 {
|
fn release_timestamp(&self) -> i64 {
|
||||||
self.release_date
|
self.release_date
|
||||||
.and_then(|date| date.and_hms_opt(0, 0, 0))
|
.and_then(|date| date.and_hms_opt(0, 0, 0))
|
||||||
.map(|d| DateTime::<Utc>::from_utc(d, Utc))
|
.map(|d| DateTime::<Utc>::from_naive_utc_and_offset(d, Utc))
|
||||||
.unwrap_or_else(Utc::now)
|
.unwrap_or_else(Utc::now)
|
||||||
.timestamp()
|
.timestamp()
|
||||||
}
|
}
|
||||||
@ -139,6 +141,8 @@ impl ChangeLog<'_> {
|
|||||||
let config = Config {
|
let config = Config {
|
||||||
changelog: default_changelog_config(None, self.release_link.as_deref()),
|
changelog: default_changelog_config(None, self.release_link.as_deref()),
|
||||||
git: default_git_config(),
|
git: default_git_config(),
|
||||||
|
remote: RemoteConfig::default(),
|
||||||
|
bump: Bump::default(),
|
||||||
};
|
};
|
||||||
|
|
||||||
config
|
config
|
||||||
@ -148,6 +152,8 @@ impl ChangeLog<'_> {
|
|||||||
let config = Config {
|
let config = Config {
|
||||||
changelog: default_changelog_config(header, self.release_link.as_deref()),
|
changelog: default_changelog_config(header, self.release_link.as_deref()),
|
||||||
git: default_git_config(),
|
git: default_git_config(),
|
||||||
|
remote: RemoteConfig::default(),
|
||||||
|
bump: Bump::default(),
|
||||||
};
|
};
|
||||||
|
|
||||||
config
|
config
|
||||||
@ -173,6 +179,10 @@ fn default_commit_parsers() -> Vec<CommitParser> {
|
|||||||
default_scope: None,
|
default_scope: None,
|
||||||
scope: None,
|
scope: None,
|
||||||
skip: None,
|
skip: None,
|
||||||
|
field: None,
|
||||||
|
pattern: None,
|
||||||
|
sha: None,
|
||||||
|
footer: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -191,6 +201,10 @@ fn default_commit_parsers() -> Vec<CommitParser> {
|
|||||||
default_scope: None,
|
default_scope: None,
|
||||||
skip: None,
|
skip: None,
|
||||||
scope: None,
|
scope: None,
|
||||||
|
field: None,
|
||||||
|
pattern: None,
|
||||||
|
sha: None,
|
||||||
|
footer: None,
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@ -210,6 +224,7 @@ fn default_changelog_config(header: Option<String>, release_link: Option<&str>)
|
|||||||
body: Some(default_changelog_body_config(release_link)),
|
body: Some(default_changelog_body_config(release_link)),
|
||||||
footer: None,
|
footer: None,
|
||||||
trim: Some(true),
|
trim: Some(true),
|
||||||
|
postprocessors: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -79,6 +79,13 @@ impl VcsClient {
|
|||||||
let stdout = std::str::from_utf8(&checkout_branch.stdout)?;
|
let stdout = std::str::from_utf8(&checkout_branch.stdout)?;
|
||||||
let stderr = std::str::from_utf8(&checkout_branch.stderr)?;
|
let stderr = std::str::from_utf8(&checkout_branch.stderr)?;
|
||||||
tracing::debug!(stdout = stdout, stderr = stderr, "git {}", args.join(" "));
|
tracing::debug!(stdout = stdout, stderr = stderr, "git {}", args.join(" "));
|
||||||
|
let exit_code = checkout_branch.status;
|
||||||
|
if !exit_code.success() {
|
||||||
|
anyhow::bail!(
|
||||||
|
"failed to run git command: {}",
|
||||||
|
exit_code.code().unwrap_or(-1)
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -330,7 +330,7 @@ impl RemoteGitEngine for GiteaClient {
|
|||||||
base: base.into(),
|
base: base.into(),
|
||||||
body: body.into(),
|
body: body.into(),
|
||||||
head: "cuddle-please/release".into(),
|
head: "cuddle-please/release".into(),
|
||||||
title: format!("chore(release): {}", version),
|
title: format!("chore(release): v{}", version),
|
||||||
};
|
};
|
||||||
|
|
||||||
tracing::trace!(
|
tracing::trace!(
|
||||||
@ -377,7 +377,7 @@ impl RemoteGitEngine for GiteaClient {
|
|||||||
|
|
||||||
let request = CreatePullRequestOption {
|
let request = CreatePullRequestOption {
|
||||||
body: body.into(),
|
body: body.into(),
|
||||||
title: format!("chore(release): {}", version),
|
title: format!("chore(release): v{}", version),
|
||||||
};
|
};
|
||||||
|
|
||||||
tracing::trace!(
|
tracing::trace!(
|
||||||
@ -429,9 +429,9 @@ impl RemoteGitEngine for GiteaClient {
|
|||||||
let request = CreateReleaseOption {
|
let request = CreateReleaseOption {
|
||||||
body: body.into(),
|
body: body.into(),
|
||||||
draft: false,
|
draft: false,
|
||||||
name: version.into(),
|
name: format!("v{version}"),
|
||||||
prerelease,
|
prerelease,
|
||||||
tag_name: version.into(),
|
tag_name: format!("v{version}"),
|
||||||
};
|
};
|
||||||
|
|
||||||
tracing::trace!(
|
tracing::trace!(
|
||||||
@ -517,6 +517,7 @@ impl TryFrom<Tag> for Version {
|
|||||||
tracing::trace!(name = &value.name, "parsing tag into version");
|
tracing::trace!(name = &value.name, "parsing tag into version");
|
||||||
value
|
value
|
||||||
.name
|
.name
|
||||||
|
.trim_start_matches("v")
|
||||||
.parse::<Version>()
|
.parse::<Version>()
|
||||||
.context("could not get version from tag")
|
.context("could not get version from tag")
|
||||||
}
|
}
|
||||||
|
31
crates/cuddle-please-release-strategy/Cargo.toml
Normal file
31
crates/cuddle-please-release-strategy/Cargo.toml
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
[package]
|
||||||
|
name = "cuddle-please-release-strategy"
|
||||||
|
description = "A release-please inspired release manager tool, built on top of cuddle, but also useful standalone, cuddle-please supports, your ci of choice, as well as gitea, github"
|
||||||
|
repository = "https://git.front.kjuulh.io/kjuulh/cuddle-please"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
readme = "../../README.md"
|
||||||
|
license-file = "../../LICENSE"
|
||||||
|
publishable = true
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
anyhow.workspace = true
|
||||||
|
tracing.workspace = true
|
||||||
|
serde.workspace = true
|
||||||
|
semver.workspace = true
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
tracing-test = { workspace = true, features = ["no-env-filter"] }
|
||||||
|
pretty_assertions.workspace = true
|
||||||
|
tempdir.workspace = true
|
||||||
|
|
||||||
|
[features]
|
||||||
|
rust-workspace = []
|
||||||
|
rust-crate = []
|
||||||
|
toml-edit = []
|
||||||
|
json-edit = []
|
||||||
|
yaml-edit = []
|
||||||
|
|
||||||
|
default = [
|
||||||
|
"json-edit"
|
||||||
|
]
|
60
crates/cuddle-please-release-strategy/src/json_edit.rs
Normal file
60
crates/cuddle-please-release-strategy/src/json_edit.rs
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
use anyhow::Context;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Serialize, Deserialize)]
|
||||||
|
pub struct JsonEditOptions {
|
||||||
|
pub jq: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl JsonEditOptions {
|
||||||
|
pub fn execute(&self, path: &Path, next_version: impl AsRef<str>) -> anyhow::Result<()> {
|
||||||
|
let next_version = next_version.as_ref();
|
||||||
|
|
||||||
|
if !path.exists() {
|
||||||
|
anyhow::bail!("could not find file at: {}", path.display());
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Ok(metadata) = path.metadata() {
|
||||||
|
if !metadata.is_file() {
|
||||||
|
anyhow::bail!("{} is not a file", path.display());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let abs_path = path.canonicalize().context(anyhow::anyhow!(
|
||||||
|
"could not get absolute path from {}",
|
||||||
|
path.display()
|
||||||
|
))?;
|
||||||
|
|
||||||
|
let output = std::process::Command::new("jq")
|
||||||
|
.arg("--arg")
|
||||||
|
.arg("version")
|
||||||
|
.arg(next_version)
|
||||||
|
.arg(&self.jq)
|
||||||
|
.arg(
|
||||||
|
abs_path
|
||||||
|
.to_str()
|
||||||
|
.ok_or(anyhow::anyhow!("path contains non utf-8 chars"))?,
|
||||||
|
)
|
||||||
|
.output()
|
||||||
|
.context(anyhow::anyhow!(
|
||||||
|
"failed to run jq on file, jq may not be installed or query was invalid"
|
||||||
|
))?;
|
||||||
|
|
||||||
|
if !output.status.success() {
|
||||||
|
let err_content = std::str::from_utf8(output.stderr.as_slice())?;
|
||||||
|
anyhow::bail!("failed to run jq with output: {}", err_content);
|
||||||
|
}
|
||||||
|
|
||||||
|
let edited_json_content = std::str::from_utf8(output.stdout.as_slice())?;
|
||||||
|
tracing::trace!(
|
||||||
|
new_content = edited_json_content,
|
||||||
|
file = &abs_path.display().to_string(),
|
||||||
|
"applied jq to file"
|
||||||
|
);
|
||||||
|
std::fs::write(abs_path, edited_json_content)?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
6
crates/cuddle-please-release-strategy/src/lib.rs
Normal file
6
crates/cuddle-please-release-strategy/src/lib.rs
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
#[cfg(feature = "json-edit")]
|
||||||
|
mod json_edit;
|
||||||
|
mod strategy;
|
||||||
|
|
||||||
|
#[cfg(feature = "json-edit")]
|
||||||
|
pub use json_edit::JsonEditOptions;
|
59
crates/cuddle-please-release-strategy/src/strategy.rs
Normal file
59
crates/cuddle-please-release-strategy/src/strategy.rs
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
use std::path::PathBuf;
|
||||||
|
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
pub struct UpdateOptions {
|
||||||
|
next_version: String,
|
||||||
|
global_changelog: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub type Projects = Vec<Project>;
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Deserialize, Serialize)]
|
||||||
|
pub struct Project {
|
||||||
|
path: Option<PathBuf>,
|
||||||
|
r#type: ProjectType,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Deserialize, Serialize)]
|
||||||
|
#[serde(tag = "type")]
|
||||||
|
pub enum ProjectType {
|
||||||
|
#[cfg(feature = "rust-workspace")]
|
||||||
|
#[serde(alias = "rust_workspace")]
|
||||||
|
RustWorkspace,
|
||||||
|
#[cfg(feature = "rust-crate")]
|
||||||
|
#[serde(alias = "json_edit")]
|
||||||
|
RustCrate,
|
||||||
|
#[cfg(feature = "toml-edit")]
|
||||||
|
#[serde(alias = "toml_edit")]
|
||||||
|
TomlEdit,
|
||||||
|
#[cfg(feature = "yaml-edit")]
|
||||||
|
#[serde(alias = "yaml_edit")]
|
||||||
|
YamlEdit,
|
||||||
|
#[cfg(feature = "json-edit")]
|
||||||
|
#[serde(alias = "json_edit")]
|
||||||
|
JsonEdit,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Project {
|
||||||
|
pub fn new(path: Option<PathBuf>, r#type: ProjectType) -> Self {
|
||||||
|
Self { path, r#type }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn execute(&self, options: &UpdateOptions) -> anyhow::Result<()> {
|
||||||
|
match self.r#type {
|
||||||
|
#[cfg(feature = "rust-workspace")]
|
||||||
|
ProjectType::RustWorkspace => todo!(),
|
||||||
|
#[cfg(feature = "rust-crate")]
|
||||||
|
ProjectType::RustCrate => todo!(),
|
||||||
|
#[cfg(feature = "toml-edit")]
|
||||||
|
ProjectType::TomlEdit => todo!(),
|
||||||
|
#[cfg(feature = "yaml-edit")]
|
||||||
|
ProjectType::YamlEdit => todo!(),
|
||||||
|
#[cfg(feature = "json-edit")]
|
||||||
|
ProjectType::JsonEdit => todo!(),
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
50
crates/cuddle-please-release-strategy/tests/json_edit.rs
Normal file
50
crates/cuddle-please-release-strategy/tests/json_edit.rs
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
use tracing_test::traced_test;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[traced_test]
|
||||||
|
#[cfg(feature = "json-edit")]
|
||||||
|
pub fn test_can_update_version_in_jq() {
|
||||||
|
use cuddle_please_release_strategy::JsonEditOptions;
|
||||||
|
|
||||||
|
let dir = tempdir::TempDir::new("can_update_version_in_jq").unwrap();
|
||||||
|
let dir_path = dir.path();
|
||||||
|
let json_file = dir_path.join("some-test.json");
|
||||||
|
let initial_content = r#"{
|
||||||
|
"some": {
|
||||||
|
"nested": [
|
||||||
|
{
|
||||||
|
"structure": {
|
||||||
|
"version": "v1.0.1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"#;
|
||||||
|
|
||||||
|
let expected = r#"{
|
||||||
|
"some": {
|
||||||
|
"nested": [
|
||||||
|
{
|
||||||
|
"structure": {
|
||||||
|
"version": "v1.0.2"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"#;
|
||||||
|
|
||||||
|
std::fs::write(&json_file, initial_content).unwrap();
|
||||||
|
let actual_file = std::fs::read_to_string(&json_file).unwrap();
|
||||||
|
pretty_assertions::assert_eq!(initial_content, actual_file);
|
||||||
|
|
||||||
|
let edit_options = JsonEditOptions {
|
||||||
|
jq: r#".some.nested[].structure.version=$version"#.into(),
|
||||||
|
};
|
||||||
|
|
||||||
|
edit_options.execute(&json_file, "v1.0.2").unwrap();
|
||||||
|
|
||||||
|
let actual_file = std::fs::read_to_string(&json_file).unwrap();
|
||||||
|
pretty_assertions::assert_eq!(expected, &actual_file);
|
||||||
|
}
|
@ -8,7 +8,5 @@ fn main() -> anyhow::Result<()> {
|
|||||||
|
|
||||||
PleaseCommand::new().execute(current_dir)?;
|
PleaseCommand::new().execute(current_dir)?;
|
||||||
|
|
||||||
//something else asdfa 123
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -25,7 +25,6 @@ PleaseConfig
|
|||||||
api_url: https://some-example.gitea-instance
|
api_url: https://some-example.gitea-instance
|
||||||
"#;
|
"#;
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[traced_test]
|
#[traced_test]
|
||||||
fn test_config_from_current_dir() {
|
fn test_config_from_current_dir() {
|
||||||
let args = get_base_args();
|
let args = get_base_args();
|
||||||
@ -39,7 +38,6 @@ fn test_config_from_current_dir() {
|
|||||||
assert_output(ui, EXPECTED_OUTPUT, "");
|
assert_output(ui, EXPECTED_OUTPUT, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[traced_test]
|
#[traced_test]
|
||||||
fn test_config_from_source_dir() {
|
fn test_config_from_source_dir() {
|
||||||
let mut args = get_base_args();
|
let mut args = get_base_args();
|
||||||
@ -55,7 +53,6 @@ fn test_config_from_source_dir() {
|
|||||||
assert_output(ui, EXPECTED_OUTPUT, "");
|
assert_output(ui, EXPECTED_OUTPUT, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[traced_test]
|
#[traced_test]
|
||||||
fn test_config_from_stdin() {
|
fn test_config_from_stdin() {
|
||||||
let mut args = get_base_args();
|
let mut args = get_base_args();
|
||||||
@ -75,7 +72,6 @@ settings:
|
|||||||
assert_output(ui, EXPECTED_OUTPUT, "");
|
assert_output(ui, EXPECTED_OUTPUT, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[traced_test]
|
#[traced_test]
|
||||||
fn test_config_fails_when_not_path_is_set() {
|
fn test_config_fails_when_not_path_is_set() {
|
||||||
let args = get_base_args();
|
let args = get_base_args();
|
||||||
|
12
cuddle.yaml
12
cuddle.yaml
@ -16,6 +16,18 @@ please:
|
|||||||
branch: main
|
branch: main
|
||||||
settings:
|
settings:
|
||||||
api_url: https://git.front.kjuulh.io
|
api_url: https://git.front.kjuulh.io
|
||||||
|
actions:
|
||||||
|
rust:
|
||||||
|
|
||||||
|
components:
|
||||||
|
packages:
|
||||||
|
debian:
|
||||||
|
dev:
|
||||||
|
- jq
|
||||||
|
- git
|
||||||
|
release:
|
||||||
|
- jq
|
||||||
|
- git
|
||||||
|
|
||||||
scripts:
|
scripts:
|
||||||
"mkdocs:new":
|
"mkdocs:new":
|
||||||
|
@ -12,6 +12,4 @@ fi
|
|||||||
$CMD_PREFIX pull-request \
|
$CMD_PREFIX pull-request \
|
||||||
--mkdocs-image "$MKDOCS_IMAGE" \
|
--mkdocs-image "$MKDOCS_IMAGE" \
|
||||||
--caddy-image "$CADDY_IMAGE" \
|
--caddy-image "$CADDY_IMAGE" \
|
||||||
--image "$REGISTRY/$SERVICE" \
|
|
||||||
--tag "main-$(date +%s)" \
|
|
||||||
--bin-name "$SERVICE"
|
--bin-name "$SERVICE"
|
17
scripts/ci:release.sh
Executable file
17
scripts/ci:release.sh
Executable file
@ -0,0 +1,17 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
CMD_PREFIX="cargo run -p ci --"
|
||||||
|
|
||||||
|
if [[ -n "$CI_PREFIX" ]]; then
|
||||||
|
CMD_PREFIX="$CI_PREFIX"
|
||||||
|
fi
|
||||||
|
|
||||||
|
|
||||||
|
$CMD_PREFIX release \
|
||||||
|
--mkdocs-image "$MKDOCS_IMAGE" \
|
||||||
|
--caddy-image "$CADDY_IMAGE" \
|
||||||
|
--image "$REGISTRY/$SERVICE" \
|
||||||
|
--tag "$DRONE_TAG" \
|
||||||
|
--bin-name "$SERVICE"
|
Loading…
Reference in New Issue
Block a user