mirror of
https://github.com/status-im/react-native.git
synced 2025-01-14 11:34:23 +00:00
88ac40666c
Summary: public This diff replaces the RegEx module method parser with a handwritten recursive descent parser that's faster and easier to maintain. The new parser is ~8 times faster when tested on the UIManager.managerChildren() method, and uses ~1/10 as much RAM. The new parser also supports lightweight generics, and is more tolerant of white space. (This means that you now can – and should – use types like `NSArray<NSString *> *` for your exported properties and method arguments, instead of `NSStringArray`). Reviewed By: jspahrsummers Differential Revision: D2736636 fb-gh-sync-id: f6a11431935fa8acc8ac36f3471032ec9a1c8490
119 lines
2.9 KiB
Objective-C
119 lines
2.9 KiB
Objective-C
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*/
|
|
|
|
#import "RCTParserUtils.h"
|
|
|
|
#import "RCTLog.h"
|
|
|
|
@implementation RCTParserUtils
|
|
|
|
BOOL RCTReadChar(const char **input, char c)
|
|
{
|
|
if (**input == c) {
|
|
(*input)++;
|
|
return YES;
|
|
}
|
|
return NO;
|
|
}
|
|
|
|
BOOL RCTReadString(const char **input, const char *string)
|
|
{
|
|
int i;
|
|
for (i = 0; string[i] != 0; i++) {
|
|
if (string[i] != (*input)[i]) {
|
|
return NO;
|
|
}
|
|
}
|
|
*input += i;
|
|
return YES;
|
|
}
|
|
|
|
void RCTSkipWhitespace(const char **input)
|
|
{
|
|
while (isspace(**input)) {
|
|
(*input)++;
|
|
}
|
|
}
|
|
|
|
static BOOL RCTIsIdentifierHead(const char c)
|
|
{
|
|
return isalpha(c) || c == '_';
|
|
}
|
|
|
|
static BOOL RCTIsIdentifierTail(const char c)
|
|
{
|
|
return isalnum(c) || c == '_';
|
|
}
|
|
|
|
BOOL RCTParseIdentifier(const char **input, NSString **string)
|
|
{
|
|
const char *start = *input;
|
|
if (!RCTIsIdentifierHead(**input)) {
|
|
return NO;
|
|
}
|
|
(*input)++;
|
|
while (RCTIsIdentifierTail(**input)) {
|
|
(*input)++;
|
|
}
|
|
if (string) {
|
|
*string = [[NSString alloc] initWithBytes:start
|
|
length:(NSInteger)(*input - start)
|
|
encoding:NSASCIIStringEncoding];
|
|
}
|
|
return YES;
|
|
}
|
|
|
|
static BOOL RCTIsCollectionType(NSString *type)
|
|
{
|
|
static NSSet *collectionTypes;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
collectionTypes = [[NSSet alloc] initWithObjects:
|
|
@"NSArray", @"NSSet", @"NSDictionary", nil];
|
|
});
|
|
return [collectionTypes containsObject:type];
|
|
}
|
|
|
|
NSString *RCTParseType(const char **input)
|
|
{
|
|
NSString *type;
|
|
RCTParseIdentifier(input, &type);
|
|
RCTSkipWhitespace(input);
|
|
if (RCTReadChar(input, '<')) {
|
|
RCTSkipWhitespace(input);
|
|
NSString *subtype = RCTParseType(input);
|
|
if (RCTIsCollectionType(type)) {
|
|
if ([type isEqualToString:@"NSDictionary"]) {
|
|
// Dictionaries have both a key *and* value type, but the key type has
|
|
// to be a string for JSON, so we only care about the value type
|
|
if (RCT_DEBUG && ![subtype isEqualToString:@"NSString"]) {
|
|
RCTLogError(@"%@ is not a valid key type for a JSON dictionary", subtype);
|
|
}
|
|
RCTSkipWhitespace(input);
|
|
RCTReadChar(input, ',');
|
|
RCTSkipWhitespace(input);
|
|
subtype = RCTParseType(input);
|
|
}
|
|
if (![subtype isEqualToString:@"id"]) {
|
|
type = [type stringByReplacingCharactersInRange:(NSRange){0, 2 /* "NS" */}
|
|
withString:subtype];
|
|
}
|
|
} else {
|
|
// It's a protocol rather than a generic collection - ignore it
|
|
}
|
|
RCTSkipWhitespace(input);
|
|
RCTReadChar(input, '>');
|
|
}
|
|
RCTSkipWhitespace(input);
|
|
RCTReadChar(input, '*');
|
|
return type;
|
|
}
|
|
|
|
@end
|