// // SUCommandLineUserDriver.m // Sparkle // // Created by Mayur Pawashe on 4/10/16. // Copyright © 2016 Sparkle Project. All rights reserved. // #import "SPUCommandLineUserDriver.h" #import #import @implementation SPUCommandLineUserDriver { SUUpdatePermissionResponse *_updatePermissionResponse; uint64_t _bytesDownloaded; uint64_t _bytesToDownload; BOOL _deferInstallation; BOOL _verbose; } - (instancetype)initWithUpdatePermissionResponse:(nullable SUUpdatePermissionResponse *)updatePermissionResponse deferInstallation:(BOOL)deferInstallation verbose:(BOOL)verbose { self = [super init]; if (self != nil) { _updatePermissionResponse = updatePermissionResponse; _deferInstallation = deferInstallation; _verbose = verbose; } return self; } - (void)showUpdatePermissionRequest:(SPUUpdatePermissionRequest *)__unused request reply:(void (^)(SUUpdatePermissionResponse *))reply { if (_verbose) { fprintf(stderr, "Granting permission for automatic update checks with sending system profile %s...\n", _updatePermissionResponse.sendSystemProfile ? "enabled" : "disabled"); } reply(_updatePermissionResponse); } - (void)showUserInitiatedUpdateCheckWithCancellation:(void (^)(void))__unused cancellation { if (_verbose) { fprintf(stderr, "Checking for Updates...\n"); } } - (void)displayReleaseNotes:(const char * _Nullable)releaseNotes SPU_OBJC_DIRECT { if (releaseNotes != NULL) { fprintf(stderr, "Release notes:\n"); fprintf(stderr, "%s\n", releaseNotes); } } - (void)displayHTMLReleaseNotes:(NSData *)releaseNotes SPU_OBJC_DIRECT { // Note: this is the only API we rely on here that references AppKit // We shouldn't invoke it when the calling process is ran under root. // If only there was an API to translated HTML -> text that didn't rely on AppKit.. if (geteuid() != 0) { NSAttributedString *attributedString = [[NSAttributedString alloc] initWithHTML:releaseNotes documentAttributes:nil]; [self displayReleaseNotes:attributedString.string.UTF8String]; } } - (void)displayPlainTextReleaseNotes:(NSData *)releaseNotes encoding:(NSStringEncoding)encoding SPU_OBJC_DIRECT { NSString *string = [[NSString alloc] initWithData:releaseNotes encoding:encoding]; [self displayReleaseNotes:string.UTF8String]; } - (void)showUpdateWithAppcastItem:(SUAppcastItem *)appcastItem updateAdjective:(NSString *)updateAdjective { if (_verbose) { fprintf(stderr, "Found %s update! (%s)\n", updateAdjective.UTF8String, appcastItem.displayVersionString.UTF8String); if (appcastItem.itemDescription != nil) { NSData *descriptionData = [appcastItem.itemDescription dataUsingEncoding:NSUTF8StringEncoding]; if (descriptionData != nil) { NSString *itemDescriptionFormat = appcastItem.itemDescriptionFormat; if (itemDescriptionFormat != nil && [itemDescriptionFormat isEqualToString:@"plain-text"]) { [self displayPlainTextReleaseNotes:descriptionData encoding:NSUTF8StringEncoding]; } else { [self displayHTMLReleaseNotes:descriptionData]; } } } } } - (void)showUpdateFoundWithAppcastItem:(SUAppcastItem *)appcastItem state:(SPUUserUpdateState *)state reply:(void (^)(SPUUserUpdateChoice))reply { if (appcastItem.informationOnlyUpdate) { fprintf(stderr, "Found information for new update: %s\n", appcastItem.infoURL.absoluteString.UTF8String); reply(SPUUserUpdateChoiceDismiss); } else { switch (state.stage) { case SPUUserUpdateStageNotDownloaded: [self showUpdateWithAppcastItem:appcastItem updateAdjective:@"new"]; reply(SPUUserUpdateChoiceInstall); break; case SPUUserUpdateStageDownloaded: [self showUpdateWithAppcastItem:appcastItem updateAdjective:@"downloaded"]; reply(SPUUserUpdateChoiceInstall); break; case SPUUserUpdateStageInstalling: if (_deferInstallation) { if (_verbose) { fprintf(stderr, "Deferring Installation.\n"); } reply(SPUUserUpdateChoiceDismiss); } else { reply(SPUUserUpdateChoiceInstall); } break; } } } - (void)showUpdateInFocus { } - (void)showUpdateReleaseNotesWithDownloadData:(SPUDownloadData *)downloadData { if (_verbose) { if (downloadData.MIMEType != nil && [downloadData.MIMEType isEqualToString:@"text/plain"]) { NSStringEncoding encoding; if (downloadData.textEncodingName == nil) { encoding = NSUTF8StringEncoding; } else { CFStringEncoding cfEncoding = CFStringConvertIANACharSetNameToEncoding((CFStringRef)downloadData.textEncodingName); if (cfEncoding != kCFStringEncodingInvalidId) { encoding = CFStringConvertEncodingToNSStringEncoding(cfEncoding); } else { encoding = NSUTF8StringEncoding; } } [self displayPlainTextReleaseNotes:downloadData.data encoding:encoding]; } else { [self displayHTMLReleaseNotes:downloadData.data]; } } } - (void)showUpdateReleaseNotesFailedToDownloadWithError:(NSError *)error { if (_verbose) { fprintf(stderr, "Error: Unable to download release notes: %s\n", error.localizedDescription.UTF8String); } } - (void)showUpdateNotFoundWithError:(NSError *)__unused error acknowledgement:(void (^)(void))acknowledgement { acknowledgement(); } - (void)showUpdaterError:(NSError *)__unused error acknowledgement:(void (^)(void))acknowledgement { acknowledgement(); } - (void)showDownloadInitiatedWithCancellation:(void (^)(void))__unused cancellation { if (_verbose) { fprintf(stderr, "Downloading Update...\n"); } } - (void)showDownloadDidReceiveExpectedContentLength:(uint64_t)expectedContentLength { if (_verbose) { fprintf(stderr, "Downloading %llu bytes...\n", expectedContentLength); } _bytesDownloaded = 0; _bytesToDownload = expectedContentLength; } - (void)showDownloadDidReceiveDataOfLength:(uint64_t)length { _bytesDownloaded += length; // In case our expected content length was incorrect if (_bytesDownloaded > _bytesToDownload) { _bytesToDownload = _bytesDownloaded; } if (_bytesToDownload > 0 && _verbose) { fprintf(stderr, "Downloaded %llu out of %llu bytes (%.0f%%)\n", _bytesDownloaded, _bytesToDownload, (_bytesDownloaded * 100.0 / _bytesToDownload)); } } - (void)showDownloadDidStartExtractingUpdate { if (_verbose) { fprintf(stderr, "Extracting Update...\n"); } } - (void)showExtractionReceivedProgress:(double)progress { if (_verbose) { fprintf(stderr, "Extracting Update (%.0f%%)\n", progress * 100); } } - (void)showReadyToInstallAndRelaunch:(void (^)(SPUUserUpdateChoice))installUpdateHandler { if (_deferInstallation) { if (_verbose) { fprintf(stderr, "Deferring Installation.\n"); } installUpdateHandler(SPUUserUpdateChoiceDismiss); } else { installUpdateHandler(SPUUserUpdateChoiceInstall); } } - (void)showInstallingUpdateWithApplicationTerminated:(BOOL)__unused applicationTerminated retryTerminatingApplication:(void (^)(void))__unused retryTerminatingApplication { if (_verbose) { fprintf(stderr, "Installing Update...\n"); } } - (void)showUpdateInstalledAndRelaunched:(BOOL)__unused relaunched acknowledgement:(void (^)(void))acknowledgement { if (_verbose) { fprintf(stderr, "Installation Finished.\n"); } acknowledgement(); } - (void)dismissUpdateInstallation { } @end