Skip to content

Commit 36edc47

Browse files
Bring back mount_media, remove_media
Functions mount_media and remove_media were removed in #57. To parse a ovf_env file, however, both mount_media and remove_media are necessary. Bring back the functions. Adjust return types according to the new LibError interface as well. Define const PATH_MOUNT_DEVICE, PATH_MOUNT_POINT for path to mount point for media source and target to be mounted, and replace hard-coded path with the new const. Co-authored-by: Jeremy Cline <[email protected]>
1 parent 49775b4 commit 36edc47

File tree

2 files changed

+94
-11
lines changed

2 files changed

+94
-11
lines changed

libazureinit/src/media.rs

+78-9
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,8 @@ use std::fs::create_dir_all;
66
use std::fs::File;
77
use std::io::Read;
88
use std::os::unix::fs::PermissionsExt;
9+
use std::path::PathBuf;
10+
use std::process::Command;
911

1012
use serde::Deserialize;
1113
use serde_xml_rs::from_str;
@@ -66,18 +68,85 @@ fn default_preprov_type() -> String {
6668
"None".to_owned()
6769
}
6870

69-
pub fn make_temp_directory() -> Result<(), Box<dyn std::error::Error>> {
70-
let file_path = "/run/azure-init/tmp/";
71+
pub const PATH_MOUNT_DEVICE: &str = "/dev/sr0";
72+
pub const PATH_MOUNT_POINT: &str = "/run/azure-init/media/";
7173

72-
create_dir_all(file_path)?;
74+
// Some zero-sized structs that just provide states for our state machine
75+
pub struct Mounted;
76+
pub struct Unmounted;
7377

74-
let metadata = fs::metadata(file_path)?;
75-
let permissions = metadata.permissions();
76-
let mut new_permissions = permissions.clone();
77-
new_permissions.set_mode(0o700);
78-
fs::set_permissions(file_path, new_permissions)?;
78+
pub struct Media<State = Unmounted> {
79+
device_path: PathBuf,
80+
mount_path: PathBuf,
81+
state: std::marker::PhantomData<State>,
82+
}
83+
84+
impl Media<Unmounted> {
85+
pub fn new(device_path: PathBuf, mount_path: PathBuf) -> Media<Unmounted> {
86+
Media {
87+
device_path,
88+
mount_path,
89+
state: std::marker::PhantomData,
90+
}
91+
}
92+
93+
// The only thing you can do with the Media struct is call mount which, if it succeeds,
94+
// moves it into the Mounted state. From there the only thing you can do with it is
95+
// call unmount(), but you could make other functionality available by adding to the
96+
// impl Media<Mounted> block.
97+
pub fn mount(self) -> Result<Media<Mounted>, Error> {
98+
create_dir_all(&self.mount_path)?;
99+
100+
let metadata = fs::metadata(&self.mount_path)?;
101+
let permissions = metadata.permissions();
102+
let mut new_permissions = permissions.clone();
103+
new_permissions.set_mode(0o700);
104+
fs::set_permissions(&self.mount_path, new_permissions)?;
105+
106+
let mount_status = Command::new("mount")
107+
.arg("-o")
108+
.arg("ro")
109+
.arg(&self.device_path)
110+
.arg(&self.mount_path)
111+
.status()?;
79112

80-
Ok(())
113+
if !mount_status.success() {
114+
Err(Error::SubprocessFailed {
115+
command: "mount".to_string(),
116+
status: mount_status,
117+
})
118+
} else {
119+
Ok(Media {
120+
device_path: self.device_path,
121+
mount_path: self.mount_path,
122+
state: std::marker::PhantomData,
123+
})
124+
}
125+
}
126+
}
127+
128+
impl Media<Mounted> {
129+
pub fn unmount(self) -> Result<(), Error> {
130+
let umount_status =
131+
Command::new("umount").arg(self.mount_path).status()?;
132+
if !umount_status.success() {
133+
return Err(Error::SubprocessFailed {
134+
command: "umount".to_string(),
135+
status: umount_status,
136+
});
137+
}
138+
139+
let eject_status =
140+
Command::new("eject").arg(self.device_path).status()?;
141+
if !eject_status.success() {
142+
Err(Error::SubprocessFailed {
143+
command: "eject".to_string(),
144+
status: eject_status,
145+
})
146+
} else {
147+
Ok(())
148+
}
149+
}
81150
}
82151

83152
pub fn read_ovf_env_to_string() -> Result<String, Error> {

src/main.rs

+16-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
// Copyright (c) Microsoft Corporation.
22
// Licensed under the MIT License.
33

4+
use std::path::PathBuf;
45
use std::process::ExitCode;
56

67
use anyhow::Context;
@@ -9,21 +10,34 @@ use libazureinit::distro::{Distribution, Distributions};
910
use libazureinit::{
1011
error::Error as LibError,
1112
goalstate, imds, media,
13+
media::Media,
1214
reqwest::{header, Client},
1315
user,
1416
};
1517

1618
const VERSION: &str = env!("CARGO_PKG_VERSION");
1719

18-
fn get_username(imds_body: String) -> Result<String, LibError> {
20+
fn get_username(imds_body: String) -> Result<String, anyhow::Error> {
1921
if imds::is_password_authentication_disabled(&imds_body)? {
2022
// password authentication is disabled
21-
imds::get_username(imds_body.clone())
23+
Ok(imds::get_username(imds_body.clone())?)
2224
} else {
2325
// password authentication is enabled
26+
let mount_media = Media::new(
27+
PathBuf::from(media::PATH_MOUNT_DEVICE),
28+
PathBuf::from(media::PATH_MOUNT_POINT),
29+
);
30+
let mounted = mount_media
31+
.mount()
32+
.with_context(|| "Failed to mount media.")?;
33+
2434
let ovf_body = media::read_ovf_env_to_string()?;
2535
let environment = media::parse_ovf_env(ovf_body.as_str())?;
2636

37+
mounted
38+
.unmount()
39+
.with_context(|| "Failed to remove media.")?;
40+
2741
Ok(environment
2842
.provisioning_section
2943
.linux_prov_conf_set

0 commit comments

Comments
 (0)