summaryrefslogtreecommitdiff
path: root/src/cmd
diff options
context:
space:
mode:
Diffstat (limited to 'src/cmd')
-rw-r--r--src/cmd/mod.rs6
-rw-r--r--src/cmd/new.rs32
-rw-r--r--src/cmd/shutdown.rs19
-rw-r--r--src/cmd/start.rs31
4 files changed, 74 insertions, 14 deletions
diff --git a/src/cmd/mod.rs b/src/cmd/mod.rs
index 606e326..8ae8f70 100644
--- a/src/cmd/mod.rs
+++ b/src/cmd/mod.rs
@@ -12,5 +12,11 @@ pub use list::list;
pub mod delete;
pub use delete::delete;
+pub mod start;
+pub use start::start;
+
+pub mod shutdown;
+pub use shutdown::shutdown;
+
pub mod cloud_init;
pub use cloud_init::cloud_init;
diff --git a/src/cmd/new.rs b/src/cmd/new.rs
index 57f69ae..5484c08 100644
--- a/src/cmd/new.rs
+++ b/src/cmd/new.rs
@@ -3,13 +3,13 @@
use crate::cloudinit::{CloudInitConfig, CloudInitError};
use crate::image::{ImageError, VirtualMachineImage};
use crate::install::{virt_install, VirtInstallArgs, VirtInstallError};
+use crate::libvirt::{Libvirt, VirtError};
use crate::spec::Specification;
+use crate::util::wait_for_ssh;
use bytesize::GIB;
use log::info;
-use std::net::TcpStream;
-
-const SSH_PORT: i32 = 22;
+use tempfile::tempdir;
/// Errors returned by this module.
#[derive(Debug, thiserror::Error)]
@@ -25,6 +25,14 @@ pub enum NewError {
/// Problem with libvirt.
#[error(transparent)]
VirtInstallError(#[from] VirtInstallError),
+
+ /// Problem with virsh.
+ #[error(transparent)]
+ VirshError(#[from] std::io::Error),
+
+ /// Problem from libvirt server.
+ #[error(transparent)]
+ VirtError(#[from] VirtError),
}
/// The `new` sub-command.
@@ -32,6 +40,7 @@ pub enum NewError {
/// Create all the new virtual machines specified by the caller. Wait
/// until each VM's SSH port listens for connections.
pub fn new(specs: &[Specification]) -> Result<(), NewError> {
+ let libvirt = Libvirt::connect("qemu:///system")?;
for spec in specs {
info!("creating new VM {}", spec.name);
@@ -49,23 +58,18 @@ pub fn new(specs: &[Specification]) -> Result<(), NewError> {
image.resize(spec.image_size_gib * GIB)?;
info!("creating VM");
+ let dir = tempdir()?;
+ let iso = dir.path().join("cloudinit.iso");
let mut args = VirtInstallArgs::new(&spec.name, &image, &init);
args.set_memory(spec.memory_mib);
args.set_vcpus(spec.cpus);
- virt_install(&args)?;
+ virt_install(&args, &iso)?;
info!("waiting for {} to open its SSH port", spec.name);
- wait_for_port(&spec.name, SSH_PORT)?;
+ wait_for_ssh(&spec.name);
+
+ libvirt.detach_cloud_init_iso(&spec.name)?;
}
Ok(())
}
-
-fn wait_for_port(name: &str, port: i32) -> Result<(), NewError> {
- let addr = format!("{}:{}", name, port);
- loop {
- if TcpStream::connect(&addr).is_ok() {
- return Ok(());
- }
- }
-}
diff --git a/src/cmd/shutdown.rs b/src/cmd/shutdown.rs
new file mode 100644
index 0000000..b53ebd3
--- /dev/null
+++ b/src/cmd/shutdown.rs
@@ -0,0 +1,19 @@
+//! The `shutdown` sub-command.
+
+use crate::libvirt::{Libvirt, VirtError};
+use crate::spec::Specification;
+use log::{debug, info};
+
+/// Shut down VMs corresponding to specifications.
+pub fn shutdown(specs: &[Specification]) -> Result<(), VirtError> {
+ let libvirt = Libvirt::connect("qemu:///system")?;
+ for spec in specs {
+ info!("shutting down virtual machine {}", spec.name);
+ libvirt.shutdown(&spec.name)?;
+ }
+ for spec in specs {
+ debug!("waiting for {} to become inactive", spec.name);
+ libvirt.wait_for_inactive(&spec.name)?;
+ }
+ Ok(())
+}
diff --git a/src/cmd/start.rs b/src/cmd/start.rs
new file mode 100644
index 0000000..a69f54c
--- /dev/null
+++ b/src/cmd/start.rs
@@ -0,0 +1,31 @@
+//! The `start` sub-command.
+
+use crate::libvirt::{Libvirt, VirtError};
+use crate::spec::Specification;
+use crate::util::wait_for_ssh;
+use log::info;
+
+/// Errors from this module.
+#[derive(Debug, thiserror::Error)]
+pub enum StartError {
+ /// Error from libvirt.
+ #[error(transparent)]
+ VirtError(#[from] VirtError),
+
+ /// Error doing I/O.
+ #[error(transparent)]
+ IoError(#[from] std::io::Error),
+}
+
+/// Start existing VMs corresponding to specifications.
+pub fn start(specs: &[Specification]) -> Result<(), StartError> {
+ let libvirt = Libvirt::connect("qemu:///system")?;
+ for spec in specs {
+ info!("starting virtual machine {}", spec.name);
+ libvirt.start(&spec.name)?;
+ }
+ for spec in specs {
+ wait_for_ssh(&spec.name);
+ }
+ Ok(())
+}