Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[MC-2359] Add normalized event and property names evaluation #385

Merged
merged 8 commits into from
Nov 28, 2024
4 changes: 4 additions & 0 deletions CleverTapSDK.xcodeproj/project.pbxproj
Original file line number Diff line number Diff line change
Expand Up @@ -350,6 +350,7 @@
6B32A0AD2B9DBE31009ADC57 /* CTTemplatePresenterMock.m in Sources */ = {isa = PBXBuildFile; fileRef = 6B32A0AC2B9DBE31009ADC57 /* CTTemplatePresenterMock.m */; };
6B32A0B02B9DC374009ADC57 /* CTTemplateArgumentTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 6B32A0AF2B9DC374009ADC57 /* CTTemplateArgumentTest.m */; };
6B32A0B42B9F2E8F009ADC57 /* CTTestTemplateProducer.m in Sources */ = {isa = PBXBuildFile; fileRef = 6B32A0B32B9F2E8F009ADC57 /* CTTestTemplateProducer.m */; };
6B453EFE2CF74BE2003C7A89 /* CTEventAdapterTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 6B453EFD2CF74BE2003C7A89 /* CTEventAdapterTest.m */; };
6B453EF92CF621E3003C7A89 /* CTInAppDisplayViewControllerMock.m in Sources */ = {isa = PBXBuildFile; fileRef = 6B453EF82CF621E3003C7A89 /* CTInAppDisplayViewControllerMock.m */; };
6B4A0F912B45EF6D00A42C6D /* CTInAppTriggerManagerTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 6B4A0F902B45EF6D00A42C6D /* CTInAppTriggerManagerTest.m */; };
6B535FB62AD56C60002A2663 /* CTMultiDelegateManager.h in Headers */ = {isa = PBXBuildFile; fileRef = 6B535FB42AD56C60002A2663 /* CTMultiDelegateManager.h */; };
Expand Down Expand Up @@ -914,6 +915,7 @@
6B32A0B12B9F2A75009ADC57 /* CTCustomTemplatesManager+Tests.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "CTCustomTemplatesManager+Tests.h"; sourceTree = "<group>"; };
6B32A0B22B9F2E8F009ADC57 /* CTTestTemplateProducer.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = CTTestTemplateProducer.h; sourceTree = "<group>"; };
6B32A0B32B9F2E8F009ADC57 /* CTTestTemplateProducer.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = CTTestTemplateProducer.m; sourceTree = "<group>"; };
6B453EFD2CF74BE2003C7A89 /* CTEventAdapterTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = CTEventAdapterTest.m; sourceTree = "<group>"; };
6B453EF72CF621E3003C7A89 /* CTInAppDisplayViewControllerMock.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = CTInAppDisplayViewControllerMock.h; sourceTree = "<group>"; };
6B453EF82CF621E3003C7A89 /* CTInAppDisplayViewControllerMock.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = CTInAppDisplayViewControllerMock.m; sourceTree = "<group>"; };
6B4A0F902B45EF6D00A42C6D /* CTInAppTriggerManagerTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = CTInAppTriggerManagerTest.m; sourceTree = "<group>"; };
Expand Down Expand Up @@ -1476,6 +1478,7 @@
6BA3B2E72B07E207004E834B /* CTTriggersMatcher+Tests.m */,
6B4A0F902B45EF6D00A42C6D /* CTInAppTriggerManagerTest.m */,
6BB778CF2BEE4C3400A41628 /* CTNotificationActionTest.m */,
6B453EFD2CF74BE2003C7A89 /* CTEventAdapterTest.m */,
4806346E2CEB620400E39E9B /* CTInAppDisplayViewControllerTests.m */,
6B453EF72CF621E3003C7A89 /* CTInAppDisplayViewControllerMock.h */,
6B453EF82CF621E3003C7A89 /* CTInAppDisplayViewControllerMock.m */,
Expand Down Expand Up @@ -2546,6 +2549,7 @@
D02AC2DB276044F70031C1BE /* CleverTapSDKTests.m in Sources */,
32394C2129FA264B00956058 /* CTPreferencesTest.m in Sources */,
6BD334F02AF545C80099E33E /* CTInAppStoreTest.m in Sources */,
6B453EFE2CF74BE2003C7A89 /* CTEventAdapterTest.m in Sources */,
32394C1F29FA251E00956058 /* CTEventBuilderTest.m in Sources */,
6B12F7692C9466460045D743 /* CTJsonTemplateProducerTest.m in Sources */,
6BB778D02BEE4C3400A41628 /* CTNotificationActionTest.m in Sources */,
Expand Down
10 changes: 10 additions & 0 deletions CleverTapSDK/CTUtils.h
Original file line number Diff line number Diff line change
Expand Up @@ -15,4 +15,14 @@
+ (NSNumber * _Nullable)numberFromString:(NSString * _Nullable)string;
+ (NSNumber * _Nullable)numberFromString:(NSString * _Nullable)string withLocale:(NSLocale * _Nullable)locale;

/**
* Get the CT normalized version of an event or a property name.
*/
+ (NSString * _Nullable)getNormalizedName:(NSString * _Nullable)name;

/**
* Check if two event/property names are equal with applied CT normalization
*/
+ (BOOL)areEqualNormalizedName:(NSString * _Nullable)firstName andName:(NSString * _Nullable)secondName;

@end
30 changes: 30 additions & 0 deletions CleverTapSDK/CTUtils.m
Original file line number Diff line number Diff line change
Expand Up @@ -142,4 +142,34 @@ + (NSNumber * _Nullable)numberFromString:(NSString * _Nullable)string withLocale
return nil;
}

+ (NSString * _Nullable)getNormalizedName:(NSString * _Nullable)name {
if (name) {
// Lowercase with English locale for consistent behavior with the backend
// and across different device locales.
NSString *normalizedName = [name stringByReplacingOccurrencesOfString:@" " withString:@""];
NSLocale *englishLocale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US"];
normalizedName = [normalizedName lowercaseStringWithLocale:englishLocale];
normalizedName = [normalizedName stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]];
return normalizedName;
}

return nil;
}

+ (BOOL)areEqualNormalizedName:(NSString * _Nullable)firstName
andName:(NSString * _Nullable)secondName {
if (firstName == nil && secondName == nil) {
return YES;
}

if (firstName == nil || secondName == nil) {
return NO;
}

NSString *normalizedFirstName = [CTUtils getNormalizedName:firstName];
NSString *normalizedSecondName = [CTUtils getNormalizedName:secondName];

return [normalizedFirstName isEqualToString:normalizedSecondName];
}

@end
5 changes: 3 additions & 2 deletions CleverTapSDK/CTValidator.m
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
#import "CTValidationResult.h"
#import "CTConstants.h"
#import "CTKnownProfileFields.h"
#import "CTUtils.h"

static const int kMaxKeyChars = 120;
static const int kMaxValueChars = 1024;
Expand Down Expand Up @@ -231,7 +232,7 @@ + (BOOL)isRestrictedEventName:(NSString *)name {
NSArray *restrictedNames = @[@"Notification Sent", @"Notification Viewed", @"Notification Clicked",
@"UTM Visited", @"App Launched", @"Stayed", @"App Uninstalled", @"wzrk_d", @"wzrk_fetch", @"SCCampaignOptOut", CLTAP_GEOFENCE_ENTERED_EVENT_NAME, CLTAP_GEOFENCE_EXITED_EVENT_NAME];
for (NSString *x in restrictedNames)
if ([name.lowercaseString isEqualToString:x.lowercaseString]) {
if ([CTUtils areEqualNormalizedName:name andName:x]) {
// The event name is restricted
CTValidationResult *error = [[CTValidationResult alloc] init];
[error setErrorCode:513];
Expand All @@ -244,7 +245,7 @@ + (BOOL)isRestrictedEventName:(NSString *)name {

+ (BOOL)isDiscaredEventName:(NSString *)name {
for (NSString *x in discardedEvents)
if ([name.lowercaseString isEqualToString:x.lowercaseString]) {
if ([CTUtils areEqualNormalizedName:name andName:x]) {
// The event name is discarded
CTValidationResult *error = [[CTValidationResult alloc] init];
[error setErrorCode:513];
Expand Down
49 changes: 43 additions & 6 deletions CleverTapSDK/InApps/Matchers/CTEventAdapter.m
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@

#import "CTEventAdapter.h"
#import "CTConstants.h"
#import "CTUtils.h"

static NSDictionary<NSString*, NSString*> *systemPropToKey;

Expand Down Expand Up @@ -95,19 +96,41 @@ - (CTTriggerValue *)propertyValueNamed:(NSString *)name {
}

- (id)getActualPropertyValue:(NSString *)propertyName {
id value = self.eventProperties[propertyName];
id value = [self getPropertyValue:propertyName];

if (value == nil) {
if ([propertyName isEqualToString:CLTAP_PROP_CAMPAIGN_ID]) {
value = self.eventProperties[CLTAP_PROP_WZRK_ID];
value = [self getPropertyValue:CLTAP_PROP_WZRK_ID];
} else if ([propertyName isEqualToString:CLTAP_PROP_WZRK_ID]) {
value = self.eventProperties[CLTAP_PROP_CAMPAIGN_ID];
value = [self getPropertyValue:CLTAP_PROP_CAMPAIGN_ID];
} else if ([propertyName isEqualToString:CLTAP_PROP_VARIANT]) {
value = self.eventProperties[CLTAP_PROP_WZRK_PIVOT];
value = [self getPropertyValue:CLTAP_PROP_WZRK_PIVOT];
} else if ([propertyName isEqualToString:CLTAP_PROP_WZRK_PIVOT]) {
value = self.eventProperties[CLTAP_PROP_VARIANT];
value = [self getPropertyValue:CLTAP_PROP_VARIANT];
} else if (systemPropToKey[propertyName]) {
// Map App Fields
value = self.eventProperties[systemPropToKey[propertyName]];
value = [self getPropertyValue:systemPropToKey[propertyName]];
}
}
return value;
}

- (id)getPropertyValue:(NSString *)propertyName {
id value = self.eventProperties[propertyName];

if (value == nil) {
// Normalize the property name
propertyName = [CTUtils getNormalizedName:propertyName];
value = self.eventProperties[propertyName];
}

if (value == nil) {
// Check if event properties name are normalized equal
for (NSString *key in self.eventProperties) {
if ([CTUtils areEqualNormalizedName:key andName:propertyName]) {
value = self.eventProperties[key];
break;
}
}
}
return value;
Expand All @@ -120,6 +143,20 @@ - (CTTriggerValue *)itemValueNamed:(NSString *)name {
NSMutableArray *itemValues = [NSMutableArray new];
for (NSDictionary *item in self.items) {
id value = item[name];
if (value == nil) {
NSString *normalizedName = [CTUtils getNormalizedName:name];
value = item[normalizedName];
}
if (value == nil) {
// Normalize the keys in the dictionary
NSMutableDictionary *normalizedItem = [NSMutableDictionary dictionary];
for (NSString *key in item) {
NSString *normalizedKey = [CTUtils getNormalizedName:key];
normalizedItem[normalizedKey] = item[key];
}
value = normalizedItem[[CTUtils getNormalizedName:name]];
}

if (value) {
[itemValues addObject:value];
}
Expand Down
6 changes: 3 additions & 3 deletions CleverTapSDK/InApps/Matchers/CTTriggersMatcher.m
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@
#import "CTTriggerValue.h"
#import "CTConstants.h"
#import "CTTriggerEvaluator.h"
#import "CTUtils.h"

@implementation CTTriggersMatcher

Expand All @@ -30,9 +31,8 @@ - (BOOL)matchEventWhenTriggers:(NSArray *)whenTriggers event:(CTEventAdapter *)e
}

- (BOOL)match:(CTTriggerAdapter *)trigger event:(CTEventAdapter *)event {

BOOL eventNameMatch = [[event eventName] isEqualToString:[trigger eventName]];
BOOL profileAttrNameMatch = [event profileAttrName] != nil && [[event profileAttrName] isEqualToString:[trigger profileAttrName]];
BOOL eventNameMatch = [CTUtils areEqualNormalizedName:[event eventName] andName:[trigger eventName]];
BOOL profileAttrNameMatch = [event profileAttrName] != nil && [CTUtils areEqualNormalizedName:[event profileAttrName] andName:[trigger profileAttrName]];
if (!eventNameMatch && !profileAttrNameMatch) {
return NO;
}
Expand Down
19 changes: 19 additions & 0 deletions CleverTapSDKTests/CTUtilsTest.m
Original file line number Diff line number Diff line change
Expand Up @@ -176,4 +176,23 @@ - (void)test_numberFromStringWithLocale {
XCTAssertNil([CTUtils numberFromString:@"12.3" withLocale:locale]);
}

- (void)testGetNormalizedName {
XCTAssertNil([CTUtils getNormalizedName:nil]);
XCTAssertEqualObjects(@"", [CTUtils getNormalizedName:@""]);
XCTAssertEqualObjects(@"event1", [CTUtils getNormalizedName:@"Event 1"]);
XCTAssertEqualObjects(@"event1", [CTUtils getNormalizedName:@"EVENT 1"]);
XCTAssertEqualObjects(@"event1", [CTUtils getNormalizedName:@"event1"]);
}

- (void)testAreEqualNormalizedNames {
XCTAssertTrue([CTUtils areEqualNormalizedName:nil andName:nil]);
XCTAssertTrue([CTUtils areEqualNormalizedName:@"" andName:@""]);
XCTAssertTrue([CTUtils areEqualNormalizedName:@"Event 1" andName:@"Event1"]);
XCTAssertTrue([CTUtils areEqualNormalizedName:@"Event 1" andName:@"event1"]);
XCTAssertTrue([CTUtils areEqualNormalizedName:@"Event 1" andName:@"EVENT 1"]);
XCTAssertFalse([CTUtils areEqualNormalizedName:@"" andName:nil]);
XCTAssertFalse([CTUtils areEqualNormalizedName:@"Event 1" andName:nil]);
XCTAssertFalse([CTUtils areEqualNormalizedName:@"Event 1" andName:@"Event 2"]);
}

@end
Loading
Loading