use crate::error::*;
use crate::types::{ServiceStatus, SyncEngineSelection, SyncParams, SyncReason, SyncResult};
use crate::{reset, reset_all, wipe};
use error_support::breadcrumb;
use parking_lot::Mutex;
use std::collections::{BTreeMap, HashMap, HashSet};
use std::convert::TryFrom;
use std::time::SystemTime;
use sync15::client::{
sync_multiple_with_command_processor, MemoryCachedState, Sync15StorageClientInit,
SyncRequestInfo,
};
use sync15::clients_engine::{Command, CommandProcessor, CommandStatus, Settings};
use sync15::engine::{EngineSyncAssociation, SyncEngine, SyncEngineId};
#[derive(Default)]
pub struct SyncManager {
mem_cached_state: Mutex<Option<MemoryCachedState>>,
}
impl SyncManager {
pub fn new() -> Self {
Self::default()
}
fn get_engine_id(engine_name: &str) -> Result<SyncEngineId> {
SyncEngineId::try_from(engine_name).map_err(SyncManagerError::UnknownEngine)
}
fn get_engine(engine_id: &SyncEngineId) -> Option<Box<dyn SyncEngine>> {
match engine_id {
SyncEngineId::History => places::get_registered_sync_engine(engine_id),
SyncEngineId::Bookmarks => places::get_registered_sync_engine(engine_id),
SyncEngineId::Addresses => autofill::get_registered_sync_engine(engine_id),
SyncEngineId::CreditCards => autofill::get_registered_sync_engine(engine_id),
SyncEngineId::Passwords => logins::get_registered_sync_engine(engine_id),
SyncEngineId::Tabs => tabs::get_registered_sync_engine(engine_id),
}
}
pub fn wipe(&self, engine_name: &str) -> Result<()> {
if let Some(engine) = Self::get_engine(&Self::get_engine_id(engine_name)?) {
engine.wipe()?;
}
Ok(())
}
pub fn reset(&self, engine_name: &str) -> Result<()> {
if let Some(engine) = Self::get_engine(&Self::get_engine_id(engine_name)?) {
engine.reset(&EngineSyncAssociation::Disconnected)?;
}
Ok(())
}
pub fn reset_all(&self) -> Result<()> {
for (_, engine) in self.iter_registered_engines() {
engine.reset(&EngineSyncAssociation::Disconnected)?;
}
Ok(())
}
pub fn disconnect(&self) {
breadcrumb!("SyncManager disconnect()");
for engine_id in SyncEngineId::iter() {
if let Some(engine) = Self::get_engine(&engine_id) {
if let Err(e) = engine.reset(&EngineSyncAssociation::Disconnected) {
error_support::report_error!(
"sync-manager-reset",
"Failed to reset {}: {}",
engine_id,
e
);
}
} else {
log::warn!("Unable to reset {}, be sure to call register_with_sync_manager before disconnect if this is surprising", engine_id);
}
}
}
pub fn sync(&self, params: SyncParams) -> Result<SyncResult> {
breadcrumb!("SyncManager::sync started");
let mut state = self.mem_cached_state.lock();
let engines = self.calc_engines_to_sync(¶ms.engines)?;
let next_sync_after = state.as_ref().and_then(|mcs| mcs.get_next_sync_after());
let result = if !backoff_in_effect(next_sync_after, ¶ms) {
log::info!("No backoff in effect (or we decided to ignore it), starting sync");
self.do_sync(params, &mut state, engines)
} else {
breadcrumb!(
"Backoff still in effect (until {:?}), bailing out early",
next_sync_after
);
Ok(SyncResult {
status: ServiceStatus::BackedOff,
successful: Default::default(),
failures: Default::default(),
declined: None,
next_sync_allowed_at: next_sync_after,
persisted_state: params.persisted_state.unwrap_or_default(),
telemetry_json: None,
})
};
breadcrumb!("SyncManager sync ended");
result
}
fn do_sync(
&self,
mut params: SyncParams,
state: &mut Option<MemoryCachedState>,
mut engines: Vec<Box<dyn SyncEngine>>,
) -> Result<SyncResult> {
let key_bundle = sync15::KeyBundle::from_ksync_base64(¶ms.auth_info.sync_key)?;
let tokenserver_url = url::Url::parse(¶ms.auth_info.tokenserver_url)?;
let interruptee = interrupt_support::ShutdownInterruptee;
let mut mem_cached_state = state.take().unwrap_or_default();
let mut disk_cached_state = params.persisted_state.take();
for engine in engines.iter_mut() {
if let Some(key) = params.local_encryption_keys.get(&*engine.collection_name()) {
engine.set_local_encryption_key(key)?
}
}
let engine_refs: Vec<&dyn SyncEngine> = engines.iter().map(|s| &**s).collect();
let client_init = Sync15StorageClientInit {
key_id: params.auth_info.kid.clone(),
access_token: params.auth_info.fxa_access_token.clone(),
tokenserver_url,
};
let engines_to_change = if params.enabled_changes.is_empty() {
None
} else {
Some(¶ms.enabled_changes)
};
let settings = Settings {
fxa_device_id: params.device_settings.fxa_device_id,
device_name: params.device_settings.name,
device_type: params.device_settings.kind,
};
let c = SyncClient::new(settings);
let result = sync_multiple_with_command_processor(
Some(&c),
&engine_refs,
&mut disk_cached_state,
&mut mem_cached_state,
&client_init,
&key_bundle,
&interruptee,
Some(SyncRequestInfo {
engines_to_state_change: engines_to_change,
is_user_action: matches!(params.reason, SyncReason::User),
}),
);
*state = Some(mem_cached_state);
log::info!("Sync finished with status {:?}", result.service_status);
let status = ServiceStatus::from(result.service_status);
for (engine, result) in result.engine_results.iter() {
log::info!("engine {:?} status: {:?}", engine, result);
}
let mut successful: Vec<String> = Vec::new();
let mut failures: HashMap<String, String> = HashMap::new();
for (engine, result) in result.engine_results.into_iter() {
match result {
Ok(_) => {
successful.push(engine);
}
Err(err) => {
failures.insert(engine, err.to_string());
}
}
}
let telemetry_json = serde_json::to_string(&result.telemetry).unwrap();
Ok(SyncResult {
status,
successful,
failures,
declined: result.declined,
next_sync_allowed_at: result.next_sync_after,
persisted_state: disk_cached_state.unwrap_or_default(),
telemetry_json: Some(telemetry_json),
})
}
fn iter_registered_engines(&self) -> impl Iterator<Item = (SyncEngineId, Box<dyn SyncEngine>)> {
SyncEngineId::iter().filter_map(|id| Self::get_engine(&id).map(|engine| (id, engine)))
}
pub fn get_available_engines(&self) -> Vec<String> {
self.iter_registered_engines()
.map(|(name, _)| name.to_string())
.collect()
}
fn calc_engines_to_sync(
&self,
selection: &SyncEngineSelection,
) -> Result<Vec<Box<dyn SyncEngine>>> {
let mut engine_map: BTreeMap<_, _> = self.iter_registered_engines().collect();
breadcrumb!(
"Checking engines requested ({:?}) vs local engines ({:?})",
selection,
engine_map
.keys()
.map(|engine_id| engine_id.name())
.collect::<Vec<_>>(),
);
if let SyncEngineSelection::Some {
engines: engine_names,
} = selection
{
let mut selected_engine_ids: HashSet<SyncEngineId> = HashSet::new();
for name in engine_names {
let engine_id = Self::get_engine_id(name)?;
if !engine_map.contains_key(&engine_id) {
return Err(SyncManagerError::UnsupportedFeature(name.to_string()));
}
selected_engine_ids.insert(engine_id);
}
engine_map.retain(|engine_id, _| selected_engine_ids.contains(engine_id))
}
Ok(engine_map.into_values().collect())
}
}
fn backoff_in_effect(next_sync_after: Option<SystemTime>, p: &SyncParams) -> bool {
let now = SystemTime::now();
if let Some(nsa) = next_sync_after {
if nsa > now {
return if matches!(p.reason, SyncReason::User | SyncReason::EnabledChange) {
log::info!(
"Still under backoff, but syncing anyway because reason is {:?}",
p.reason
);
false
} else if !p.enabled_changes.is_empty() {
log::info!(
"Still under backoff, but syncing because we have enabled state changes."
);
false
} else {
log::info!(
"Still under backoff, and there's no compelling reason for us to ignore it"
);
true
};
}
}
log::debug!("Not under backoff");
false
}
impl From<sync15::client::ServiceStatus> for ServiceStatus {
fn from(s15s: sync15::client::ServiceStatus) -> Self {
use sync15::client::ServiceStatus::*;
match s15s {
Ok => ServiceStatus::Ok,
NetworkError => ServiceStatus::NetworkError,
ServiceError => ServiceStatus::ServiceError,
AuthenticationError => ServiceStatus::AuthError,
BackedOff => ServiceStatus::BackedOff,
Interrupted => ServiceStatus::OtherError, OtherError => ServiceStatus::OtherError,
}
}
}
struct SyncClient(Settings);
impl SyncClient {
pub fn new(settings: Settings) -> SyncClient {
SyncClient(settings)
}
}
impl CommandProcessor for SyncClient {
fn settings(&self) -> &Settings {
&self.0
}
fn apply_incoming_command(&self, command: Command) -> anyhow::Result<CommandStatus> {
let result = match command {
Command::Wipe(engine) => wipe(&engine),
Command::Reset(engine) => reset(&engine),
Command::ResetAll => reset_all(),
};
match result {
Ok(()) => Ok(CommandStatus::Applied),
Err(err) => match err {
SyncManagerError::UnknownEngine(_) => Ok(CommandStatus::Unsupported),
_ => Err(err.into()),
},
}
}
fn fetch_outgoing_commands(&self) -> anyhow::Result<HashSet<Command>> {
Ok(HashSet::new())
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn test_engine_id_sanity() {
for engine_id in SyncEngineId::iter() {
assert_eq!(engine_id, SyncEngineId::try_from(engine_id.name()).unwrap());
}
}
}