// Copyright 2018 Dolphin Emulator Project // Licensed under GPLv2+ // Refer to the license.txt file included. #include "UICommon/AutoUpdate.h" #include #include #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 #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& 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()) continue; picojson::object ver_obj = ver.get(); if (ver_obj["changelog_html"].is()) { if (!changelog.empty()) changelog += "
"; // Vertical spacing. // Try to link to the PR if we have this info. Otherwise just show shortrev. if (ver_obj["pr_url"].is()) { changelog += "() + "\">" + ver_obj["shortrev"].get() + ""; } else { changelog += ver_obj["shortrev"].get(); } changelog += " by () + "\">" + ver_obj["author"].get() + " — " + ver_obj["short_descr"].get(); } else { if (!changelog.empty()) changelog += "
"; changelog += "Dolphin " + ver_obj["shortrev"].get() + ""; changelog += "

" + ver_obj["changelog_html"].get() + "

"; } } 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(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(); if (obj["status"].get() != "outdated") { INFO_LOG(COMMON, "Auto-update status: we are up to date."); return; } NewVersionInformation nvi; nvi.this_manifest_url = obj["old"].get()["manifest"].get(); nvi.next_manifest_url = obj["new"].get()["manifest"].get(); nvi.content_store_url = obj["content-store"].get(); nvi.new_shortrev = obj["new"].get()["name"].get(); nvi.new_hash = obj["new"].get()["hash"].get(); // TODO: generate the HTML changelog from the JSON information. nvi.changelog_html = GenerateChangelog(obj["changelog"].get()); OnUpdateAvailable(nvi); } void AutoUpdateChecker::TriggerUpdate(const AutoUpdateChecker::NewVersionInformation& info, AutoUpdateChecker::RestartMode restart_mode) { #ifdef _WIN32 std::map 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(command_line.c_str()), nullptr, nullptr, FALSE, 0, nullptr, nullptr, &sinfo, &pinfo)) { ERROR_LOG(COMMON, "Could not start updater process: error=%d", GetLastError()); } #endif }