forked from dolphin-emu/dolphin
		
	
		
			
				
	
	
		
			191 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			191 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
// Copyright 2018 Dolphin Emulator Project
 | 
						|
// Licensed under GPLv2+
 | 
						|
// Refer to the license.txt file included.
 | 
						|
 | 
						|
#include "UICommon/AutoUpdate.h"
 | 
						|
 | 
						|
#include <picojson/picojson.h>
 | 
						|
#include <string>
 | 
						|
 | 
						|
#include "Common/CommonPaths.h"
 | 
						|
#include "Common/FileUtil.h"
 | 
						|
#include "Common/HttpRequest.h"
 | 
						|
#include "Common/Logging/Log.h"
 | 
						|
#include "Common/StringUtil.h"
 | 
						|
#include "Common/scmrev.h"
 | 
						|
#include "Core/ConfigManager.h"
 | 
						|
 | 
						|
#ifdef _WIN32
 | 
						|
#include <Windows.h>
 | 
						|
#endif
 | 
						|
 | 
						|
namespace
 | 
						|
{
 | 
						|
#ifdef _WIN32
 | 
						|
 | 
						|
const char UPDATER_FILENAME[] = "Updater.exe";
 | 
						|
const char UPDATER_RELOC_FILENAME[] = "Updater.2.exe";
 | 
						|
const char UPDATER_LOG_FILE[] = "Updater.log";
 | 
						|
 | 
						|
std::wstring MakeUpdaterCommandLine(const std::map<std::string, std::string>& flags)
 | 
						|
{
 | 
						|
  std::wstring cmdline = UTF8ToUTF16(UPDATER_FILENAME) + L" ";  // Start with a fake argv[0].
 | 
						|
  for (const auto& pair : flags)
 | 
						|
  {
 | 
						|
    std::string value = "--" + pair.first + "=" + pair.second;
 | 
						|
    value = ReplaceAll(value, "\"", "\\\"");  // Escape double quotes.
 | 
						|
    value = "\"" + value + "\" ";
 | 
						|
    cmdline += UTF8ToUTF16(value);
 | 
						|
  }
 | 
						|
  return cmdline;
 | 
						|
}
 | 
						|
 | 
						|
// Used to remove the relocated updater file once we don't need it anymore.
 | 
						|
void CleanupFromPreviousUpdate()
 | 
						|
{
 | 
						|
  std::string reloc_updater_path = File::GetExeDirectory() + DIR_SEP + UPDATER_RELOC_FILENAME;
 | 
						|
  File::Delete(reloc_updater_path);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
// This ignores i18n because most of the text in there (change descriptions) is only going to be
 | 
						|
// written in english anyway.
 | 
						|
std::string GenerateChangelog(const picojson::array& versions)
 | 
						|
{
 | 
						|
  std::string changelog;
 | 
						|
  for (const auto& ver : versions)
 | 
						|
  {
 | 
						|
    if (!ver.is<picojson::object>())
 | 
						|
      continue;
 | 
						|
    picojson::object ver_obj = ver.get<picojson::object>();
 | 
						|
 | 
						|
    if (ver_obj["changelog_html"].is<picojson::null>())
 | 
						|
    {
 | 
						|
      if (!changelog.empty())
 | 
						|
        changelog += "<div style=\"margin-top: 0.4em;\"></div>";  // Vertical spacing.
 | 
						|
 | 
						|
      // Try to link to the PR if we have this info. Otherwise just show shortrev.
 | 
						|
      if (ver_obj["pr_url"].is<std::string>())
 | 
						|
      {
 | 
						|
        changelog += "<a href=\"" + ver_obj["pr_url"].get<std::string>() + "\">" +
 | 
						|
                     ver_obj["shortrev"].get<std::string>() + "</a>";
 | 
						|
      }
 | 
						|
      else
 | 
						|
      {
 | 
						|
        changelog += ver_obj["shortrev"].get<std::string>();
 | 
						|
      }
 | 
						|
 | 
						|
      changelog += " by <a href = \"" + ver_obj["author_url"].get<std::string>() + "\">" +
 | 
						|
                   ver_obj["author"].get<std::string>() + "</a> — " +
 | 
						|
                   ver_obj["short_descr"].get<std::string>();
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
      if (!changelog.empty())
 | 
						|
        changelog += "<hr>";
 | 
						|
      changelog += "<b>Dolphin " + ver_obj["shortrev"].get<std::string>() + "</b>";
 | 
						|
      changelog += "<p>" + ver_obj["changelog_html"].get<std::string>() + "</p>";
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return changelog;
 | 
						|
}
 | 
						|
}  // namespace
 | 
						|
 | 
						|
bool AutoUpdateChecker::SystemSupportsAutoUpdates()
 | 
						|
{
 | 
						|
#ifdef _WIN32
 | 
						|
  return true;
 | 
						|
#else
 | 
						|
  return false;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
void AutoUpdateChecker::CheckForUpdate()
 | 
						|
{
 | 
						|
  // Don't bother checking if updates are not supported or not enabled.
 | 
						|
  if (!SystemSupportsAutoUpdates() || SConfig::GetInstance().m_auto_update_track.empty())
 | 
						|
    return;
 | 
						|
 | 
						|
#ifdef _WIN32
 | 
						|
  CleanupFromPreviousUpdate();
 | 
						|
#endif
 | 
						|
 | 
						|
  std::string version_hash = SConfig::GetInstance().m_auto_update_hash_override.empty() ?
 | 
						|
                                 SCM_REV_STR :
 | 
						|
                                 SConfig::GetInstance().m_auto_update_hash_override;
 | 
						|
  std::string url = "https://dolphin-emu.org/update/check/v0/" +
 | 
						|
                    SConfig::GetInstance().m_auto_update_track + "/" + version_hash;
 | 
						|
 | 
						|
  Common::HttpRequest req{std::chrono::seconds{10}};
 | 
						|
  auto resp = req.Get(url);
 | 
						|
  if (!resp)
 | 
						|
  {
 | 
						|
    ERROR_LOG(COMMON, "Auto-update request failed");
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  std::string contents(reinterpret_cast<char*>(resp->data()), resp->size());
 | 
						|
  INFO_LOG(COMMON, "Auto-update JSON response: %s", contents.c_str());
 | 
						|
 | 
						|
  picojson::value json;
 | 
						|
  std::string err = picojson::parse(json, contents);
 | 
						|
  if (!err.empty())
 | 
						|
  {
 | 
						|
    ERROR_LOG(COMMON, "Invalid JSON received from auto-update service: %s", err.c_str());
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  picojson::object obj = json.get<picojson::object>();
 | 
						|
 | 
						|
  if (obj["status"].get<std::string>() != "outdated")
 | 
						|
  {
 | 
						|
    INFO_LOG(COMMON, "Auto-update status: we are up to date.");
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  NewVersionInformation nvi;
 | 
						|
  nvi.this_manifest_url = obj["old"].get<picojson::object>()["manifest"].get<std::string>();
 | 
						|
  nvi.next_manifest_url = obj["new"].get<picojson::object>()["manifest"].get<std::string>();
 | 
						|
  nvi.content_store_url = obj["content-store"].get<std::string>();
 | 
						|
  nvi.new_shortrev = obj["new"].get<picojson::object>()["name"].get<std::string>();
 | 
						|
  nvi.new_hash = obj["new"].get<picojson::object>()["hash"].get<std::string>();
 | 
						|
 | 
						|
  // TODO: generate the HTML changelog from the JSON information.
 | 
						|
  nvi.changelog_html = GenerateChangelog(obj["changelog"].get<picojson::array>());
 | 
						|
 | 
						|
  OnUpdateAvailable(nvi);
 | 
						|
}
 | 
						|
 | 
						|
void AutoUpdateChecker::TriggerUpdate(const AutoUpdateChecker::NewVersionInformation& info,
 | 
						|
                                      AutoUpdateChecker::RestartMode restart_mode)
 | 
						|
{
 | 
						|
#ifdef _WIN32
 | 
						|
  std::map<std::string, std::string> updater_flags;
 | 
						|
  updater_flags["this-manifest-url"] = info.this_manifest_url;
 | 
						|
  updater_flags["next-manifest-url"] = info.next_manifest_url;
 | 
						|
  updater_flags["content-store-url"] = info.content_store_url;
 | 
						|
  updater_flags["parent-pid"] = std::to_string(GetCurrentProcessId());
 | 
						|
  updater_flags["install-base-path"] = File::GetExeDirectory();
 | 
						|
  updater_flags["log-file"] = File::GetExeDirectory() + DIR_SEP + UPDATER_LOG_FILE;
 | 
						|
 | 
						|
  if (restart_mode == RestartMode::RESTART_AFTER_UPDATE)
 | 
						|
    updater_flags["binary-to-restart"] = File::GetExePath();
 | 
						|
 | 
						|
  // Copy the updater so it can update itself if needed.
 | 
						|
  std::string updater_path = File::GetExeDirectory() + DIR_SEP + UPDATER_FILENAME;
 | 
						|
  std::string reloc_updater_path = File::GetExeDirectory() + DIR_SEP + UPDATER_RELOC_FILENAME;
 | 
						|
  File::Copy(updater_path, reloc_updater_path);
 | 
						|
 | 
						|
  // Run the updater!
 | 
						|
  std::wstring command_line = MakeUpdaterCommandLine(updater_flags);
 | 
						|
  STARTUPINFO sinfo = {sizeof(info)};
 | 
						|
  sinfo.dwFlags = STARTF_FORCEOFFFEEDBACK;  // No hourglass cursor after starting the process.
 | 
						|
  PROCESS_INFORMATION pinfo;
 | 
						|
  INFO_LOG(COMMON, "Updater command line: %s", UTF16ToUTF8(command_line).c_str());
 | 
						|
  if (!CreateProcessW(UTF8ToUTF16(reloc_updater_path).c_str(),
 | 
						|
                      const_cast<wchar_t*>(command_line.c_str()), nullptr, nullptr, FALSE, 0,
 | 
						|
                      nullptr, nullptr, &sinfo, &pinfo))
 | 
						|
  {
 | 
						|
    ERROR_LOG(COMMON, "Could not start updater process: error=%d", GetLastError());
 | 
						|
  }
 | 
						|
#endif
 | 
						|
}
 |