Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: support VanillaOS images #206

Draft
wants to merge 14 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions recipe/src/recipe.rs
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,11 @@ pub struct Recipe<'a> {
#[builder(setter(into))]
pub base_image: Cow<'a, str>,

/// The base image type for user's image.
#[serde(alias = "base-image-type", skip_serializing_if = "Option::is_none")]
#[builder(setter(into))]
pub base_image_type: Option<Cow<'a, str>>,

/// The version/tag of the base image.
#[serde(alias = "image-version")]
#[builder(setter(into))]
Expand Down
80 changes: 62 additions & 18 deletions src/commands/generate.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,9 @@ use std::{
};

use blue_build_recipe::Recipe;
use blue_build_template::{ContainerFileTemplate, Template};
use blue_build_template::{
ContainerFileTemplate, OstreeContainerFileTemplate, VanillaContainerFileTemplate,
};
use blue_build_utils::{
constants::{
CI_PROJECT_NAME, CI_PROJECT_NAMESPACE, CI_REGISTRY, CONFIG_PATH, GITHUB_REPOSITORY_OWNER,
Expand Down Expand Up @@ -114,11 +116,40 @@ impl GenerateCommand {

info!("Templating for recipe at {}", recipe_path.display());

let template = ContainerFileTemplate::builder()
.os_version(Driver::get_os_version(&recipe_de)?)
let template: Box<dyn ContainerFileTemplate> = match &recipe_de.base_image_type {
Some(cow) => match cow.as_ref() {
"vanilla" => Box::new(self.build_vanilla_template(&recipe_de, &recipe_path)?),
"ostree" => Box::new(self.build_ostree_template(&recipe_de, &recipe_path)?),
_ => Box::new(self.build_ostree_template(&recipe_de, &recipe_path)?),
},
None => Box::new(self.build_ostree_template(&recipe_de, &recipe_path)?),
};

let output_str = template.render().into_diagnostic()?;
if let Some(output) = self.output.as_ref() {
debug!("Templating to file {}", output.display());
trace!("Containerfile:\n{output_str}");

std::fs::write(output, output_str).into_diagnostic()?;
} else {
debug!("Templating to stdout");
syntax_highlighting::print(&output_str, "Dockerfile", self.syntax_theme)?;
}

Ok(())
}

fn build_ostree_template<'a>(
&self,
recipe_de: &'a Recipe<'a>,
recipe_path: &'a Path,
) -> Result<OstreeContainerFileTemplate<'a>> {
info!("Using ostree template");
Ok(OstreeContainerFileTemplate::builder()
.os_version(Driver::get_os_version(recipe_de)?)
.build_id(Driver::get_build_id())
.recipe(&recipe_de)
.recipe_path(recipe_path.as_path())
.recipe(recipe_de)
.recipe_path(recipe_path)
.registry(self.get_registry())
.exports_tag({
#[allow(clippy::const_is_empty)]
Expand All @@ -131,20 +162,33 @@ impl GenerateCommand {
shadow::COMMIT_HASH.to_string()
}
})
.build();

let output_str = template.render().into_diagnostic()?;
if let Some(output) = self.output.as_ref() {
debug!("Templating to file {}", output.display());
trace!("Containerfile:\n{output_str}");

std::fs::write(output, output_str).into_diagnostic()?;
} else {
debug!("Templating to stdout");
syntax_highlighting::print(&output_str, "Dockerfile", self.syntax_theme)?;
}
.build())
}

Ok(())
fn build_vanilla_template<'a>(
&self,
recipe_de: &'a Recipe<'a>,
recipe_path: &'a Path,
) -> Result<VanillaContainerFileTemplate<'a>> {
info!("Using vanilla template");
Ok(VanillaContainerFileTemplate::builder()
.os_version(Driver::get_os_version(recipe_de)?)
.build_id(Driver::get_build_id())
.recipe(recipe_de)
.recipe_path(recipe_path)
.registry(self.get_registry())
.exports_tag({
#[allow(clippy::const_is_empty)]
if shadow::COMMIT_HASH.is_empty() {
// This is done for users who install via
// cargo. Cargo installs do not carry git
// information via shadow
format!("v{}", crate_version!())
} else {
shadow::COMMIT_HASH.to_string()
}
})
.build())
}

fn get_registry(&self) -> String {
Expand Down
9 changes: 2 additions & 7 deletions src/drivers.rs
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ use std::{
};

use blue_build_recipe::Recipe;
use blue_build_utils::constants::IMAGE_VERSION_LABEL;
// use blue_build_utils::constants::IMAGE_VERSION_LABEL;
use log::{debug, info, trace};
use miette::{bail, miette, Result};
use once_cell::sync::Lazy;
Expand Down Expand Up @@ -389,12 +389,7 @@ impl Driver<'_> {
.build();
let inspection = INSPECT_DRIVER.get_metadata(&inspect_opts)?;

let os_version = inspection.get_version().ok_or_else(|| {
miette!(
help = format!("Please check with the image author about using '{IMAGE_VERSION_LABEL}' to report the os version."),
"Unable to get the OS version from the labels"
)
})?;
let os_version = inspection.get_version().unwrap_or(0);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's leave this function alone and instead call a different function for whatever the vanilla template needs. We want to try to keep the fedora templates and the vanilla templates separate as possible.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree. I just did it like this because I couldn't figure out any easier way to fix this error. We should have some way to detect and declare what the base image is, and call different functions based on that, and remove the hard requirement for os_version in the tagging system, etc.

trace!("os_version: {os_version}");

os_version
Expand Down
48 changes: 46 additions & 2 deletions template/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,20 @@ use uuid::Uuid;

pub use rinja::Template;

pub trait ContainerFileTemplate {
/// # Errors
///
/// Will return error from rinja if there is an issue rendering the template.
fn render(&self) -> Result<String, rinja::Error>;
}

#[derive(Debug, Clone, Template, TypedBuilder)]
#[template(path = "Containerfile.j2", escape = "none", whitespace = "minimize")]
pub struct ContainerFileTemplate<'a> {
#[template(
path = "Containerfile.ostree.j2",
escape = "none",
whitespace = "minimize"
)]
pub struct OstreeContainerFileTemplate<'a> {
recipe: &'a Recipe<'a>,

#[builder(setter(into))]
Expand All @@ -31,7 +42,40 @@ pub struct ContainerFileTemplate<'a> {
#[builder(setter(into))]
exports_tag: Cow<'a, str>,
}
impl ContainerFileTemplate for OstreeContainerFileTemplate<'_> {
fn render(&self) -> Result<String, rinja::Error> {
Template::render(&self)
}
}

#[derive(Debug, Clone, Template, TypedBuilder)]
#[template(
path = "Containerfile.vanilla.j2",
escape = "none",
whitespace = "minimize"
)]
pub struct VanillaContainerFileTemplate<'a> {
recipe: &'a Recipe<'a>,

#[builder(setter(into))]
recipe_path: &'a Path,

#[builder(setter(into))]
build_id: Uuid,

os_version: u64,

#[builder(setter(into))]
registry: Cow<'a, str>,

#[builder(setter(into))]
exports_tag: Cow<'a, str>,
}
impl ContainerFileTemplate for VanillaContainerFileTemplate<'_> {
fn render(&self) -> Result<String, rinja::Error> {
Template::render(&self)
}
}
#[derive(Debug, Clone, Template, TypedBuilder)]
#[template(path = "github_issue.j2", escape = "md")]
pub struct GithubIssueTemplate<'a> {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ RUN --mount=type=bind,from=stage-bins,src=/bins,dst=/tmp/bins \
&& cp /tmp/bins/* /usr/bin/ \
&& ostree container commit

{% call modules::main_modules_run(recipe.modules_ext, os_version) %}
{% call modules::ostree_modules_run(recipe.modules_ext, os_version) %}

RUN rm -fr /tmp/* /var/* && ostree container commit

Expand Down
59 changes: 59 additions & 0 deletions template/templates/Containerfile.vanilla.j2
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
{%- import "modules/modules.j2" as modules -%}
{%- include "stages.j2" %}

# Main image
FROM {{ recipe.base_image }}:{{ recipe.image_version }} AS {{ recipe.name|replace('/', "-") }}

ARG RECIPE={{ recipe_path.display() }}
ARG IMAGE_REGISTRY={{ registry }}

{%- if self::files_dir_exists() %}
ARG CONFIG_DIRECTORY="/tmp/files"
{%- else if self::config_dir_exists() %}
ARG CONFIG_DIRECTORY="/tmp/config"
{%- endif %}
ARG MODULE_DIRECTORY="/tmp/modules"
ARG IMAGE_NAME="{{ recipe.name }}"
ARG BASE_IMAGE="{{ recipe.base_image }}"

# Key RUN
# RUN --mount=type=bind,from=stage-keys,src=/keys,dst=/tmp/keys \
# mkdir -p /usr/etc/pki/containers/ \
# && cp /tmp/keys/* /usr/etc/pki/containers/

# Bin RUN
RUN --mount=type=bind,from=stage-bins,src=/bins,dst=/tmp/bins \
mkdir -p /usr/bin/ \
&& cp /tmp/bins/* /usr/bin/

# Init step copied from VanillaOS template
RUN lpkg --unlock && apt-get update

{% call modules::generic_modules_run(recipe.modules_ext, os_version) %}

# Cleanup step copied from VanillaOS template
RUN apt-get autoremove -y && apt-get clean && lpkg --lock

# FsGuard step copied from VanillaOS template
# first download the required python script from the vib-fsguard module and the FsGuard binary
RUN mkdir -p /sources/fsguard/ && \
wget https://github.com/linux-immutability-tools/FsGuard/releases/download/v0.1.2-2/FsGuard_0.1.2-2_linux_amd64.tar.gz -O /tmp/fsguard.tar.gz && tar -xf /tmp/fsguard.tar.gz -C /sources/fsguard/ && \
curl https://raw.githubusercontent.com/Vanilla-OS/vib-fsguard/main/genfilelist.py -o /sources/fsguard/genfilelist.py && \
rm -rf /FsGuard && rm -f ./minisign.pub ./minisign.key && chmod +x /usr/sbin/init && mkdir /FsGuard && \
chmod +x /sources/fsguard/genfilelist.py && minisign -WG -s ./minisign.key && \
python3 /sources/fsguard/genfilelist.py /usr/bin /FsGuard/filelist /usr/sbin/FsGuard && \
minisign -Sm /FsGuard/filelist -p .//minisign.pub -s .//minisign.key && touch /FsGuard/signature && \
echo -n "----begin attach----" >> /FsGuard/signature && cat /FsGuard/filelist.minisig >> /FsGuard/signature && \
echo -n "----begin second attach----" >> /FsGuard/signature && tail -n1 .//minisign.pub >> /FsGuard/signature && \
cat /FsGuard/signature >> /sources/fsguard/FsGuard && mv /sources/fsguard/FsGuard /usr/sbin/FsGuard && rm ./minisign.key ./minisign.pub

RUN rm -fr /tmp/* /var/tmp/* /sources/*

# Labels are added last since they cause cache misses with buildah
LABEL {{ blue_build_utils::constants::BUILD_ID_LABEL }}="{{ build_id }}"
LABEL org.opencontainers.image.title="{{ recipe.name }}"
LABEL org.opencontainers.image.description="{{ recipe.description }}"
{%- if let Some(repo) = self::get_repo_url() %}
LABEL org.opencontainers.image.source="{{ repo }}"
{%- endif %}
LABEL io.artifacthub.package.readme-url=https://raw.githubusercontent.com/blue-build/cli/main/README.md
24 changes: 13 additions & 11 deletions template/templates/modules/modules.j2
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
{% macro main_modules_run(modules_ext, os_version) %}
{% macro generic_modules_run(modules_ext, os_version) %}
# Module RUNs
{%- for module in modules_ext.modules %}
{%- if let Some(module) = module.required_fields %}

{%- if module.no_cache %}
ARG CACHEBUST="{{ build_id }}"
{%- endif %}
Expand All @@ -22,22 +23,18 @@ RUN \
{%- else %}
--mount=type=bind,from=stage-modules,src=/modules,dst=/tmp/modules,rw \
{%- endif %}
{%- if module.module_type == "akmods" %}
--mount=type=bind,from=stage-akmods-{{ module.generate_akmods_info(os_version).stage_name }},src=/rpms,dst=/tmp/rpms,rw \
{%- endif %}
--mount=type=bind,from=ghcr.io/blue-build/cli:{{ exports_tag }}-build-scripts,src=/scripts/,dst=/tmp/scripts/ \
--mount=type=cache,dst=/var/cache/rpm-ostree,id=rpm-ostree-cache-{{ recipe.name }}-{{ recipe.image_version }},sharing=locked \
/tmp/scripts/run_module.sh '{{ module.module_type }}' '{{ module.print_module_context() }}' \
&& ostree container commit
/tmp/scripts/run_module.sh '{{ module.module_type }}' '{{ module.print_module_context() }}'
{%- endif %}
{%- endif %}
{%- endfor %}
{% endmacro %}
{% macro stage_modules_run(modules_ext, os_version) %}


{% macro ostree_modules_run(modules_ext, os_version) %}
# Module RUNs
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The original module macros here should remain unchanged. I would suggest making a new module macro for the vanilla OS.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I split this to generic / ostree macros, because only rpm-ostree module calls require the things added by the ostree macro, VanillaOS, stages, etc., do not and probably never will require anything like ostree container commit. If there is some sort of integration needed like the we have with the akmods module, it would be trivial to split generic into generic and vanilla and switch the vanilla template to use the new macro.

I'm envisioning that we'd also ship a generic base image type, which would not add any OS-specific things to the Containerfile, and could thus be usable for basically any operating system that supports OCI images as a distribution mechanism (without extra work from us, but with extra work from the custom image maintainer).

{%- for module in modules_ext.modules %}
{%- if let Some(module) = module.required_fields %}

{%- if module.no_cache %}
ARG CACHEBUST="{{ build_id }}"
{%- endif %}
Expand All @@ -58,9 +55,14 @@ RUN \
{%- else %}
--mount=type=bind,from=stage-modules,src=/modules,dst=/tmp/modules,rw \
{%- endif %}
{%- if module.module_type == "akmods" %}
--mount=type=bind,from=stage-akmods-{{ module.generate_akmods_info(os_version).stage_name }},src=/rpms,dst=/tmp/rpms,rw \
{%- endif %}
--mount=type=bind,from=ghcr.io/blue-build/cli:{{ exports_tag }}-build-scripts,src=/scripts/,dst=/tmp/scripts/ \
/tmp/scripts/run_module.sh '{{ module.module_type }}' '{{ module.print_module_context() }}'
--mount=type=cache,dst=/var/cache/rpm-ostree,id=rpm-ostree-cache-{{ recipe.name }}-{{ recipe.image_version }},sharing=locked \
/tmp/scripts/run_module.sh '{{ module.module_type }}' '{{ module.print_module_context() }}' \
&& ostree container commit
{%- endif %}
{%- endif %}
{%- endfor %}
{% endmacro %}
{% endmacro %}
2 changes: 1 addition & 1 deletion template/templates/stages.j2
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@ SHELL ["bash", "-c"]
{%- endif %}
{%- endif %}

{% call modules::stage_modules_run(stage.modules_ext, os_version) %}
{% call modules::generic_modules_run(stage.modules_ext, os_version) %}
{%- endif %}
{%- endfor %}
{%- endif %}
Loading