2017-06-21 06:18:46 +02:00
|
|
|
//
|
|
|
|
// RSHTMLMetadataParser.m
|
2017-06-26 01:32:07 +02:00
|
|
|
// RSParser
|
2017-06-21 06:18:46 +02:00
|
|
|
//
|
|
|
|
// Created by Brent Simmons on 3/6/16.
|
|
|
|
// Copyright © 2016 Ranchero Software, LLC. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#import <libxml/xmlstring.h>
|
2017-10-04 22:28:48 +02:00
|
|
|
#import <RSParser/RSHTMLMetadataParser.h>
|
|
|
|
#import <RSParser/RSHTMLMetadata.h>
|
|
|
|
#import <RSParser/RSSAXHTMLParser.h>
|
|
|
|
#import <RSParser/RSSAXParser.h>
|
|
|
|
#import <RSParser/RSParserInternal.h>
|
|
|
|
#import <RSParser/ParserData.h>
|
2017-11-26 20:38:03 +01:00
|
|
|
#import <RSParser/RSHTMLTag.h>
|
2017-06-21 06:18:46 +02:00
|
|
|
|
|
|
|
@interface RSHTMLMetadataParser () <RSSAXHTMLParserDelegate>
|
|
|
|
|
2017-06-26 01:32:07 +02:00
|
|
|
@property (nonatomic, readonly) ParserData *parserData;
|
2017-06-21 06:18:46 +02:00
|
|
|
@property (nonatomic, readwrite) RSHTMLMetadata *metadata;
|
2017-11-26 20:38:03 +01:00
|
|
|
@property (nonatomic) NSMutableArray *tags;
|
2017-06-21 06:18:46 +02:00
|
|
|
@property (nonatomic) BOOL didFinishParsing;
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
|
|
|
|
@implementation RSHTMLMetadataParser
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark - Class Methods
|
|
|
|
|
2017-06-26 01:32:07 +02:00
|
|
|
+ (RSHTMLMetadata *)HTMLMetadataWithParserData:(ParserData *)parserData {
|
2017-06-21 06:18:46 +02:00
|
|
|
|
2017-06-26 01:32:07 +02:00
|
|
|
RSHTMLMetadataParser *parser = [[self alloc] initWithParserData:parserData];
|
2017-06-21 06:18:46 +02:00
|
|
|
return parser.metadata;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark - Init
|
|
|
|
|
2017-06-26 01:32:07 +02:00
|
|
|
- (instancetype)initWithParserData:(ParserData *)parserData {
|
2017-06-21 06:18:46 +02:00
|
|
|
|
2017-06-26 01:32:07 +02:00
|
|
|
NSParameterAssert(parserData.data);
|
|
|
|
NSParameterAssert(parserData.url);
|
2017-06-21 06:18:46 +02:00
|
|
|
|
|
|
|
self = [super init];
|
|
|
|
if (!self) {
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
2017-06-26 01:32:07 +02:00
|
|
|
_parserData = parserData;
|
2017-11-26 20:38:03 +01:00
|
|
|
_tags = [NSMutableArray new];
|
2017-06-21 06:18:46 +02:00
|
|
|
|
|
|
|
[self parse];
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark - Parse
|
|
|
|
|
|
|
|
- (void)parse {
|
|
|
|
|
|
|
|
RSSAXHTMLParser *parser = [[RSSAXHTMLParser alloc] initWithDelegate:self];
|
2017-06-26 01:32:07 +02:00
|
|
|
[parser parseData:self.parserData.data];
|
2017-06-21 06:18:46 +02:00
|
|
|
[parser finishParsing];
|
|
|
|
|
2017-11-26 20:38:03 +01:00
|
|
|
self.metadata = [[RSHTMLMetadata alloc] initWithURLString:self.parserData.url tags:self.tags];
|
2017-06-21 06:18:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static NSString *kHrefKey = @"href";
|
|
|
|
static NSString *kSrcKey = @"src";
|
|
|
|
static NSString *kRelKey = @"rel";
|
|
|
|
|
|
|
|
- (NSString *)linkForDictionary:(NSDictionary *)d {
|
|
|
|
|
2017-06-25 23:06:01 +02:00
|
|
|
NSString *link = [d rsparser_objectForCaseInsensitiveKey:kHrefKey];
|
2017-06-21 06:18:46 +02:00
|
|
|
if (link) {
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
2017-06-25 23:06:01 +02:00
|
|
|
return [d rsparser_objectForCaseInsensitiveKey:kSrcKey];
|
2017-06-21 06:18:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)handleLinkAttributes:(NSDictionary *)d {
|
|
|
|
|
2017-06-25 23:06:01 +02:00
|
|
|
if (RSParserStringIsEmpty([d rsparser_objectForCaseInsensitiveKey:kRelKey])) {
|
2017-06-21 06:18:46 +02:00
|
|
|
return;
|
|
|
|
}
|
2017-06-25 23:06:01 +02:00
|
|
|
if (RSParserStringIsEmpty([self linkForDictionary:d])) {
|
2017-06-21 06:18:46 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-11-26 20:38:03 +01:00
|
|
|
RSHTMLTag *tag = [RSHTMLTag linkTagWithAttributes:d];
|
|
|
|
[self.tags addObject:tag];
|
2017-06-21 06:18:46 +02:00
|
|
|
}
|
|
|
|
|
2017-11-26 20:38:03 +01:00
|
|
|
- (void)handleMetaAttributes:(NSDictionary *)d {
|
|
|
|
|
|
|
|
RSHTMLTag *tag = [RSHTMLTag metaTagWithAttributes:d];
|
|
|
|
[self.tags addObject:tag];
|
|
|
|
}
|
2017-06-21 06:18:46 +02:00
|
|
|
|
|
|
|
#pragma mark - RSSAXHTMLParserDelegate
|
|
|
|
|
|
|
|
static const char *kBody = "body";
|
|
|
|
static const NSInteger kBodyLength = 5;
|
|
|
|
static const char *kLink = "link";
|
|
|
|
static const NSInteger kLinkLength = 5;
|
2017-11-26 20:38:03 +01:00
|
|
|
static const char *kMeta = "meta";
|
|
|
|
static const NSInteger kMetaLength = 5;
|
2017-06-21 06:18:46 +02:00
|
|
|
|
|
|
|
- (void)saxParser:(RSSAXHTMLParser *)SAXParser XMLStartElement:(const xmlChar *)localName attributes:(const xmlChar **)attributes {
|
|
|
|
|
|
|
|
if (self.didFinishParsing) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (RSSAXEqualTags(localName, kBody, kBodyLength)) {
|
|
|
|
self.didFinishParsing = YES;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-11-26 20:38:03 +01:00
|
|
|
if (RSSAXEqualTags(localName, kLink, kLinkLength)) {
|
|
|
|
NSDictionary *d = [SAXParser attributesDictionary:attributes];
|
|
|
|
if (!RSParserObjectIsEmpty(d)) {
|
|
|
|
[self handleLinkAttributes:d];
|
|
|
|
}
|
2017-06-21 06:18:46 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-11-26 20:38:03 +01:00
|
|
|
if (RSSAXEqualTags(localName, kMeta, kMetaLength)) {
|
|
|
|
NSDictionary *d = [SAXParser attributesDictionary:attributes];
|
|
|
|
if (!RSParserObjectIsEmpty(d)) {
|
|
|
|
[self handleMetaAttributes:d];
|
|
|
|
}
|
2017-06-21 06:18:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|