remote_settings/context.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
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
use serde_json::{Map, Value};
use std::collections::HashMap;
/// Remote settings context object
///
/// This is used to filter the records returned. We always fetch all `records` from the
/// remote-settings storage. Some records could have a `filter_expression`. If this is passed in
/// and the record has a `filter_expression`, then only returns where the expression is true will
/// be returned.
///
/// See https://remote-settings.readthedocs.io/en/latest/target-filters.html for details.
#[derive(Debug, Clone, Default, uniffi::Record)]
pub struct RemoteSettingsContext {
/// The delivery channel of the application (e.g "nightly")
#[uniffi(default = None)]
pub channel: Option<String>,
/// User visible version string (e.g. "1.0.3")
#[uniffi(default = None)]
pub app_version: Option<String>,
/// String containing the XUL application app_id
#[uniffi(default = None)]
pub app_id: Option<String>,
/// The locale of the application during initialization (e.g. "es-ES")
#[uniffi(default = None)]
pub locale: Option<String>,
/// The name of the operating system (e.g. "Android", "iOS", "Darwin", "WINNT")
#[uniffi(default = None)]
pub os: Option<String>,
/// The user-visible version of the operating system (e.g. "1.2.3")
#[uniffi(default = None)]
pub os_version: Option<String>,
/// Form-factor of the device ("phone", "tablet", or "desktop")
#[uniffi(default = None)]
pub form_factor: Option<String>,
/// Country of the user.
///
/// This is usually populated in one of two ways:
/// - The second component of the locale
/// - By using a geolocation service, which determines country via the user's IP.
/// Firefox apps usually have a module that integrates with these services,
/// for example `Region` on Desktop and `RegionMiddleware` on Android.
#[uniffi(default = None)]
pub country: Option<String>,
/// Extra attributes to add to the env for JEXL filtering.
///
/// Use this for prototyping / testing new features. In the long-term, new fields should be
/// added to the official list and supported by both the Rust and Gecko clients.
#[uniffi(default = None)]
pub custom_targetting_attributes: Option<HashMap<String, String>>,
}
impl RemoteSettingsContext {
/// Convert this into the `env` value for the remote settings JEXL filter
///
/// https://remote-settings.readthedocs.io/en/latest/target-filters.html
pub(crate) fn into_env(self) -> Value {
let mut v = Map::new();
v.insert("channel".to_string(), self.channel.into());
if let Some(version) = self.app_version {
v.insert("version".to_string(), version.into());
}
if let Some(locale) = self.locale {
v.insert("locale".to_string(), locale.into());
}
if self.app_id.is_some() || self.os.is_some() || self.os_version.is_some() {
let mut appinfo = Map::default();
if let Some(app_id) = self.app_id {
appinfo.insert("ID".to_string(), app_id.into());
}
// The "os" object is the new way to represent OS-related data
if self.os.is_some() || self.os_version.is_some() {
let mut os = Map::default();
if let Some(os_name) = &self.os {
os.insert("name".to_string(), os_name.to_string().into());
}
if let Some(os_version) = self.os_version {
os.insert("version".to_string(), os_version.into());
}
appinfo.insert("os".to_string(), os.into());
}
// The "OS" string is for backwards compatibility
if let Some(os_name) = self.os {
appinfo.insert("OS".to_string(), os_name.into());
}
v.insert("appinfo".to_string(), appinfo.into());
}
if let Some(form_factor) = self.form_factor {
v.insert("formFactor".to_string(), form_factor.into());
}
if let Some(country) = self.country {
v.insert("country".to_string(), country.into());
}
if let Some(custom) = self.custom_targetting_attributes {
v.extend(custom.into_iter().map(|(k, v)| (k, v.into())));
}
v.into()
}
}
#[cfg(test)]
mod test {
use super::*;
use serde_json::json;
/// Test that the remote settings context is normalized to match
/// https://remote-settings.readthedocs.io/en/latest/target-filters.html, regardless of what
/// the fields are named in Rust.
#[test]
fn test_context_normalization() {
let context = RemoteSettingsContext {
channel: Some("beta".into()),
app_id: Some("{aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee}".into()),
app_version: Some("1.0.0".into()),
os: Some("MS-DOS".into()),
os_version: Some("6.1".into()),
locale: Some("en-US".into()),
form_factor: Some("tablet".into()),
country: Some("US".into()),
custom_targetting_attributes: Some(HashMap::from([("extra".into(), "test".into())])),
};
assert_eq!(
context.into_env(),
json!({
// Official fields
"version": "1.0.0",
"channel": "beta",
"locale": "en-US",
"appinfo": {
"ID": "{aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee}",
"OS": "MS-DOS",
"os": {
"name": "MS-DOS",
"version": "6.1"
}
},
// Unofficial fields that we need for Suggest geo-expansion. These should be made
// into official fields that both the Gecko and Rust client support.
"formFactor": "tablet",
"country": "US",
"extra": "test",
})
);
}
}