//
// CKDKVSStore.m
// Security
//
// Created by Mitch Adler on 5/15/16.
//
//
#import "CKDKVSStore.h"
#import "CKDKVSProxy.h"
#include "SOSCloudKeychainConstants.h"
#include <utilities/debugging.h>
#import <Foundation/NSUbiquitousKeyValueStore.h>
#import <Foundation/NSUbiquitousKeyValueStore_Private.h>
#import "SyncedDefaults/SYDConstants.h"
#include <os/activity.h>
#import "Analytics/Clients/SOSAnalytics.h"
struct CKDKVSCounters {
uint64_t synchronize;
uint64_t synchronizeWithCompletionHandler;
uint64_t incomingMessages;
uint64_t outgoingMessages;
uint64_t totalWaittimeSynchronize;
uint64_t longestWaittimeSynchronize;
uint64_t synchronizeFailures;
};
@interface CKDKVSStore ()
@property (readwrite, weak) UbiqitousKVSProxy* proxy;
@property (readwrite) NSUbiquitousKeyValueStore* cloudStore;
@property (assign,readwrite) struct CKDKVSCounters* perfCounters;
@property dispatch_queue_t perfQueue;
@end
@implementation CKDKVSStore
+ (instancetype)kvsInterface {
return [[CKDKVSStore alloc] init];
}
- (instancetype)init {
self = [super init];
self->_cloudStore = [NSUbiquitousKeyValueStore defaultStore];
self->_proxy = nil;
if (!self.cloudStore) {
secerror("NO NSUbiquitousKeyValueStore defaultStore!!!");
return nil;
}
self.perfQueue = dispatch_queue_create("CKDKVSStorePerfQueue", NULL);
self.perfCounters = calloc(1, sizeof(struct CKDKVSCounters));
[self setupSamplers];
return self;
}
- (void)dealloc {
if (_perfCounters)
free(_perfCounters);
}
- (void) connectToProxy: (UbiqitousKVSProxy*) proxy {
_proxy = proxy;
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(kvsStoreChangedAsync:)
name:NSUbiquitousKeyValueStoreDidChangeExternallyNotification
object:nil];
}
- (void)setObject:(id)obj forKey:(NSString*)key {
secdebug("kvsdebug", " [self.cloudStore setObject:obj forKey:key];
}
- (NSDictionary<NSString *, id>*) copyAsDictionary {
return [self.cloudStore dictionaryRepresentation];
}
- (void)addEntriesFromDictionary:(NSDictionary<NSString*, NSObject*> *)otherDictionary {
[otherDictionary enumerateKeysAndObjectsUsingBlock:^(NSString * _Nonnull key, NSObject * _Nonnull obj, BOOL * _Nonnull stop) {
[self setObject:obj forKey:key];
}];
}
- (id)objectForKey:(NSString*)key {
return [self.cloudStore objectForKey:key];
}
- (void)removeObjectForKey:(NSString*)key {
return [self.cloudStore removeObjectForKey:key];
}
- (void)removeAllObjects {
[[[[self.cloudStore dictionaryRepresentation] allKeys] copy] enumerateObjectsUsingBlock:^(NSString * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
[self.cloudStore removeObjectForKey:obj];
}];
}
- (void)pushWrites {
[[self cloudStore] synchronize];
dispatch_async(self.perfQueue, ^{
self.perfCounters->synchronize++;
});
}
// Runs on the same thread that posted the notification, and that thread _may_ be the
// kdkvsproxy_queue (see 30470419). Avoid deadlock by bouncing through global queue.
- (void)kvsStoreChangedAsync:(NSNotification *)notification
{
secnotice("CloudKeychainProxy", " dispatch_async(self.perfQueue, ^{
self.perfCounters->incomingMessages++;
});
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
[self kvsStoreChanged:notification];
});
}
- (void) kvsStoreChanged:(NSNotification *)notification {
/*
Posted when the value of one or more keys in the local key-value store
changed due to incoming data pushed from iCloud. This notification is
sent only upon a change received from iCloud; it is not sent when your
app sets a value.
The user info dictionary can contain the reason for the notification as
well as a list of which values changed, as follows:
The value of the NSUbiquitousKeyValueStoreChangeReasonKey key, when
present, indicates why the key-value store changed. Its value is one of
the constants in "Change Reason Values."
The value of the NSUbiquitousKeyValueStoreChangedKeysKey, when present,
is an array of strings, each the name of a key whose value changed. The
notification object is the NSUbiquitousKeyValueStore object whose contents
changed.
NSUbiquitousKeyValueStoreInitialSyncChange is only posted if there is any
local value that has been overwritten by a distant value. If there is no
conflict between the local and the distant values when doing the initial
sync (e.g. if the cloud has no data stored or the client has not stored
any data yet), you'll never see that notification.
NSUbiquitousKeyValueStoreInitialSyncChange implies an initial round trip
with server but initial round trip with server does not imply
NSUbiquitousKeyValueStoreInitialSyncChange.
*/
os_activity_initiate("cloudChanged", OS_ACTIVITY_FLAG_DEFAULT, ^{
secdebug(XPROXYSCOPE, "
UbiqitousKVSProxy* proxy = self.proxy;
if(!proxy) {
// Weak reference went away.
return;
}
NSDictionary *userInfo = [notification userInfo];
NSNumber *reason = [userInfo objectForKey:NSUbiquitousKeyValueStoreChangeReasonKey];
NSArray *keysChangedArray = [userInfo objectForKey:NSUbiquitousKeyValueStoreChangedKeysKey];
NSSet *keysChanged = keysChangedArray ? [NSSet setWithArray: keysChangedArray] : nil;
if (reason) switch ([reason integerValue]) {
case NSUbiquitousKeyValueStoreInitialSyncChange:
[proxy storeKeysChanged: keysChanged initial: YES];
break;
case NSUbiquitousKeyValueStoreServerChange:
[proxy storeKeysChanged: keysChanged initial: NO];
break;
case NSUbiquitousKeyValueStoreQuotaViolationChange:
seccritical("Received NSUbiquitousKeyValueStoreQuotaViolationChange");
break;
case NSUbiquitousKeyValueStoreAccountChange:
[proxy storeAccountChanged];
break;
default:
secinfo("kvsstore", "ignoring unknown notification: break;
}
});
}
- (BOOL) pullUpdates:(NSError **)failure
{
__block NSError *tempFailure = nil;
dispatch_semaphore_t freshSemaphore = dispatch_semaphore_create(0);
secnoticeq("fresh", "
dispatch_async(self.perfQueue, ^{
self.perfCounters->synchronizeWithCompletionHandler++;
});
[[self cloudStore] synchronizeWithCompletionHandler:^(NSError *error) {
if (error) {
dispatch_async(self.perfQueue, ^{
self.perfCounters->synchronizeFailures++;
});
tempFailure = error;
secnotice("fresh", " } else {
dispatch_async(self.perfQueue, ^{
self.perfCounters->synchronize++;
});
secnotice("fresh", " [[self cloudStore] synchronize]; // Per olivier in <rdar://problem/13412631>, sync before getting values
secnotice("fresh", " }
dispatch_semaphore_signal(freshSemaphore);
}];
dispatch_semaphore_wait(freshSemaphore, DISPATCH_TIME_FOREVER);
if (failure && (*failure == NULL)) {
*failure = tempFailure;
}
return tempFailure == nil;
}
- (void)perfCounters:(void(^)(NSDictionary *counters))callback
{
dispatch_async(self.perfQueue, ^{
callback(@{
CKDKVSPerfCounterSynchronize : @(self.perfCounters->synchronize),
CKDKVSPerfCounterSynchronizeWithCompletionHandler : @(self.perfCounters->synchronizeWithCompletionHandler),
CKDKVSPerfCounterIncomingMessages : @(self.perfCounters->incomingMessages),
CKDKVSPerfCounterOutgoingMessages : @(self.perfCounters->outgoingMessages),
CKDKVSPerfCounterTotalWaitTimeSynchronize : @(self.perfCounters->totalWaittimeSynchronize),
CKDKVSPerfCounterLongestWaitTimeSynchronize : @(self.perfCounters->longestWaittimeSynchronize),
CKDKVSPerfCounterSynchronizeFailures : @(self.perfCounters->synchronizeFailures),
});
});
}
#if __OBJC2__
- (void)setupSamplers
{
[[SOSAnalytics logger] AddMultiSamplerForName:CKDKVSPerformanceCountersSampler
withTimeInterval:SFAnalyticsSamplerIntervalOncePerReport
block:^NSDictionary<NSString *,NSNumber *> *{
__block NSDictionary<NSString *,NSNumber *>* data;
[self perfCounters:^(NSDictionary *counters) {
data = counters;
}];
dispatch_sync(self.perfQueue, ^{
memset(self.perfCounters, 0, sizeof(struct CKDKVSCounters));
});
return data;
}];
}
#else
- (void)setupSamplers
{
// SFA is only for 64 bit cool kids
}
#endif
- (void)addOneToOutGoing
{
dispatch_async(self.perfQueue, ^{
self.perfCounters->outgoingMessages++;
});
}
@end