blob: db14fb6b74dfe181ee2eac80520058c1f285e00c [file] [log] [blame]
// Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <string>
#include <gflags/gflags.h>
#include <glib.h>
#include "update_engine/marshal.glibmarshal.h"
#include "update_engine/dbus_constants.h"
#include "update_engine/subprocess.h"
#include "update_engine/utils.h"
extern "C" {
#include "update_engine/update_engine.dbusclient.h"
}
using chromeos_update_engine::kUpdateEngineServiceName;
using chromeos_update_engine::kUpdateEngineServicePath;
using chromeos_update_engine::kUpdateEngineServiceInterface;
using chromeos_update_engine::utils::GetAndFreeGError;
using std::string;
DEFINE_string(app_version, "", "Force the current app version.");
DEFINE_bool(check_for_update, false, "Initiate check for updates.");
DEFINE_string(omaha_url, "", "The URL of the Omaha update server.");
DEFINE_bool(reboot, false, "Initiate a reboot if needed.");
DEFINE_bool(show_track, false, "Show the update track.");
DEFINE_bool(status, false, "Print the status to stdout.");
DEFINE_string(track, "", "Permanently change the update track.");
DEFINE_bool(update, false, "Forces an update and waits for its completion. "
"Exit status is 0 if the update succeeded, and 1 otherwise.");
DEFINE_bool(watch_for_updates, false,
"Listen for status updates and print them to the screen.");
namespace {
bool GetProxy(DBusGProxy** out_proxy) {
DBusGConnection* bus;
DBusGProxy* proxy = NULL;
GError* error = NULL;
const int kTries = 4;
const int kRetrySeconds = 10;
bus = dbus_g_bus_get(DBUS_BUS_SYSTEM, &error);
LOG_IF(FATAL, !bus) << "Failed to get bus: " << GetAndFreeGError(&error);
for (int i = 0; !proxy && i < kTries; ++i) {
if (i > 0) {
LOG(INFO) << "Retrying to get dbus proxy. Try "
<< (i + 1) << "/" << kTries;
sleep(kRetrySeconds);
}
proxy = dbus_g_proxy_new_for_name_owner(bus,
kUpdateEngineServiceName,
kUpdateEngineServicePath,
kUpdateEngineServiceInterface,
&error);
LOG_IF(WARNING, !proxy) << "Error getting dbus proxy for "
<< kUpdateEngineServiceName << ": "
<< GetAndFreeGError(&error);
}
LOG_IF(FATAL, !proxy) << "Giving up -- unable to get dbus proxy for "
<< kUpdateEngineServiceName;
*out_proxy = proxy;
return true;
}
static void StatusUpdateSignalHandler(DBusGProxy* proxy,
int64_t last_checked_time,
double progress,
gchar* current_operation,
gchar* new_version,
int64_t new_size,
void* user_data) {
LOG(INFO) << "Got status update:";
LOG(INFO) << " last_checked_time: " << last_checked_time;
LOG(INFO) << " progress: " << progress;
LOG(INFO) << " current_operation: " << current_operation;
LOG(INFO) << " new_version: " << new_version;
LOG(INFO) << " new_size: " << new_size;
}
// If |op| is non-NULL, sets it to the current operation string or an
// empty string if unable to obtain the current status.
bool GetStatus(string* op) {
DBusGProxy* proxy;
GError* error = NULL;
CHECK(GetProxy(&proxy));
gint64 last_checked_time = 0;
gdouble progress = 0.0;
char* current_op = NULL;
char* new_version = NULL;
gint64 new_size = 0;
gboolean rc = org_chromium_UpdateEngineInterface_get_status(
proxy,
&last_checked_time,
&progress,
&current_op,
&new_version,
&new_size,
&error);
if (rc == FALSE) {
LOG(INFO) << "Error getting status: " << GetAndFreeGError(&error);
}
printf("LAST_CHECKED_TIME=%" PRIi64 "\nPROGRESS=%f\nCURRENT_OP=%s\n"
"NEW_VERSION=%s\nNEW_SIZE=%" PRIi64 "\n",
last_checked_time,
progress,
current_op,
new_version,
new_size);
if (op) {
*op = current_op ? current_op : "";
}
return true;
}
// Should never return.
void WatchForUpdates() {
DBusGProxy* proxy;
CHECK(GetProxy(&proxy));
// Register marshaller
dbus_g_object_register_marshaller(
update_engine_VOID__INT64_DOUBLE_STRING_STRING_INT64,
G_TYPE_NONE,
G_TYPE_INT64,
G_TYPE_DOUBLE,
G_TYPE_STRING,
G_TYPE_STRING,
G_TYPE_INT64,
G_TYPE_INVALID);
static const char kStatusUpdate[] = "StatusUpdate";
dbus_g_proxy_add_signal(proxy,
kStatusUpdate,
G_TYPE_INT64,
G_TYPE_DOUBLE,
G_TYPE_STRING,
G_TYPE_STRING,
G_TYPE_INT64,
G_TYPE_INVALID);
GMainLoop* loop = g_main_loop_new (NULL, TRUE);
dbus_g_proxy_connect_signal(proxy,
kStatusUpdate,
G_CALLBACK(StatusUpdateSignalHandler),
NULL,
NULL);
g_main_loop_run(loop);
g_main_loop_unref(loop);
}
bool CheckForUpdates(const string& app_version, const string& omaha_url) {
DBusGProxy* proxy;
GError* error = NULL;
CHECK(GetProxy(&proxy));
gboolean rc =
org_chromium_UpdateEngineInterface_attempt_update(proxy,
app_version.c_str(),
omaha_url.c_str(),
&error);
CHECK_EQ(rc, TRUE) << "Error checking for update: "
<< GetAndFreeGError(&error);
return true;
}
bool RebootIfNeeded() {
DBusGProxy* proxy;
GError* error = NULL;
CHECK(GetProxy(&proxy));
gboolean rc =
org_chromium_UpdateEngineInterface_reboot_if_needed(proxy, &error);
// Reboot error code doesn't necessarily mean that a reboot
// failed. For example, D-Bus may be shutdown before we receive the
// result.
LOG_IF(INFO, !rc) << "Reboot error message: " << GetAndFreeGError(&error);
return true;
}
void SetTrack(const string& track) {
DBusGProxy* proxy;
GError* error = NULL;
CHECK(GetProxy(&proxy));
gboolean rc =
org_chromium_UpdateEngineInterface_set_track(proxy,
track.c_str(),
&error);
CHECK_EQ(rc, true) << "Error setting the track: "
<< GetAndFreeGError(&error);
LOG(INFO) << "Track permanently set to: " << track;
}
string GetTrack() {
DBusGProxy* proxy;
GError* error = NULL;
CHECK(GetProxy(&proxy));
char* track = NULL;
gboolean rc =
org_chromium_UpdateEngineInterface_get_track(proxy,
&track,
&error);
CHECK_EQ(rc, true) << "Error getting the track: " << GetAndFreeGError(&error);
string output = track;
g_free(track);
return output;
}
static gboolean CompleteUpdateSource(gpointer data) {
string current_op;
if (!GetStatus(&current_op) || current_op == "UPDATE_STATUS_IDLE") {
LOG(ERROR) << "Update failed.";
exit(1);
}
if (current_op == "UPDATE_STATUS_UPDATED_NEED_REBOOT") {
LOG(INFO) << "Update succeeded -- reboot needed.";
exit(0);
}
return TRUE;
}
// This is similar to watching for updates but rather than registering
// a signal watch, activelly poll the daemon just in case it stops
// sending notifications.
void CompleteUpdate() {
GMainLoop* loop = g_main_loop_new (NULL, TRUE);
g_timeout_add_seconds(5, CompleteUpdateSource, NULL);
g_main_loop_run(loop);
g_main_loop_unref(loop);
}
} // namespace {}
int main(int argc, char** argv) {
// Boilerplate init commands.
g_type_init();
g_thread_init(NULL);
dbus_g_thread_init();
chromeos_update_engine::Subprocess::Init();
google::ParseCommandLineFlags(&argc, &argv, true);
if (FLAGS_status) {
LOG(INFO) << "Querying Update Engine status...";
if (!GetStatus(NULL)) {
LOG(FATAL) << "GetStatus failed.";
return 1;
}
return 0;
}
// First, update the track if requested.
if (!FLAGS_track.empty()) {
SetTrack(FLAGS_track);
}
// Show the track if requested.
if (FLAGS_show_track) {
LOG(INFO) << "Track: " << GetTrack();
}
// Initiate an update check, if necessary.
if (FLAGS_check_for_update ||
FLAGS_update ||
!FLAGS_app_version.empty() ||
!FLAGS_omaha_url.empty()) {
LOG_IF(WARNING, FLAGS_reboot) << "-reboot flag ignored.";
string app_version = FLAGS_app_version;
if (FLAGS_update && app_version.empty()) {
app_version = "ForcedUpdate";
LOG(INFO) << "Forcing an update by setting app_version to ForcedUpdate.";
}
LOG(INFO) << "Initiating update check and install.";
CHECK(CheckForUpdates(app_version, FLAGS_omaha_url))
<< "Update check/initiate update failed.";
// Wait for an update to complete.
if (FLAGS_update) {
LOG(INFO) << "Waiting for update to complete.";
CompleteUpdate(); // Should never return.
return 1;
}
return 0;
}
// Start watching for updates.
if (FLAGS_watch_for_updates) {
LOG_IF(WARNING, FLAGS_reboot) << "-reboot flag ignored.";
LOG(INFO) << "Watching for status updates.";
WatchForUpdates(); // Should never return.
return 1;
}
if (FLAGS_reboot) {
LOG(INFO) << "Requesting a reboot...";
CHECK(RebootIfNeeded());
return 0;
}
LOG(INFO) << "Done.";
return 0;
}