update/
args.rs

1// Copyright 2019 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5use argh::FromArgs;
6
7#[derive(Debug, Eq, FromArgs, PartialEq)]
8/// Manage updates.
9pub struct Update {
10    #[argh(subcommand)]
11    pub cmd: Command,
12}
13
14#[derive(Debug, Eq, FromArgs, PartialEq)]
15#[argh(subcommand)]
16pub enum Command {
17    // fuchsia.update.channelcontrol.ChannelControl protocol:
18    Channel(Channel),
19
20    // fuchsia.update Manager protocol:
21    CheckNow(CheckNow),
22    MonitorUpdates(MonitorUpdates),
23
24    // fuchsia.update.installer protocol:
25    ForceInstall(ForceInstall),
26
27    // fuchsia.update CommitStatusProvider protocol:
28    WaitForCommit(WaitForCommit),
29    Revert(Revert),
30}
31
32#[derive(Debug, Eq, FromArgs, PartialEq)]
33#[argh(subcommand, name = "channel")]
34/// Get the current (running) channel.
35pub struct Channel {
36    #[argh(subcommand)]
37    pub cmd: channel::Command,
38}
39
40pub mod channel {
41    use argh::FromArgs;
42
43    #[derive(Debug, Eq, FromArgs, PartialEq)]
44    #[argh(subcommand)]
45    pub enum Command {
46        Get(Get),
47        Target(Target),
48        Set(Set),
49        List(List),
50    }
51
52    #[derive(Debug, Eq, FromArgs, PartialEq)]
53    #[argh(subcommand, name = "get")]
54    /// Get the current (running) channel.
55    pub struct Get {}
56
57    #[derive(Debug, Eq, FromArgs, PartialEq)]
58    #[argh(subcommand, name = "target")]
59    /// Get the target channel.
60    pub struct Target {}
61
62    #[derive(Debug, Eq, FromArgs, PartialEq)]
63    #[argh(subcommand, name = "set")]
64    /// Set the target channel.
65    pub struct Set {
66        #[argh(positional)]
67        pub channel: String,
68    }
69
70    #[derive(Debug, Eq, FromArgs, PartialEq)]
71    #[argh(subcommand, name = "list")]
72    /// List of known target channels.
73    pub struct List {}
74}
75
76#[derive(Debug, Eq, FromArgs, PartialEq)]
77#[argh(subcommand, name = "check-now")]
78/// Start an update.
79pub struct CheckNow {
80    /// the update check was initiated by a service, in the background.
81    #[argh(switch)]
82    pub service_initiated: bool,
83
84    /// monitor for state update.
85    #[argh(switch)]
86    pub monitor: bool,
87}
88
89#[derive(Debug, Eq, FromArgs, PartialEq)]
90#[argh(subcommand, name = "monitor-updates")]
91/// Monitor all update checks.
92pub struct MonitorUpdates {}
93
94#[derive(Debug, Eq, FromArgs, PartialEq)]
95#[argh(subcommand, name = "force-install")]
96/// Directly invoke the system updater to install the provided update, bypassing any update checks.
97pub struct ForceInstall {
98    /// whether or not the system updater should reboot into the new system.
99    #[argh(option, default = "true")]
100    pub reboot: bool,
101
102    /// the url of the update package describing the update to install. Ex:
103    /// fuchsia-pkg://fuchsia.example/update.
104    #[argh(positional)]
105    pub update_pkg_url: String,
106
107    /// the force install was initiated by a service, in the background.
108    #[argh(switch)]
109    pub service_initiated: bool,
110}
111
112#[derive(Debug, Eq, FromArgs, PartialEq)]
113#[argh(subcommand, name = "wait-for-commit")]
114/// Wait for the update to be committed.
115pub struct WaitForCommit {}
116
117#[derive(Debug, Eq, FromArgs, PartialEq)]
118#[argh(subcommand, name = "revert")]
119/// Revert the update.
120pub struct Revert {}
121
122#[cfg(test)]
123mod tests {
124    use super::*;
125    use assert_matches::assert_matches;
126
127    #[test]
128    fn test_unknown_option() {
129        assert_matches!(Update::from_args(&["update"], &["--unknown"]), Err(_));
130    }
131
132    #[test]
133    fn test_unknown_subcommand() {
134        assert_matches!(Update::from_args(&["update"], &["unknown"]), Err(_));
135    }
136
137    #[test]
138    fn test_channel_get() {
139        let update = Update::from_args(&["update"], &["channel", "get"]).unwrap();
140        assert_eq!(
141            update,
142            Update {
143                cmd: Command::Channel(Channel { cmd: channel::Command::Get(channel::Get {}) })
144            }
145        );
146    }
147
148    #[test]
149    fn test_channel_target() {
150        let update = Update::from_args(&["update"], &["channel", "target"]).unwrap();
151        assert_eq!(
152            update,
153            Update {
154                cmd: Command::Channel(Channel {
155                    cmd: channel::Command::Target(channel::Target {})
156                })
157            }
158        );
159    }
160
161    #[test]
162    fn test_channel_set() {
163        let update = Update::from_args(&["update"], &["channel", "set", "new-channel"]).unwrap();
164        assert_eq!(
165            update,
166            Update {
167                cmd: Command::Channel(Channel {
168                    cmd: channel::Command::Set(channel::Set { channel: "new-channel".to_string() })
169                })
170            }
171        );
172    }
173
174    #[test]
175    fn test_channel_list() {
176        let update = Update::from_args(&["update"], &["channel", "list"]).unwrap();
177        assert_eq!(
178            update,
179            Update {
180                cmd: Command::Channel(Channel { cmd: channel::Command::List(channel::List {}) })
181            }
182        );
183    }
184
185    #[test]
186    fn test_check_now() {
187        let update = Update::from_args(&["update"], &["check-now"]).unwrap();
188        assert_eq!(
189            update,
190            Update {
191                cmd: Command::CheckNow(CheckNow { service_initiated: false, monitor: false })
192            }
193        );
194    }
195
196    #[test]
197    fn test_check_now_monitor() {
198        let update = Update::from_args(&["update"], &["check-now", "--monitor"]).unwrap();
199        assert_eq!(
200            update,
201            Update { cmd: Command::CheckNow(CheckNow { service_initiated: false, monitor: true }) }
202        );
203    }
204
205    #[test]
206    fn test_check_now_service_initiated() {
207        let update = Update::from_args(&["update"], &["check-now", "--service-initiated"]).unwrap();
208        assert_eq!(
209            update,
210            Update { cmd: Command::CheckNow(CheckNow { service_initiated: true, monitor: false }) }
211        );
212    }
213
214    #[test]
215    fn test_force_install_requires_positional_arg() {
216        assert_matches!(Update::from_args(&["update"], &["force-install"]), Err(_));
217    }
218
219    #[test]
220    fn test_force_install() {
221        let update = Update::from_args(&["update"], &["force-install", "url"]).unwrap();
222        assert_eq!(
223            update,
224            Update {
225                cmd: Command::ForceInstall(ForceInstall {
226                    update_pkg_url: "url".to_owned(),
227                    reboot: true,
228                    service_initiated: false,
229                })
230            }
231        );
232    }
233
234    #[test]
235    fn test_force_install_no_reboot() {
236        let update =
237            Update::from_args(&["update"], &["force-install", "--reboot", "false", "url"]).unwrap();
238        assert_eq!(
239            update,
240            Update {
241                cmd: Command::ForceInstall(ForceInstall {
242                    update_pkg_url: "url".to_owned(),
243                    reboot: false,
244                    service_initiated: false,
245                })
246            }
247        );
248    }
249
250    #[test]
251    fn test_force_install_service_initiated() {
252        let update =
253            Update::from_args(&["update"], &["force-install", "--service-initiated", "url"])
254                .unwrap();
255        assert_eq!(
256            update,
257            Update {
258                cmd: Command::ForceInstall(ForceInstall {
259                    update_pkg_url: "url".to_owned(),
260                    reboot: true,
261                    service_initiated: true,
262                })
263            }
264        );
265    }
266
267    #[test]
268    fn test_wait_for_commit() {
269        let update = Update::from_args(&["update"], &["wait-for-commit"]).unwrap();
270        assert_eq!(update, Update { cmd: Command::WaitForCommit(WaitForCommit {}) });
271    }
272
273    #[test]
274    fn test_revert() {
275        let update = Update::from_args(&["update"], &["revert"]).unwrap();
276        assert_eq!(update, Update { cmd: Command::Revert(Revert {}) });
277    }
278
279    #[test]
280    fn test_monitor() {
281        let update = Update::from_args(&["update"], &["monitor-updates"]).unwrap();
282        assert_eq!(update, Update { cmd: Command::MonitorUpdates(MonitorUpdates {}) });
283    }
284}