Skip to content

Commit

Permalink
builder: support --parent-bootstrap for merge
Browse files Browse the repository at this point in the history
This option allows merging multiple bootstraps of upper layer with
the bootstrap of a parent image, so that we can implement container
commit operation for nydus image.

Signed-off-by: Yan Song <[email protected]>
  • Loading branch information
imeoer committed Mar 9, 2023
1 parent 2a83e06 commit ef2033c
Show file tree
Hide file tree
Showing 2 changed files with 40 additions and 8 deletions.
13 changes: 12 additions & 1 deletion src/bin/nydus-image/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -331,11 +331,18 @@ fn prepare_cmd_args(bti_string: String) -> ArgMatches<'static> {
.subcommand(
SubCommand::with_name("merge")
.about("Merge multiple bootstraps into a overlaid bootstrap")
.arg(
Arg::with_name("parent-bootstrap")
.long("parent-bootstrap")
.help("File path of the parent/referenced RAFS metadata blob (optional)")
.required(false)
.takes_value(true),
)
.arg(
Arg::with_name("bootstrap")
.long("bootstrap")
.short("B")
.help("output path of nydus overlaid bootstrap")
.help("Output path of nydus overlaid bootstrap")
.required(true)
.takes_value(true),
)
Expand Down Expand Up @@ -704,8 +711,12 @@ impl Command {
prefetch: Self::get_prefetch(matches)?,
..Default::default()
};

let parent_bootstrap_path = matches.value_of("parent-bootstrap");

let output = Merger::merge(
&mut ctx,
parent_bootstrap_path,
source_bootstrap_paths,
target_bootstrap_path.to_path_buf(),
chunk_dict_path,
Expand Down
35 changes: 28 additions & 7 deletions src/bin/nydus-image/merge.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
//
// SPDX-License-Identifier: Apache-2.0

use std::collections::HashSet;
use std::collections::{HashMap, HashSet};
use std::convert::TryFrom;
use std::path::{Path, PathBuf};

Expand Down Expand Up @@ -61,6 +61,7 @@ impl Merger {

pub fn merge(
ctx: &mut BuildContext,
parent_bootstrap_path: Option<&str>,
sources: Vec<PathBuf>,
target: PathBuf,
chunk_dict: Option<PathBuf>,
Expand All @@ -73,6 +74,22 @@ impl Merger {
let mut blob_mgr = BlobManager::new();
let mut flags: Option<Flags> = None;

// Load parent bootstrap
let mut blob_idx_map = HashMap::new();
let mut parent_layers = 0;
if let Some(parent_bootstrap_path) = &parent_bootstrap_path {
let rs = RafsSuper::load_from_metadata(parent_bootstrap_path, RafsMode::Direct, true)
.context(format!("load parent bootstrap {:?}", parent_bootstrap_path))?;
tree = Some(Tree::from_bootstrap(&rs, &mut ())?);
let blobs = rs.superblock.get_blob_infos();
for blob in &blobs {
let blob_ctx = BlobContext::from(ctx, blob, ChunkSource::Parent);
blob_idx_map.insert(blob_ctx.blob_id.clone(), blob_mgr.len());
blob_mgr.add(blob_ctx);
}
parent_layers = blobs.len();
}

// Get the blobs come from chunk dict bootstrap.
let mut chunk_dict_blobs = HashSet::new();
if let Some(chunk_dict_path) = &chunk_dict {
Expand Down Expand Up @@ -117,7 +134,6 @@ impl Merger {

let parent_blobs = rs.superblock.get_blob_infos();
let blob_hash = Self::get_blob_hash(bootstrap_path)?;
let mut blob_idx_map = Vec::new();
let mut parent_blob_added = false;

for blob in &parent_blobs {
Expand All @@ -135,8 +151,10 @@ impl Merger {
chunk_size = Some(blob_ctx.chunk_size);
parent_blob_added = true;
}
blob_idx_map.push(blob_mgr.len() as u32);
blob_mgr.add(blob_ctx);
if !blob_idx_map.contains_key(blob.blob_id()) {
blob_idx_map.insert(blob.blob_id().to_string(), blob_mgr.len());
blob_mgr.add(blob_ctx);
}
}

if let Some(tree) = &mut tree {
Expand All @@ -153,16 +171,19 @@ impl Merger {
))?;
for chunk in &mut node.chunks {
let origin_blob_index = chunk.inner.blob_index() as usize;
// Set the blob index of chunk to real index in blob table of final bootstrap.
chunk.inner.set_blob_index(blob_idx_map[origin_blob_index]);
let blob_ctx = parent_blobs[origin_blob_index].as_ref();
if let Some(blob_index) = blob_idx_map.get(blob_ctx.blob_id()) {
// Set the blob index of chunk to real index in blob table of final bootstrap.
chunk.inner.set_blob_index(*blob_index as u32);
}
}
// Set node's layer index to distinguish same inode number (from bootstrap)
// between different layers.
node.layer_idx = u16::try_from(layer_idx).context(format!(
"too many layers {}, limited to {}",
layer_idx,
u16::MAX
))?;
))? + parent_layers as u16;
node.overlay = Overlay::UpperAddition;
match node.whiteout_type(WhiteoutSpec::Oci) {
Some(_) => {
Expand Down

0 comments on commit ef2033c

Please sign in to comment.