You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
435 lines
15 KiB
435 lines
15 KiB
/*
|
|
* Author: Andreas Linde <mail@andreaslinde.de>
|
|
* Kent Sutherland
|
|
*
|
|
* Copyright (c) 2011 Andreas Linde & Kent Sutherland.
|
|
* All rights reserved.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person
|
|
* obtaining a copy of this software and associated documentation
|
|
* files (the "Software"), to deal in the Software without
|
|
* restriction, including without limitation the rights to use,
|
|
* copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following
|
|
* conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
|
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
* OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
#import "BWQuincyManager.h"
|
|
#import "BWQuincyUI.h"
|
|
#import <sys/sysctl.h>
|
|
|
|
#define SDK_NAME @"Quincy"
|
|
#define SDK_VERSION @"2.1.6"
|
|
|
|
@interface BWQuincyManager ()
|
|
{
|
|
NSString *_customAppVersion;
|
|
}
|
|
|
|
@end
|
|
|
|
@interface BWQuincyManager(private)
|
|
|
|
- (void) startManager;
|
|
|
|
- (void) _postXML:(NSString*)xml toURL:(NSURL*)url;
|
|
- (void) searchCrashLogFile:(NSString *)path;
|
|
- (BOOL) hasPendingCrashReport;
|
|
- (void) returnToMainApplication;
|
|
@end
|
|
|
|
|
|
@implementation BWQuincyManager
|
|
|
|
+ (BWQuincyManager *)sharedQuincyManager {
|
|
static BWQuincyManager *quincyManager = nil;
|
|
|
|
if (quincyManager == nil) {
|
|
quincyManager = [[BWQuincyManager alloc] init];
|
|
}
|
|
|
|
return quincyManager;
|
|
}
|
|
|
|
- (id) init {
|
|
if ((self = [super init])) {
|
|
_serverResult = CrashReportStatusFailureDatabaseNotAvailable;
|
|
_quincyUI = nil;
|
|
|
|
_submissionURL = nil;
|
|
_appIdentifier = nil;
|
|
|
|
_crashFile = nil;
|
|
|
|
self.delegate = nil;
|
|
self.companyName = @"";
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)dealloc {
|
|
_companyName = nil;
|
|
_delegate = nil;
|
|
_submissionURL = nil;
|
|
_appIdentifier = nil;
|
|
}
|
|
|
|
- (void) searchCrashLogFile:(NSString *)path {
|
|
NSFileManager* fman = [NSFileManager defaultManager];
|
|
|
|
NSError* error;
|
|
NSMutableArray* filesWithModificationDate = [NSMutableArray array];
|
|
NSArray* crashLogFiles = [fman contentsOfDirectoryAtPath:path error:&error];
|
|
NSEnumerator* filesEnumerator = [crashLogFiles objectEnumerator];
|
|
NSString* crashFile;
|
|
while((crashFile = [filesEnumerator nextObject])) {
|
|
NSString* crashLogPath = [path stringByAppendingPathComponent:crashFile];
|
|
NSDate* modDate = [[[NSFileManager defaultManager] attributesOfItemAtPath:crashLogPath error:&error] fileModificationDate];
|
|
[filesWithModificationDate addObject:[NSDictionary dictionaryWithObjectsAndKeys:crashFile,@"name",crashLogPath,@"path",modDate,@"modDate",nil]];
|
|
}
|
|
|
|
NSSortDescriptor* dateSortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"modDate" ascending:YES];
|
|
NSArray* sortedFiles = [filesWithModificationDate sortedArrayUsingDescriptors:[NSArray arrayWithObject:dateSortDescriptor]];
|
|
|
|
NSPredicate* filterPredicate = [NSPredicate predicateWithFormat:@"name BEGINSWITH %@", [self applicationName]];
|
|
NSArray* filteredFiles = [sortedFiles filteredArrayUsingPredicate:filterPredicate];
|
|
|
|
_crashFile = [[[filteredFiles valueForKeyPath:@"path"] lastObject] copy];
|
|
}
|
|
|
|
#pragma mark -
|
|
#pragma mark setter
|
|
- (void)setSubmissionURL:(NSString *)anSubmissionURL {
|
|
if (_submissionURL != anSubmissionURL) {
|
|
_submissionURL = [anSubmissionURL copy];
|
|
}
|
|
|
|
[self performSelector:@selector(startManager) withObject:nil afterDelay:0.1f];
|
|
}
|
|
|
|
- (void)setAppIdentifier:(NSString *)anAppIdentifier {
|
|
if (_appIdentifier != anAppIdentifier) {
|
|
_appIdentifier = [anAppIdentifier copy];
|
|
}
|
|
|
|
[self setSubmissionURL:@"https://rink.hockeyapp.net/"];
|
|
}
|
|
|
|
- (void)storeLastCrashDate:(NSDate *) date {
|
|
[[NSUserDefaults standardUserDefaults] setValue:date forKey:@"CrashReportSender.lastCrashDate"];
|
|
[[NSUserDefaults standardUserDefaults] synchronize];
|
|
}
|
|
|
|
- (NSDate *)loadLastCrashDate {
|
|
NSDate *date = [[NSUserDefaults standardUserDefaults] valueForKey:@"CrashReportSender.lastCrashDate"];
|
|
return date ?: [NSDate distantPast];
|
|
}
|
|
|
|
- (void)storeAppVersion:(NSString *) version {
|
|
[[NSUserDefaults standardUserDefaults] setValue:version forKey:@"CrashReportSender.appVersion"];
|
|
[[NSUserDefaults standardUserDefaults] synchronize];
|
|
}
|
|
|
|
- (NSString *)loadAppVersion {
|
|
NSString *appVersion = [[NSUserDefaults standardUserDefaults] valueForKey:@"CrashReportSender.appVersion"];
|
|
return appVersion ?: nil;
|
|
}
|
|
|
|
#pragma mark -
|
|
#pragma mark GetCrashData
|
|
|
|
- (BOOL) hasPendingCrashReport {
|
|
BOOL returnValue = NO;
|
|
|
|
NSString *appVersion = [self loadAppVersion];
|
|
NSDate *lastCrashDate = [self loadLastCrashDate];
|
|
|
|
if (!appVersion || ![appVersion isEqualToString:[self applicationVersion]] || [lastCrashDate isEqualToDate:[NSDate distantPast]]) {
|
|
[self storeAppVersion:[self applicationVersion]];
|
|
[self storeLastCrashDate:[NSDate date]];
|
|
return NO;
|
|
}
|
|
|
|
NSArray* libraryDirectories = NSSearchPathForDirectoriesInDomains(NSLibraryDirectory, NSUserDomainMask, TRUE);
|
|
// Snow Leopard is having the log files in another location
|
|
[self searchCrashLogFile:[[libraryDirectories lastObject] stringByAppendingPathComponent:@"Logs/DiagnosticReports"]];
|
|
if (_crashFile == nil) {
|
|
[self searchCrashLogFile:[[libraryDirectories lastObject] stringByAppendingPathComponent:@"Logs/CrashReporter"]];
|
|
if (_crashFile == nil) {
|
|
NSString *sandboxFolder = [NSString stringWithFormat:@"/Containers/%@/Data/Library", [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleIdentifier"]];
|
|
if ([[libraryDirectories lastObject] rangeOfString:sandboxFolder].location != NSNotFound) {
|
|
NSString *libFolderName = [[libraryDirectories lastObject] stringByReplacingOccurrencesOfString:sandboxFolder withString:@""];
|
|
[self searchCrashLogFile:[libFolderName stringByAppendingPathComponent:@"Logs/DiagnosticReports"]];
|
|
}
|
|
}
|
|
// Search machine diagnostic reports directory
|
|
if (_crashFile == nil) {
|
|
NSArray* libraryDirectories = NSSearchPathForDirectoriesInDomains(NSLibraryDirectory, NSLocalDomainMask, TRUE);
|
|
[self searchCrashLogFile:[[libraryDirectories lastObject] stringByAppendingPathComponent:@"Logs/DiagnosticReports"]];
|
|
if (_crashFile == nil) {
|
|
[self searchCrashLogFile:[[libraryDirectories lastObject] stringByAppendingPathComponent:@"Logs/CrashReporter"]];
|
|
}
|
|
}
|
|
}
|
|
|
|
if (_crashFile) {
|
|
NSError* error;
|
|
|
|
NSDate *crashLogModificationDate = [[[NSFileManager defaultManager] attributesOfItemAtPath:_crashFile error:&error] fileModificationDate];
|
|
unsigned long long crashLogFileSize = [[[NSFileManager defaultManager] attributesOfItemAtPath:_crashFile error:&error] fileSize];
|
|
if ([crashLogModificationDate compare: lastCrashDate] == NSOrderedDescending && crashLogFileSize > 0) {
|
|
[self storeLastCrashDate:crashLogModificationDate];
|
|
returnValue = YES;
|
|
}
|
|
}
|
|
|
|
return returnValue;
|
|
}
|
|
|
|
- (void) returnToMainApplication {
|
|
if ( self.delegate != nil && [self.delegate respondsToSelector:@selector(showMainApplicationWindow)])
|
|
[self.delegate showMainApplicationWindow];
|
|
}
|
|
|
|
- (void) startManager {
|
|
if ([self hasPendingCrashReport]) {
|
|
if (!self.autoSubmitCrashReport) {
|
|
_quincyUI = [[BWQuincyUI alloc] initWithManager:self crashFile:_crashFile companyName:_companyName applicationName:[self applicationName]];
|
|
[_quincyUI askCrashReportDetails];
|
|
} else {
|
|
NSError* error = nil;
|
|
NSString *crashLogs = [NSString stringWithContentsOfFile:_crashFile encoding:NSUTF8StringEncoding error:&error];
|
|
if (!error) {
|
|
NSString *lastCrash = [[crashLogs componentsSeparatedByString: @"**********\n\n"] lastObject];
|
|
|
|
NSString* description = @"";
|
|
|
|
if (_delegate && [_delegate respondsToSelector:@selector(crashReportDescription)]) {
|
|
description = [_delegate crashReportDescription];
|
|
}
|
|
|
|
[self sendReportCrash:lastCrash description:description];
|
|
} else {
|
|
[self returnToMainApplication];
|
|
}
|
|
}
|
|
} else {
|
|
[self returnToMainApplication];
|
|
}
|
|
}
|
|
|
|
- (NSString*) modelVersion {
|
|
NSString * modelString = nil;
|
|
int modelInfo[2] = { CTL_HW, HW_MODEL };
|
|
size_t modelSize;
|
|
|
|
if (sysctl(modelInfo,
|
|
2,
|
|
NULL,
|
|
&modelSize,
|
|
NULL, 0) == 0) {
|
|
void * modelData = malloc(modelSize);
|
|
|
|
if (modelData) {
|
|
if (sysctl(modelInfo,
|
|
2,
|
|
modelData,
|
|
&modelSize,
|
|
NULL, 0) == 0) {
|
|
modelString = [NSString stringWithUTF8String:modelData];
|
|
}
|
|
|
|
free(modelData);
|
|
}
|
|
}
|
|
|
|
return modelString;
|
|
}
|
|
|
|
|
|
|
|
- (void) cancelReport {
|
|
[self returnToMainApplication];
|
|
}
|
|
|
|
|
|
- (void) sendReportCrash:(NSString*)crashContent
|
|
description:(NSString*)notes
|
|
{
|
|
NSString *userid = @"";
|
|
NSString *contact = @"";
|
|
|
|
SInt32 versionMajor, versionMinor, versionBugFix;
|
|
if (Gestalt(gestaltSystemVersionMajor, &versionMajor) != noErr) versionMajor = 0;
|
|
if (Gestalt(gestaltSystemVersionMinor, &versionMinor) != noErr) versionMinor= 0;
|
|
if (Gestalt(gestaltSystemVersionBugFix, &versionBugFix) != noErr) versionBugFix = 0;
|
|
|
|
NSString* xml = [NSString stringWithFormat:@"<crash><applicationname>%s</applicationname><bundleidentifier>%s</bundleidentifier><systemversion>%@</systemversion><senderversion>%@</senderversion><version>%@</version><platform>%@</platform><userid>%@</userid><contact>%@</contact><description><![CDATA[%@]]></description><log><![CDATA[%@]]></log></crash>",
|
|
[[self applicationName] UTF8String],
|
|
[[[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleIdentifier"] UTF8String],
|
|
[NSString stringWithFormat:@"%i.%i.%i", versionMajor, versionMinor, versionBugFix],
|
|
[self applicationVersion],
|
|
[self applicationVersion],
|
|
[self modelVersion],
|
|
userid,
|
|
contact,
|
|
notes,
|
|
crashContent
|
|
];
|
|
|
|
|
|
[self returnToMainApplication];
|
|
|
|
[self _postXML:[NSString stringWithFormat:@"<crashes>%@</crashes>", xml] toURL:[NSURL URLWithString:self.submissionURL]];
|
|
}
|
|
|
|
- (void)_postXML:(NSString*)xml toURL:(NSURL*)url {
|
|
NSMutableURLRequest *request = nil;
|
|
NSString *boundary = @"----FOO";
|
|
|
|
if (self.appIdentifier) {
|
|
request = [NSMutableURLRequest requestWithURL:
|
|
[NSURL URLWithString:[NSString stringWithFormat:@"%@api/2/apps/%@/crashes?sdk=%@&sdk_version=%@",
|
|
self.submissionURL,
|
|
[self.appIdentifier stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding],
|
|
SDK_NAME,
|
|
SDK_VERSION
|
|
]
|
|
]];
|
|
} else {
|
|
request = [NSMutableURLRequest requestWithURL:url];
|
|
}
|
|
|
|
[request setValue:@"Quincy/Mac" forHTTPHeaderField:@"User-Agent"];
|
|
[request setValue:@"gzip" forHTTPHeaderField:@"Accept-Encoding"];
|
|
[request setTimeoutInterval: 15];
|
|
[request setHTTPMethod:@"POST"];
|
|
NSString *contentType = [NSString stringWithFormat:@"multipart/form-data; boundary=%@", boundary];
|
|
[request setValue:contentType forHTTPHeaderField:@"Content-type"];
|
|
|
|
NSMutableData *postBody = [NSMutableData data];
|
|
[postBody appendData:[[NSString stringWithFormat:@"--%@\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]];
|
|
if (self.appIdentifier) {
|
|
[postBody appendData:[@"Content-Disposition: form-data; name=\"xml\"; filename=\"crash.xml\"\r\n" dataUsingEncoding:NSUTF8StringEncoding]];
|
|
[postBody appendData:[[NSString stringWithFormat:@"Content-Type: text/xml\r\n\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
|
|
} else {
|
|
[postBody appendData:[@"Content-Disposition: form-data; name=\"xmlstring\"\r\n\r\n" dataUsingEncoding:NSUTF8StringEncoding]];
|
|
}
|
|
[postBody appendData:[xml dataUsingEncoding:NSUTF8StringEncoding]];
|
|
[postBody appendData:[[NSString stringWithFormat:@"\r\n--%@--\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]];
|
|
[request setHTTPBody:postBody];
|
|
|
|
_serverResult = CrashReportStatusUnknown;
|
|
_statusCode = 200;
|
|
|
|
NSHTTPURLResponse *response = nil;
|
|
NSError *error = nil;
|
|
|
|
NSData *responseData = nil;
|
|
responseData = [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error];
|
|
_statusCode = [response statusCode];
|
|
|
|
if (responseData != nil) {
|
|
if (_statusCode >= 200 && _statusCode < 400) {
|
|
NSXMLParser *parser = [[NSXMLParser alloc] initWithData:responseData];
|
|
// Set self as the delegate of the parser so that it will receive the parser delegate methods callbacks.
|
|
[parser setDelegate:self];
|
|
// Depending on the XML document you're parsing, you may want to enable these features of NSXMLParser.
|
|
[parser setShouldProcessNamespaces:NO];
|
|
[parser setShouldReportNamespacePrefixes:NO];
|
|
[parser setShouldResolveExternalEntities:NO];
|
|
|
|
[parser parse];
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#pragma mark NSXMLParser
|
|
|
|
- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict {
|
|
if (qName) {
|
|
elementName = qName;
|
|
}
|
|
|
|
if ([elementName isEqualToString:@"result"]) {
|
|
_contentOfProperty = [NSMutableString string];
|
|
}
|
|
}
|
|
|
|
- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName {
|
|
if (qName) {
|
|
elementName = qName;
|
|
}
|
|
|
|
if ([elementName isEqualToString:@"result"]) {
|
|
if ([_contentOfProperty intValue] > _serverResult) {
|
|
_serverResult = [_contentOfProperty intValue];
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string {
|
|
if (_contentOfProperty) {
|
|
// If the current element is one whose content we care about, append 'string'
|
|
// to the property that holds the content of the current element.
|
|
if (string != nil) {
|
|
[_contentOfProperty appendString:string];
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#pragma mark GetterSetter
|
|
|
|
- (NSString *) applicationName {
|
|
NSString *applicationName = [[[NSBundle mainBundle] localizedInfoDictionary] valueForKey: @"CFBundleExecutable"];
|
|
|
|
if (!applicationName)
|
|
applicationName = [[[NSBundle mainBundle] infoDictionary] valueForKey: @"CFBundleExecutable"];
|
|
|
|
return applicationName;
|
|
}
|
|
|
|
|
|
- (NSString*) applicationVersionString {
|
|
NSString* string = [[[NSBundle mainBundle] localizedInfoDictionary] valueForKey: @"CFBundleShortVersionString"];
|
|
|
|
if (!string)
|
|
string = [[[NSBundle mainBundle] infoDictionary] valueForKey: @"CFBundleShortVersionString"];
|
|
|
|
return string;
|
|
}
|
|
|
|
- (void)setApplicationVersion:(NSString *)appVersion
|
|
{
|
|
_customAppVersion = appVersion;
|
|
}
|
|
|
|
- (NSString *) applicationVersion {
|
|
if (_customAppVersion)
|
|
return _customAppVersion;
|
|
|
|
NSString* string = [[[NSBundle mainBundle] localizedInfoDictionary] valueForKey: @"CFBundleVersion"];
|
|
|
|
if (!string)
|
|
string = [[[NSBundle mainBundle] infoDictionary] valueForKey: @"CFBundleVersion"];
|
|
|
|
return string;
|
|
}
|
|
|
|
@end
|
|
|