mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2024-10-01 11:05:46 +00:00
10676b74dc
In order to allow for human readable output for the VM configuration, we pull it out of the snapshot, which becomes effectively the list of states from the VM. The configuration is stored through a dedicated file in JSON format (not including any binary output). Having the ability to read and modify the VM configuration manually between the snapshot and restore phases makes debugging easier, as well as empowers users for extending the use cases relying on the snapshot/restore feature. Signed-off-by: Sebastien Boeuf <sebastien.boeuf@intel.com>
73 lines
2.3 KiB
Rust
73 lines
2.3 KiB
Rust
// Copyright © 2020 Intel Corporation
|
|
//
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
use crate::{
|
|
config::VmConfig,
|
|
vm::{VmSnapshot, VM_SNAPSHOT_ID},
|
|
};
|
|
use anyhow::anyhow;
|
|
use std::fs::File;
|
|
use std::io::BufReader;
|
|
use std::path::PathBuf;
|
|
use vm_migration::{MigratableError, Snapshot};
|
|
|
|
pub const SNAPSHOT_STATE_FILE: &str = "state.json";
|
|
pub const SNAPSHOT_CONFIG_FILE: &str = "config.json";
|
|
|
|
pub fn url_to_path(url: &str) -> std::result::Result<PathBuf, MigratableError> {
|
|
let path: PathBuf = url
|
|
.strip_prefix("file://")
|
|
.ok_or_else(|| {
|
|
MigratableError::MigrateSend(anyhow!("Could not extract path from URL: {}", url))
|
|
})
|
|
.map(|s| s.into())?;
|
|
|
|
if !path.is_dir() {
|
|
return Err(MigratableError::MigrateSend(anyhow!(
|
|
"Destination is not a directory"
|
|
)));
|
|
}
|
|
|
|
Ok(path)
|
|
}
|
|
|
|
pub fn recv_vm_config(source_url: &str) -> std::result::Result<VmConfig, MigratableError> {
|
|
let mut vm_config_path = url_to_path(source_url)?;
|
|
|
|
vm_config_path.push(SNAPSHOT_CONFIG_FILE);
|
|
|
|
// Try opening the snapshot file
|
|
let vm_config_file =
|
|
File::open(vm_config_path).map_err(|e| MigratableError::MigrateSend(e.into()))?;
|
|
let vm_config_reader = BufReader::new(vm_config_file);
|
|
serde_json::from_reader(vm_config_reader).map_err(|e| MigratableError::MigrateReceive(e.into()))
|
|
}
|
|
|
|
pub fn recv_vm_state(source_url: &str) -> std::result::Result<Snapshot, MigratableError> {
|
|
let mut vm_state_path = url_to_path(source_url)?;
|
|
|
|
vm_state_path.push(SNAPSHOT_STATE_FILE);
|
|
|
|
// Try opening the snapshot file
|
|
let vm_state_file =
|
|
File::open(vm_state_path).map_err(|e| MigratableError::MigrateSend(e.into()))?;
|
|
let vm_state_reader = BufReader::new(vm_state_file);
|
|
serde_json::from_reader(vm_state_reader).map_err(|e| MigratableError::MigrateReceive(e.into()))
|
|
}
|
|
|
|
pub fn get_vm_snapshot(snapshot: &Snapshot) -> std::result::Result<VmSnapshot, MigratableError> {
|
|
if let Some(vm_section) = snapshot
|
|
.snapshot_data
|
|
.get(&format!("{}-section", VM_SNAPSHOT_ID))
|
|
{
|
|
return serde_json::from_slice(&vm_section.snapshot).map_err(|e| {
|
|
MigratableError::Restore(anyhow!("Could not deserialize VM snapshot {}", e))
|
|
});
|
|
}
|
|
|
|
Err(MigratableError::Restore(anyhow!(
|
|
"Could not find VM config snapshot section"
|
|
)))
|
|
}
|