nimbus_cli/output/
deeplink.rsuse anyhow::{anyhow, Result};
use percent_encoding::{AsciiSet, CONTROLS};
use crate::protocol::StartAppProtocol;
use crate::{AppOpenArgs, LaunchableApp};
impl LaunchableApp {
pub(crate) fn copy_to_clipboard(
&self,
app_protocol: &StartAppProtocol,
open: &AppOpenArgs,
) -> Result<usize> {
let url = self.longform_url(app_protocol, open)?;
let len = url.len();
if let Err(e) = set_clipboard(url) {
anyhow::bail!("Can't copy URL to clipboard: {}", e)
};
Ok(len)
}
pub(crate) fn longform_url(
&self,
app_protocol: &StartAppProtocol,
open: &AppOpenArgs,
) -> Result<String> {
let deeplink = match (&open.deeplink, self.app_opening_deeplink()) {
(Some(deeplink), _) => deeplink.to_owned(),
(_, Some(deeplink)) => join_query(deeplink, "--nimbus-cli&--is-launcher"),
_ => anyhow::bail!("A deeplink must be provided"),
};
let url = longform_deeplink_url(deeplink.as_str(), app_protocol)?;
self.prepend_scheme(url.as_str())
}
fn app_opening_deeplink(&self) -> Option<&str> {
match self {
Self::Android { open_deeplink, .. } => open_deeplink.as_deref(),
Self::Ios { .. } => Some("noop"),
}
}
pub(crate) fn deeplink(&self, open: &AppOpenArgs) -> Result<Option<String>> {
let deeplink = &open.deeplink;
if deeplink.is_none() {
return Ok(None);
}
let deeplink = self.prepend_scheme(deeplink.as_ref().unwrap())?;
Ok(Some(deeplink))
}
fn prepend_scheme(&self, deeplink: &str) -> Result<String> {
Ok(if deeplink.contains("://") {
deeplink.to_string()
} else {
let scheme = self.mandatory_scheme()?;
format!("{scheme}://{deeplink}")
})
}
fn mandatory_scheme(&self) -> Result<&str> {
match self {
Self::Android { scheme, .. } | Self::Ios { scheme, .. } => scheme
.as_deref()
.ok_or_else(|| anyhow!("A scheme is not defined for this app")),
}
}
}
const QUERY: &AsciiSet = &CONTROLS
.add(b' ')
.add(b'"')
.add(b'<')
.add(b'>')
.add(b'#')
.add(b'\'')
.add(b'{')
.add(b'}')
.add(b':')
.add(b'/')
.add(b'?')
.add(b'&');
pub(crate) fn longform_deeplink_url(
deeplink: &str,
app_protocol: &StartAppProtocol,
) -> Result<String> {
let StartAppProtocol {
reset_db,
experiments,
log_state,
} = app_protocol;
if !reset_db && experiments.is_none() && !log_state {
return Ok(deeplink.to_string());
}
let mut parts: Vec<_> = Default::default();
if !deeplink.contains("--nimbus-cli") {
parts.push("--nimbus-cli".to_string());
}
if let Some(v) = experiments {
let json = serde_json::to_string(v)?;
let string = percent_encoding::utf8_percent_encode(&json, QUERY).to_string();
parts.push(format!("--experiments={string}"));
}
if *reset_db {
parts.push("--reset-db".to_string());
}
if *log_state {
parts.push("--log-state".to_string());
}
Ok(join_query(deeplink, &parts.join("&")))
}
fn join_query(url: &str, item: &str) -> String {
let suffix = if url.contains('?') { '&' } else { '?' };
format!("{url}{suffix}{item}")
}
fn set_clipboard(contents: String) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
use copypasta::{ClipboardContext, ClipboardProvider};
let mut ctx = ClipboardContext::new()?;
ctx.set_contents(contents)?;
Ok(())
}
#[cfg(test)]
mod unit_tests {
use super::*;
use serde_json::json;
#[test]
fn test_url_noop() -> Result<()> {
let p = StartAppProtocol {
reset_db: false,
experiments: None,
log_state: false,
};
assert_eq!("host".to_string(), longform_deeplink_url("host", &p)?);
assert_eq!(
"host?query=1".to_string(),
longform_deeplink_url("host?query=1", &p)?
);
Ok(())
}
#[test]
fn test_url_reset_db() -> Result<()> {
let p = StartAppProtocol {
reset_db: true,
experiments: None,
log_state: false,
};
assert_eq!(
"host?--nimbus-cli&--reset-db".to_string(),
longform_deeplink_url("host", &p)?
);
assert_eq!(
"host?query=1&--nimbus-cli&--reset-db".to_string(),
longform_deeplink_url("host?query=1", &p)?
);
Ok(())
}
#[test]
fn test_url_log_state() -> Result<()> {
let p = StartAppProtocol {
reset_db: false,
experiments: None,
log_state: true,
};
assert_eq!(
"host?--nimbus-cli&--log-state".to_string(),
longform_deeplink_url("host", &p)?
);
assert_eq!(
"host?query=1&--nimbus-cli&--log-state".to_string(),
longform_deeplink_url("host?query=1", &p)?
);
Ok(())
}
#[test]
fn test_url_experiments() -> Result<()> {
let v = json!({"data": []});
let p = StartAppProtocol {
reset_db: false,
experiments: Some(&v),
log_state: false,
};
assert_eq!(
"host?--nimbus-cli&--experiments=%7B%22data%22%3A[]%7D".to_string(),
longform_deeplink_url("host", &p)?
);
assert_eq!(
"host?query=1&--nimbus-cli&--experiments=%7B%22data%22%3A[]%7D".to_string(),
longform_deeplink_url("host?query=1", &p)?
);
Ok(())
}
#[test]
fn test_deeplink_has_is_launcher_param_if_no_deeplink_is_specified() -> Result<()> {
let app =
LaunchableApp::try_from_app_channel_device(Some("fenix"), Some("developer"), None)?;
let payload: StartAppProtocol = Default::default();
let open: AppOpenArgs = Default::default();
assert_eq!(
"fenix-dev://open?--nimbus-cli&--is-launcher".to_string(),
app.longform_url(&payload, &open)?
);
let open = AppOpenArgs {
deeplink: Some("deeplink".to_string()),
..Default::default()
};
assert_eq!(
"fenix-dev://deeplink".to_string(),
app.longform_url(&payload, &open)?
);
let payload = StartAppProtocol {
log_state: true,
..Default::default()
};
assert_eq!(
"fenix-dev://open?--nimbus-cli&--is-launcher&--log-state".to_string(),
app.longform_url(&payload, &Default::default())?
);
let open = AppOpenArgs {
deeplink: Some("deeplink".to_string()),
..Default::default()
};
assert_eq!(
"fenix-dev://deeplink?--nimbus-cli&--log-state".to_string(),
app.longform_url(&payload, &open)?
);
Ok(())
}
}