pipette_protocol/
lib.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

//! The pipette protocol used for host-to-guest agent communications. It is
//! defined as messages over a mesh point-to-point connection.

#![forbid(unsafe_code)]

use mesh::MeshPayload;
use mesh::pipe::ReadPipe;
use mesh::pipe::WritePipe;
use mesh::rpc::FailableRpc;
use mesh::rpc::Rpc;

/// The port used for the pipette connection over AF_VSOCK.
pub const PIPETTE_VSOCK_PORT: u32 = 0x1337;

/// The bootstrap message sent from the agent to the host.
#[derive(MeshPayload)]
pub struct PipetteBootstrap {
    /// The sender for requests to the agent.
    pub requests: mesh::Sender<PipetteRequest>,
    /// The receiver for diagnostics files from the agent.
    pub diag_file_recv: mesh::Receiver<DiagnosticFile>,
    /// The receiver on a channel closed when the agent exits.
    pub watch: mesh::OneshotReceiver<()>,
    /// The log channel.
    pub log: ReadPipe,
}

/// A request to the agent.
#[derive(MeshPayload)]
pub enum PipetteRequest {
    /// Pings the agent to check if it's alive.
    Ping(Rpc<(), ()>),
    /// Executes a command inside the guest.
    Execute(FailableRpc<ExecuteRequest, ExecuteResponse>),
    /// Powers off or reboots the guest.
    ///
    /// A successful response to this request may be lost depending on when
    /// pipette is terminated during the shutdown process.
    Shutdown(FailableRpc<ShutdownRequest, ()>),
    /// Reads the full contents of a file.
    ReadFile(FailableRpc<ReadFileRequest, ()>),
    /// Writes a file
    WriteFile(FailableRpc<WriteFileRequest, ()>),
}

/// A request to execute a command inside the guest.
#[derive(MeshPayload, Default)]
pub struct ExecuteRequest {
    /// The program to execute.
    pub program: String,
    /// The arguments to the program.
    pub args: Vec<String>,
    /// The current working directory for the program.
    pub current_dir: Option<String>,
    /// The stdin for the program.
    pub stdin: Option<ReadPipe>,
    /// The stdout for the program.
    pub stdout: Option<WritePipe>,
    /// The stderr for the program.
    pub stderr: Option<WritePipe>,
    /// The environment variables for the program.
    pub env: Vec<EnvPair>,
    /// Whether to clear the environment before setting the new environment.
    pub clear_env: bool,
}

impl std::fmt::Debug for ExecuteRequest {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("ExecuteRequest")
            .field("program", &self.program)
            .field("args", &self.args)
            .field("current_dir", &self.current_dir)
            .field("stdin", &self.stdin.is_some())
            .field("stdout", &self.stdout.is_some())
            .field("stderr", &self.stderr.is_some())
            .field("env", &self.env)
            .field("clear_env", &self.clear_env)
            .finish()
    }
}

/// A pair of environment variable name and value.
#[derive(MeshPayload, Clone, Debug)]
pub struct EnvPair {
    /// The name of the environment variable.
    pub name: String,
    /// The value of the environment variable, or `None` to remove the variable.
    pub value: Option<String>,
}

/// The response to a request to execute a command inside the guest.
#[derive(MeshPayload)]
pub struct ExecuteResponse {
    /// The process ID of the executed command.
    pub pid: u32,
    /// The process result channel. Receives the exit status of the process.
    pub result: mesh::OneshotReceiver<ExitStatus>,
}

/// The exit status of a process.
#[derive(Debug, MeshPayload, Clone)]
pub enum ExitStatus {
    /// The process exited normally with the given exit code.
    Normal(i32),
    /// The process was terminated by the given signal.
    Signal(i32),
    /// The process exited with an unknown status.
    Unknown,
}

impl From<std::process::ExitStatus> for ExitStatus {
    fn from(status: std::process::ExitStatus) -> Self {
        if let Some(code) = status.code() {
            return Self::Normal(code);
        }
        #[cfg(unix)]
        if let Some(signal) = std::os::unix::process::ExitStatusExt::signal(&status) {
            return Self::Signal(signal);
        }
        Self::Unknown
    }
}

/// A request to power off or reboot the guest.
#[derive(Copy, Clone, MeshPayload)]
pub struct ShutdownRequest {
    /// The type of shutdown to perform.
    pub shutdown_type: ShutdownType,
}

/// The type of shutdown to perform.
#[derive(Copy, Clone, Debug, MeshPayload)]
pub enum ShutdownType {
    /// Powers off the guest.
    PowerOff,
    /// Reboots the guest.
    Reboot,
}

/// A request to read a file.
#[derive(MeshPayload)]
pub struct ReadFileRequest {
    /// The path to read the file from.
    pub path: String,
    /// The sender for the contents of the file.
    pub sender: WritePipe,
}

/// A request to write a file.
#[derive(MeshPayload)]
pub struct WriteFileRequest {
    /// The path to write the file to.
    pub path: String,
    /// The receiver of the contents of the file.
    pub receiver: ReadPipe,
}

/// A file that the guest client wishes to be logged on the host for diagnostic purposes.
#[derive(MeshPayload)]
pub struct DiagnosticFile {
    /// The name of the file.
    pub name: String,
    /// The receiver of the contents of the file.
    pub receiver: ReadPipe,
}