174 lines
		
	
	
	
		
			5.5 KiB
			
		
	
	
	
		
			Text
		
	
	
	
	
	
			
		
		
	
	
			174 lines
		
	
	
	
		
			5.5 KiB
			
		
	
	
	
		
			Text
		
	
	
	
	
	
// Copyright (c) 2013 GitHub, Inc.
 | 
						|
// Use of this source code is governed by the MIT license that can be
 | 
						|
// found in the LICENSE file.
 | 
						|
 | 
						|
#include "atom/browser/auto_updater.h"
 | 
						|
 | 
						|
#import <ReactiveCocoa/NSObject+RACPropertySubscribing.h>
 | 
						|
#import <ReactiveCocoa/RACCommand.h>
 | 
						|
#import <ReactiveCocoa/RACSignal.h>
 | 
						|
#import <Squirrel/Squirrel.h>
 | 
						|
 | 
						|
#include "atom/browser/browser.h"
 | 
						|
#include "atom/common/native_mate_converters/value_converter.h"
 | 
						|
#include "base/bind.h"
 | 
						|
#include "base/strings/sys_string_conversions.h"
 | 
						|
#include "base/time/time.h"
 | 
						|
#include "native_mate/converter.h"
 | 
						|
#include "native_mate/dictionary.h"
 | 
						|
 | 
						|
namespace auto_updater {
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
// The gloal SQRLUpdater object.
 | 
						|
SQRLUpdater* g_updater = nil;
 | 
						|
 | 
						|
}  // namespace
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
bool g_update_available = false;
 | 
						|
std::string update_url_ = "";
 | 
						|
 | 
						|
}  // namespace
 | 
						|
 | 
						|
std::string AutoUpdater::GetFeedURL() {
 | 
						|
  return update_url_;
 | 
						|
}
 | 
						|
 | 
						|
// static
 | 
						|
void AutoUpdater::SetFeedURL(mate::Arguments* args) {
 | 
						|
  mate::Dictionary opts;
 | 
						|
  std::string feed;
 | 
						|
  HeaderMap requestHeaders;
 | 
						|
  std::string serverType = "default";
 | 
						|
  if (args->GetNext(&opts)) {
 | 
						|
    if (!opts.Get("url", &feed)) {
 | 
						|
      args->ThrowError(
 | 
						|
          "Expected options object to contain a 'url' string property in "
 | 
						|
          "setFeedUrl call");
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    opts.Get("headers", &requestHeaders);
 | 
						|
    opts.Get("serverType", &serverType);
 | 
						|
    if (serverType != "default" && serverType != "json") {
 | 
						|
      args->ThrowError("Expected serverType to be 'default' or 'json'");
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  } else if (args->GetNext(&feed)) {
 | 
						|
    args->GetNext(&requestHeaders);
 | 
						|
  } else {
 | 
						|
    args->ThrowError(
 | 
						|
        "Expected an options object with a 'url' property to be provided");
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  Delegate* delegate = GetDelegate();
 | 
						|
  if (!delegate)
 | 
						|
    return;
 | 
						|
 | 
						|
  update_url_ = feed;
 | 
						|
 | 
						|
  NSURL* url = [NSURL URLWithString:base::SysUTF8ToNSString(feed)];
 | 
						|
  NSMutableURLRequest* urlRequest = [NSMutableURLRequest requestWithURL:url];
 | 
						|
 | 
						|
  for (const auto& it : requestHeaders) {
 | 
						|
    [urlRequest setValue:base::SysUTF8ToNSString(it.second)
 | 
						|
        forHTTPHeaderField:base::SysUTF8ToNSString(it.first)];
 | 
						|
  }
 | 
						|
 | 
						|
  if (g_updater)
 | 
						|
    [g_updater release];
 | 
						|
 | 
						|
  // Initialize the SQRLUpdater.
 | 
						|
  @try {
 | 
						|
    if (serverType == "json") {
 | 
						|
      NSString* nsAppVersion =
 | 
						|
          base::SysUTF8ToNSString(atom::Browser::Get()->GetVersion());
 | 
						|
      g_updater = [[SQRLUpdater alloc] initWithUpdateRequest:urlRequest
 | 
						|
                                                  forVersion:nsAppVersion];
 | 
						|
    } else {
 | 
						|
      // default
 | 
						|
      g_updater = [[SQRLUpdater alloc] initWithUpdateRequest:urlRequest];
 | 
						|
    }
 | 
						|
  } @catch (NSException* error) {
 | 
						|
    delegate->OnError(base::SysNSStringToUTF8(error.reason));
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  [[g_updater rac_valuesForKeyPath:@"state" observer:g_updater]
 | 
						|
      subscribeNext:^(NSNumber* stateNumber) {
 | 
						|
        int state = [stateNumber integerValue];
 | 
						|
        // Dispatching the event on main thread.
 | 
						|
        dispatch_async(dispatch_get_main_queue(), ^{
 | 
						|
          if (state == SQRLUpdaterStateCheckingForUpdate)
 | 
						|
            delegate->OnCheckingForUpdate();
 | 
						|
          else if (state == SQRLUpdaterStateDownloadingUpdate)
 | 
						|
            delegate->OnUpdateAvailable();
 | 
						|
        });
 | 
						|
      }];
 | 
						|
}
 | 
						|
 | 
						|
// static
 | 
						|
void AutoUpdater::CheckForUpdates() {
 | 
						|
  Delegate* delegate = GetDelegate();
 | 
						|
  if (!delegate)
 | 
						|
    return;
 | 
						|
 | 
						|
  [[[[g_updater.checkForUpdatesCommand execute:nil]
 | 
						|
      // Send a `nil` after everything...
 | 
						|
      concat:[RACSignal return:nil]]
 | 
						|
      // But only take the first value. If an update is sent, we'll get that.
 | 
						|
      // Otherwise, we'll get our inserted `nil` value.
 | 
						|
      take:1]
 | 
						|
      subscribeNext:^(SQRLDownloadedUpdate* downloadedUpdate) {
 | 
						|
        if (downloadedUpdate) {
 | 
						|
          g_update_available = true;
 | 
						|
          SQRLUpdate* update = downloadedUpdate.update;
 | 
						|
          // There is a new update that has been downloaded.
 | 
						|
          delegate->OnUpdateDownloaded(
 | 
						|
              base::SysNSStringToUTF8(update.releaseNotes),
 | 
						|
              base::SysNSStringToUTF8(update.releaseName),
 | 
						|
              base::Time::FromDoubleT(update.releaseDate.timeIntervalSince1970),
 | 
						|
              base::SysNSStringToUTF8(update.updateURL.absoluteString));
 | 
						|
        } else {
 | 
						|
          g_update_available = false;
 | 
						|
          // When the completed event is sent with no update, then we know there
 | 
						|
          // is no update available.
 | 
						|
          delegate->OnUpdateNotAvailable();
 | 
						|
        }
 | 
						|
      }
 | 
						|
      error:^(NSError* error) {
 | 
						|
        NSMutableString* failureString =
 | 
						|
            [NSMutableString stringWithString:error.localizedDescription];
 | 
						|
        if (error.localizedFailureReason) {
 | 
						|
          [failureString appendString:@": "];
 | 
						|
          [failureString appendString:error.localizedFailureReason];
 | 
						|
        }
 | 
						|
        if (error.localizedRecoverySuggestion) {
 | 
						|
          if (![failureString hasSuffix:@"."])
 | 
						|
            [failureString appendString:@"."];
 | 
						|
          [failureString appendString:@" "];
 | 
						|
          [failureString appendString:error.localizedRecoverySuggestion];
 | 
						|
        }
 | 
						|
        delegate->OnError(base::SysNSStringToUTF8(failureString), error.code,
 | 
						|
                          base::SysNSStringToUTF8(error.domain));
 | 
						|
      }];
 | 
						|
}
 | 
						|
 | 
						|
void AutoUpdater::QuitAndInstall() {
 | 
						|
  Delegate* delegate = AutoUpdater::GetDelegate();
 | 
						|
  if (g_update_available) {
 | 
						|
    [[g_updater relaunchToInstallUpdate] subscribeError:^(NSError* error) {
 | 
						|
      if (delegate)
 | 
						|
        delegate->OnError(base::SysNSStringToUTF8(error.localizedDescription),
 | 
						|
                          error.code, base::SysNSStringToUTF8(error.domain));
 | 
						|
    }];
 | 
						|
  } else {
 | 
						|
    if (delegate)
 | 
						|
      delegate->OnError("No update available, can't quit and install");
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace auto_updater
 |