mirror of
https://github.com/google/santa.git
synced 2026-01-14 00:37:56 -05:00
* Support CDHash rules * Ensure hardened runtime for cdhash eval. Update docs. * minor fixups * Clarify docs
310 lines
10 KiB
Objective-C
310 lines
10 KiB
Objective-C
/// Copyright 2015 Google Inc. All rights reserved.
|
|
///
|
|
/// Licensed under the Apache License, Version 2.0 (the "License");
|
|
/// you may not use this file except in compliance with the License.
|
|
/// You may obtain a copy of the License at
|
|
///
|
|
/// http://www.apache.org/licenses/LICENSE-2.0
|
|
///
|
|
/// Unless required by applicable law or agreed to in writing, software
|
|
/// distributed under the License is distributed on an "AS IS" BASIS,
|
|
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
/// See the License for the specific language governing permissions and
|
|
/// limitations under the License.
|
|
|
|
#import "Source/common/SNTRule.h"
|
|
|
|
#include <CommonCrypto/CommonCrypto.h>
|
|
#include <Kernel/kern/cs_blobs.h>
|
|
#include <os/base.h>
|
|
|
|
#import "Source/common/SNTSyncConstants.h"
|
|
|
|
// https://developer.apple.com/help/account/manage-your-team/locate-your-team-id/
|
|
static const NSUInteger kExpectedTeamIDLength = 10;
|
|
|
|
@interface SNTRule ()
|
|
@property(readwrite) NSUInteger timestamp;
|
|
@end
|
|
|
|
@implementation SNTRule
|
|
|
|
- (instancetype)initWithIdentifier:(NSString *)identifier
|
|
state:(SNTRuleState)state
|
|
type:(SNTRuleType)type
|
|
customMsg:(NSString *)customMsg
|
|
timestamp:(NSUInteger)timestamp {
|
|
self = [super init];
|
|
if (self) {
|
|
if (identifier.length == 0) {
|
|
return nil;
|
|
}
|
|
|
|
NSCharacterSet *nonHex =
|
|
[[NSCharacterSet characterSetWithCharactersInString:@"0123456789abcdef"] invertedSet];
|
|
NSCharacterSet *nonUppercaseAlphaNumeric = [[NSCharacterSet
|
|
characterSetWithCharactersInString:@"ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"] invertedSet];
|
|
|
|
switch (type) {
|
|
case SNTRuleTypeBinary: OS_FALLTHROUGH;
|
|
case SNTRuleTypeCertificate: {
|
|
// For binary and certificate rules, force the hash identifier to be lowercase hex.
|
|
identifier = [identifier lowercaseString];
|
|
|
|
identifier = [identifier stringByTrimmingCharactersInSet:nonHex];
|
|
if (identifier.length != (CC_SHA256_DIGEST_LENGTH * 2)) {
|
|
return nil;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case SNTRuleTypeTeamID: {
|
|
// TeamIDs are always [0-9A-Z], so enforce that the identifier is uppercase
|
|
identifier =
|
|
[[identifier uppercaseString] stringByTrimmingCharactersInSet:nonUppercaseAlphaNumeric];
|
|
if (identifier.length != kExpectedTeamIDLength) {
|
|
return nil;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case SNTRuleTypeSigningID: {
|
|
// SigningID rules are a combination of `TeamID:SigningID`. The TeamID should
|
|
// be forced to be uppercase, but because very loose rules exist for SigningIDs,
|
|
// their case will be kept as-is. However, platform binaries are expected to
|
|
// have the hardcoded string "platform" as the team ID and the case will be left
|
|
// as is.
|
|
NSArray *sidComponents = [identifier componentsSeparatedByString:@":"];
|
|
if (!sidComponents || sidComponents.count < 2) {
|
|
return nil;
|
|
}
|
|
|
|
// The first component is the TeamID
|
|
NSString *teamID = sidComponents[0];
|
|
|
|
if (![teamID isEqualToString:@"platform"]) {
|
|
teamID =
|
|
[[teamID uppercaseString] stringByTrimmingCharactersInSet:nonUppercaseAlphaNumeric];
|
|
if (teamID.length != kExpectedTeamIDLength) {
|
|
return nil;
|
|
}
|
|
}
|
|
|
|
// The rest of the components are the Signing ID since ":" a legal character.
|
|
// Join all but the last element of the components to rebuild the SigningID.
|
|
NSString *signingID = [[sidComponents
|
|
subarrayWithRange:NSMakeRange(1, sidComponents.count - 1)] componentsJoinedByString:@":"];
|
|
if (signingID.length == 0) {
|
|
return nil;
|
|
}
|
|
|
|
identifier = [NSString stringWithFormat:@"%@:%@", teamID, signingID];
|
|
break;
|
|
}
|
|
|
|
case SNTRuleTypeCDHash: {
|
|
identifier = [[identifier lowercaseString] stringByTrimmingCharactersInSet:nonHex];
|
|
if (identifier.length != CS_CDHASH_LEN * 2) {
|
|
return nil;
|
|
}
|
|
break;
|
|
}
|
|
|
|
default: {
|
|
break;
|
|
}
|
|
}
|
|
|
|
_identifier = identifier;
|
|
_state = state;
|
|
_type = type;
|
|
_customMsg = customMsg;
|
|
_timestamp = timestamp;
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (instancetype)initWithIdentifier:(NSString *)identifier
|
|
state:(SNTRuleState)state
|
|
type:(SNTRuleType)type
|
|
customMsg:(NSString *)customMsg {
|
|
self = [self initWithIdentifier:identifier state:state type:type customMsg:customMsg timestamp:0];
|
|
// Initialize timestamp to current time if rule is transitive.
|
|
if (self && state == SNTRuleStateAllowTransitive) {
|
|
[self resetTimestamp];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
// Converts rule information downloaded from the server into a SNTRule. Because any information
|
|
// not recorded by SNTRule is thrown away here, this method is also responsible for dealing with
|
|
// the extra bundle rule information (bundle_hash & rule_count).
|
|
- (instancetype)initWithDictionary:(NSDictionary *)dict {
|
|
if (![dict isKindOfClass:[NSDictionary class]]) return nil;
|
|
|
|
NSString *identifier = dict[kRuleIdentifier];
|
|
if (![identifier isKindOfClass:[NSString class]] || !identifier.length) {
|
|
identifier = dict[kRuleSHA256];
|
|
}
|
|
if (![identifier isKindOfClass:[NSString class]] || !identifier.length) return nil;
|
|
|
|
NSString *policyString = dict[kRulePolicy];
|
|
SNTRuleState state;
|
|
if (![policyString isKindOfClass:[NSString class]]) return nil;
|
|
if ([policyString isEqual:kRulePolicyAllowlist] ||
|
|
[policyString isEqual:kRulePolicyAllowlistDeprecated]) {
|
|
state = SNTRuleStateAllow;
|
|
} else if ([policyString isEqual:kRulePolicyAllowlistCompiler] ||
|
|
[policyString isEqual:kRulePolicyAllowlistCompilerDeprecated]) {
|
|
state = SNTRuleStateAllowCompiler;
|
|
} else if ([policyString isEqual:kRulePolicyBlocklist] ||
|
|
[policyString isEqual:kRulePolicyBlocklistDeprecated]) {
|
|
state = SNTRuleStateBlock;
|
|
} else if ([policyString isEqual:kRulePolicySilentBlocklist] ||
|
|
[policyString isEqual:kRulePolicySilentBlocklistDeprecated]) {
|
|
state = SNTRuleStateSilentBlock;
|
|
} else if ([policyString isEqual:kRulePolicyRemove]) {
|
|
state = SNTRuleStateRemove;
|
|
} else {
|
|
return nil;
|
|
}
|
|
|
|
NSString *ruleTypeString = dict[kRuleType];
|
|
SNTRuleType type;
|
|
if (![ruleTypeString isKindOfClass:[NSString class]]) return nil;
|
|
if ([ruleTypeString isEqual:kRuleTypeBinary]) {
|
|
type = SNTRuleTypeBinary;
|
|
} else if ([ruleTypeString isEqual:kRuleTypeCertificate]) {
|
|
type = SNTRuleTypeCertificate;
|
|
} else if ([ruleTypeString isEqual:kRuleTypeTeamID]) {
|
|
type = SNTRuleTypeTeamID;
|
|
} else if ([ruleTypeString isEqual:kRuleTypeSigningID]) {
|
|
type = SNTRuleTypeSigningID;
|
|
} else if ([ruleTypeString isEqual:kRuleTypeCDHash]) {
|
|
type = SNTRuleTypeCDHash;
|
|
} else {
|
|
return nil;
|
|
}
|
|
|
|
NSString *customMsg = dict[kRuleCustomMsg];
|
|
if (![customMsg isKindOfClass:[NSString class]] || customMsg.length == 0) {
|
|
customMsg = nil;
|
|
}
|
|
|
|
NSString *customURL = dict[kRuleCustomURL];
|
|
if (![customURL isKindOfClass:[NSString class]] || customURL.length == 0) {
|
|
customURL = nil;
|
|
}
|
|
|
|
SNTRule *r = [self initWithIdentifier:identifier state:state type:type customMsg:customMsg];
|
|
r.customURL = customURL;
|
|
return r;
|
|
}
|
|
|
|
#pragma mark NSSecureCoding
|
|
|
|
#pragma clang diagnostic push
|
|
#pragma clang diagnostic ignored "-Wobjc-literal-conversion"
|
|
#define ENCODE(obj, key) \
|
|
if (obj) [coder encodeObject:obj forKey:key]
|
|
#define DECODE(cls, key) [decoder decodeObjectOfClass:[cls class] forKey:key]
|
|
|
|
+ (BOOL)supportsSecureCoding {
|
|
return YES;
|
|
}
|
|
|
|
- (void)encodeWithCoder:(NSCoder *)coder {
|
|
ENCODE(self.identifier, @"identifier");
|
|
ENCODE(@(self.state), @"state");
|
|
ENCODE(@(self.type), @"type");
|
|
ENCODE(self.customMsg, @"custommsg");
|
|
ENCODE(self.customURL, @"customurl");
|
|
ENCODE(@(self.timestamp), @"timestamp");
|
|
}
|
|
|
|
- (instancetype)initWithCoder:(NSCoder *)decoder {
|
|
self = [super init];
|
|
if (self) {
|
|
_identifier = DECODE(NSString, @"identifier");
|
|
_state = [DECODE(NSNumber, @"state") intValue];
|
|
_type = [DECODE(NSNumber, @"type") intValue];
|
|
_customMsg = DECODE(NSString, @"custommsg");
|
|
_customURL = DECODE(NSString, @"customurl");
|
|
_timestamp = [DECODE(NSNumber, @"timestamp") unsignedIntegerValue];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (NSString *)ruleStateToPolicyString:(SNTRuleState)state {
|
|
switch (state) {
|
|
case SNTRuleStateAllow: return kRulePolicyAllowlist;
|
|
case SNTRuleStateAllowCompiler: return kRulePolicyAllowlistCompiler;
|
|
case SNTRuleStateBlock: return kRulePolicyBlocklist;
|
|
case SNTRuleStateSilentBlock: return kRulePolicySilentBlocklist;
|
|
case SNTRuleStateRemove: return kRulePolicyRemove;
|
|
case SNTRuleStateAllowTransitive: return @"AllowTransitive";
|
|
// This should never be hit. But is here for completion.
|
|
default: return @"Unknown";
|
|
}
|
|
}
|
|
|
|
- (NSString *)ruleTypeToString:(SNTRuleType)ruleType {
|
|
switch (ruleType) {
|
|
case SNTRuleTypeBinary: return kRuleTypeBinary;
|
|
case SNTRuleTypeCertificate: return kRuleTypeCertificate;
|
|
case SNTRuleTypeTeamID: return kRuleTypeTeamID;
|
|
case SNTRuleTypeSigningID: return kRuleTypeSigningID;
|
|
// This should never be hit. If we have rule types of Unknown then there's a
|
|
// coding error somewhere.
|
|
default: return @"Unknown";
|
|
}
|
|
}
|
|
|
|
// Returns an NSDictionary representation of the rule. Primarily use for
|
|
// exporting rules.
|
|
- (NSDictionary *)dictionaryRepresentation {
|
|
return @{
|
|
kRuleIdentifier : self.identifier,
|
|
kRulePolicy : [self ruleStateToPolicyString:self.state],
|
|
kRuleType : [self ruleTypeToString:self.type],
|
|
kRuleCustomMsg : self.customMsg ?: @"",
|
|
kRuleCustomURL : self.customURL ?: @""
|
|
};
|
|
}
|
|
|
|
#undef DECODE
|
|
#undef ENCODE
|
|
#pragma clang diagnostic pop
|
|
|
|
- (BOOL)isEqual:(id)other {
|
|
if (other == self) return YES;
|
|
if (![other isKindOfClass:[SNTRule class]]) return NO;
|
|
SNTRule *o = other;
|
|
return ([self.identifier isEqual:o.identifier] && self.state == o.state && self.type == o.type);
|
|
}
|
|
|
|
- (NSUInteger)hash {
|
|
NSUInteger prime = 31;
|
|
NSUInteger result = 1;
|
|
result = prime * result + [self.identifier hash];
|
|
result = prime * result + self.state;
|
|
result = prime * result + self.type;
|
|
return result;
|
|
}
|
|
|
|
- (NSString *)description {
|
|
return [NSString
|
|
stringWithFormat:@"SNTRule: Identifier: %@, State: %ld, Type: %ld, Timestamp: %lu",
|
|
self.identifier, self.state, self.type, (unsigned long)self.timestamp];
|
|
}
|
|
|
|
#pragma mark Last-access Timestamp
|
|
|
|
- (void)resetTimestamp {
|
|
self.timestamp = (NSUInteger)[[NSDate date] timeIntervalSinceReferenceDate];
|
|
}
|
|
|
|
@end
|