mirror of
https://git.proxmox.com/git/fwupd
synced 2025-05-23 21:26:58 +00:00

If somehow we forget to add a case section, return with an error as the code suggests, rather than always returning TRUE. Found with Coverity.
2793 lines
84 KiB
C
2793 lines
84 KiB
C
/*
|
|
* Copyright (C) 2015-2018 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
#define G_LOG_DOMAIN "FuMain"
|
|
|
|
#include "config.h"
|
|
|
|
#include <fwupd.h>
|
|
#include <xmlb.h>
|
|
#include <fcntl.h>
|
|
#include <glib/gstdio.h>
|
|
#include <gio/gio.h>
|
|
#include <gio/gunixfdlist.h>
|
|
#include <glib/gi18n.h>
|
|
#ifdef HAVE_GIO_UNIX
|
|
#include <glib-unix.h>
|
|
#endif
|
|
#include <json-glib/json-glib.h>
|
|
#include <locale.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
|
|
#include "fu-history.h"
|
|
#include "fu-plugin-private.h"
|
|
#include "fu-progressbar.h"
|
|
#include "fu-util-common.h"
|
|
#include "fwupd-common-private.h"
|
|
|
|
#ifdef HAVE_SYSTEMD
|
|
#include "fu-systemd.h"
|
|
#endif
|
|
|
|
/* custom return code */
|
|
#define EXIT_NOTHING_TO_DO 2
|
|
|
|
typedef enum {
|
|
FU_UTIL_HISTORY_DO_NOTHING,
|
|
FU_UTIL_HISTORY_NEVER,
|
|
FU_UTIL_HISTORY_PROMPT,
|
|
FU_UTIL_HISTORY_AUTOMATIC,
|
|
FU_UTIL_HISTORY_LAST,
|
|
} FuUtilHistoryAction;
|
|
|
|
typedef enum {
|
|
FU_UTIL_OPERATION_UNKNOWN,
|
|
FU_UTIL_OPERATION_UPDATE,
|
|
FU_UTIL_OPERATION_DOWNGRADE,
|
|
FU_UTIL_OPERATION_INSTALL,
|
|
FU_UTIL_OPERATION_LAST
|
|
} FuUtilOperation;
|
|
|
|
struct FuUtilPrivate {
|
|
GCancellable *cancellable;
|
|
GMainLoop *loop;
|
|
GOptionContext *context;
|
|
SoupSession *soup_session;
|
|
FwupdInstallFlags flags;
|
|
FwupdClient *client;
|
|
FuProgressbar *progressbar;
|
|
gboolean no_metadata_check;
|
|
gboolean no_reboot_check;
|
|
gboolean no_unreported_check;
|
|
gboolean no_safety_check;
|
|
gboolean assume_yes;
|
|
gboolean sign;
|
|
gboolean show_all_devices;
|
|
gboolean disable_ssl_strict;
|
|
/* only valid in update and downgrade */
|
|
FuUtilOperation current_operation;
|
|
FwupdDevice *current_device;
|
|
gchar *current_message;
|
|
FwupdDeviceFlags completion_flags;
|
|
FwupdDeviceFlags filter_include;
|
|
FwupdDeviceFlags filter_exclude;
|
|
};
|
|
|
|
static gboolean fu_util_report_history (FuUtilPrivate *priv, gchar **values, GError **error);
|
|
static gboolean fu_util_download_file (FuUtilPrivate *priv,
|
|
SoupURI *uri,
|
|
const gchar *fn,
|
|
const gchar *checksum_expected,
|
|
GError **error);
|
|
|
|
static void
|
|
fu_util_client_notify_cb (GObject *object,
|
|
GParamSpec *pspec,
|
|
FuUtilPrivate *priv)
|
|
{
|
|
fu_progressbar_update (priv->progressbar,
|
|
fwupd_client_get_status (priv->client),
|
|
fwupd_client_get_percentage (priv->client));
|
|
}
|
|
|
|
static void
|
|
fu_util_update_device_changed_cb (FwupdClient *client,
|
|
FwupdDevice *device,
|
|
FuUtilPrivate *priv)
|
|
{
|
|
g_autofree gchar *str = NULL;
|
|
|
|
/* allowed to set whenever the device has changed */
|
|
if (fwupd_device_has_flag (device, FWUPD_DEVICE_FLAG_NEEDS_SHUTDOWN))
|
|
priv->completion_flags |= FWUPD_DEVICE_FLAG_NEEDS_SHUTDOWN;
|
|
if (fwupd_device_has_flag (device, FWUPD_DEVICE_FLAG_NEEDS_REBOOT))
|
|
priv->completion_flags |= FWUPD_DEVICE_FLAG_NEEDS_REBOOT;
|
|
|
|
/* same as last time, so ignore */
|
|
if (priv->current_device != NULL &&
|
|
fwupd_device_compare (priv->current_device, device) == 0)
|
|
return;
|
|
|
|
/* show message in progressbar */
|
|
if (priv->current_operation == FU_UTIL_OPERATION_UPDATE) {
|
|
/* TRANSLATORS: %1 is a device name */
|
|
str = g_strdup_printf (_("Updating %s…"),
|
|
fwupd_device_get_name (device));
|
|
fu_progressbar_set_title (priv->progressbar, str);
|
|
} else if (priv->current_operation == FU_UTIL_OPERATION_DOWNGRADE) {
|
|
/* TRANSLATORS: %1 is a device name */
|
|
str = g_strdup_printf (_("Downgrading %s…"),
|
|
fwupd_device_get_name (device));
|
|
fu_progressbar_set_title (priv->progressbar, str);
|
|
} else if (priv->current_operation == FU_UTIL_OPERATION_INSTALL) {
|
|
/* TRANSLATORS: %1 is a device name */
|
|
str = g_strdup_printf (_("Installing on %s…"),
|
|
fwupd_device_get_name (device));
|
|
fu_progressbar_set_title (priv->progressbar, str);
|
|
} else {
|
|
g_warning ("no FuUtilOperation set");
|
|
}
|
|
g_set_object (&priv->current_device, device);
|
|
|
|
if (priv->current_message == NULL) {
|
|
const gchar *tmp = fwupd_device_get_update_message (priv->current_device);
|
|
if (tmp != NULL)
|
|
priv->current_message = g_strdup (tmp);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_filter_device (FuUtilPrivate *priv, FwupdDevice *dev)
|
|
{
|
|
for (guint i = 0; i < 64; i++) {
|
|
FwupdDeviceFlags flag = 1LLU << i;
|
|
if (priv->filter_include & flag) {
|
|
if (!fwupd_device_has_flag (dev, flag))
|
|
return FALSE;
|
|
}
|
|
if (priv->filter_exclude & flag) {
|
|
if (fwupd_device_has_flag (dev, flag))
|
|
return FALSE;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static FwupdDevice *
|
|
fu_util_prompt_for_device (FuUtilPrivate *priv, GError **error)
|
|
{
|
|
FwupdDevice *dev;
|
|
guint idx;
|
|
g_autoptr(GPtrArray) devices = NULL;
|
|
g_autoptr(GPtrArray) devices_filtered = NULL;
|
|
|
|
/* get devices from daemon */
|
|
devices = fwupd_client_get_devices (priv->client, NULL, error);
|
|
if (devices == NULL)
|
|
return NULL;
|
|
|
|
/* filter results */
|
|
devices_filtered = g_ptr_array_new ();
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
dev = g_ptr_array_index (devices, i);
|
|
if (!fu_util_filter_device (priv, dev))
|
|
continue;
|
|
g_ptr_array_add (devices_filtered, dev);
|
|
}
|
|
|
|
/* nothing */
|
|
if (devices_filtered->len == 0) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"No supported devices");
|
|
return NULL;
|
|
}
|
|
|
|
/* exactly one */
|
|
if (devices_filtered->len == 1) {
|
|
dev = g_ptr_array_index (devices_filtered, 0);
|
|
/* TRANSLATORS: Device has been chosen by the daemon for the user */
|
|
g_print ("%s: %s\n", _("Selected device"), fwupd_device_get_name (dev));
|
|
return g_object_ref (dev);
|
|
}
|
|
|
|
/* TRANSLATORS: get interactive prompt */
|
|
g_print ("%s\n", _("Choose a device:"));
|
|
/* TRANSLATORS: this is to abort the interactive prompt */
|
|
g_print ("0.\t%s\n", _("Cancel"));
|
|
for (guint i = 0; i < devices_filtered->len; i++) {
|
|
dev = g_ptr_array_index (devices_filtered, i);
|
|
g_print ("%u.\t%s (%s)\n",
|
|
i + 1,
|
|
fwupd_device_get_id (dev),
|
|
fwupd_device_get_name (dev));
|
|
}
|
|
idx = fu_util_prompt_for_number (devices_filtered->len);
|
|
if (idx == 0) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"Request canceled");
|
|
return NULL;
|
|
}
|
|
dev = g_ptr_array_index (devices_filtered, idx - 1);
|
|
return g_object_ref (dev);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_maybe_enable_automatic (FuUtilPrivate *priv, GPtrArray *remotes, GError **error)
|
|
{
|
|
guint idx;
|
|
|
|
/* TRANSLATORS: Display a message asking every time an update is performed */
|
|
g_print ("%d.\t%s\n",
|
|
FU_UTIL_HISTORY_DO_NOTHING,
|
|
ngettext ("Do not upload report at this time, but prompt again for future updates",
|
|
"Do not upload reports at this time, but prompt again for future updates",
|
|
remotes->len));
|
|
/* TRANSLATORS: Display a message asking every time an update is performed */
|
|
g_print ("%d.\t%s\n",
|
|
FU_UTIL_HISTORY_NEVER,
|
|
ngettext ("Do not upload report, and never ask to upload reports for future updates",
|
|
"Do not upload reports, and never ask to upload reports for future updates",
|
|
remotes->len));
|
|
/* TRANSLATORS: Display a message asking every time an update is performed */
|
|
g_print ("%d.\t%s\n",
|
|
FU_UTIL_HISTORY_PROMPT,
|
|
ngettext ("Upload report just this one time, but prompt again for future updates",
|
|
"Upload reports just this one time, but prompt again for future updates",
|
|
remotes->len));
|
|
/* TRANSLATORS: Display a message asking every time an update is performed */
|
|
g_print ("%d.\t%s\n",
|
|
FU_UTIL_HISTORY_AUTOMATIC,
|
|
ngettext ("Upload report this time and automatically upload reports after completing future updates",
|
|
"Upload reports this time and automatically upload reports after completing future updates",
|
|
remotes->len));
|
|
idx = fu_util_prompt_for_number (FU_UTIL_HISTORY_LAST - 1);
|
|
|
|
switch (idx) {
|
|
case FU_UTIL_HISTORY_NEVER:
|
|
for (guint i = 0; i < remotes->len; i++) {
|
|
FwupdRemote *remote = g_ptr_array_index (remotes, i);
|
|
const gchar *remote_id = fwupd_remote_get_id (remote);
|
|
if (fwupd_remote_get_report_uri (remote) == NULL)
|
|
continue;
|
|
if (!fwupd_client_modify_remote (priv->client,
|
|
remote_id, "ReportURI", "",
|
|
NULL, error))
|
|
return FALSE;
|
|
}
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"Reporting disabled");
|
|
return FALSE;
|
|
case FU_UTIL_HISTORY_DO_NOTHING:
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"Request canceled");
|
|
return FALSE;
|
|
case FU_UTIL_HISTORY_AUTOMATIC:
|
|
for (guint i = 0; i < remotes->len; i++) {
|
|
FwupdRemote *remote = g_ptr_array_index (remotes, i);
|
|
const gchar *remote_id = fwupd_remote_get_id (remote);
|
|
if (fwupd_remote_get_report_uri (remote) == NULL)
|
|
continue;
|
|
if (fwupd_remote_get_automatic_reports (remote))
|
|
continue;
|
|
if (!fwupd_client_modify_remote (priv->client,
|
|
remote_id, "AutomaticReports", "true",
|
|
NULL, error))
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"invalid option selected");
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_perhaps_show_unreported (FuUtilPrivate *priv, GError **error)
|
|
{
|
|
g_autoptr(GError) error_local = NULL;
|
|
g_autoptr(GPtrArray) devices = NULL;
|
|
g_autoptr(GPtrArray) devices_failed = g_ptr_array_new ();
|
|
g_autoptr(GPtrArray) devices_success = g_ptr_array_new ();
|
|
g_autoptr(GPtrArray) remotes = NULL;
|
|
g_autoptr(GHashTable) remote_id_uri_map = NULL;
|
|
gboolean all_automatic = FALSE;
|
|
|
|
/* we don't want to ask anything */
|
|
if (priv->no_unreported_check) {
|
|
g_debug ("skipping unreported check");
|
|
return TRUE;
|
|
}
|
|
|
|
/* get all devices from the history database */
|
|
devices = fwupd_client_get_history (priv->client, NULL, &error_local);
|
|
if (devices == NULL) {
|
|
if (g_error_matches (error_local, FWUPD_ERROR, FWUPD_ERROR_NOTHING_TO_DO))
|
|
return TRUE;
|
|
g_propagate_error (error, g_steal_pointer (&error_local));
|
|
return FALSE;
|
|
}
|
|
|
|
/* create a map of RemoteID to RemoteURI */
|
|
remotes = fwupd_client_get_remotes (priv->client, NULL, error);
|
|
if (remotes == NULL)
|
|
return FALSE;
|
|
remote_id_uri_map = g_hash_table_new (g_str_hash, g_str_equal);
|
|
for (guint i = 0; i < remotes->len; i++) {
|
|
FwupdRemote *remote = g_ptr_array_index (remotes, i);
|
|
gboolean remote_automatic;
|
|
if (fwupd_remote_get_id (remote) == NULL)
|
|
continue;
|
|
if (fwupd_remote_get_report_uri (remote) == NULL)
|
|
continue;
|
|
g_debug ("adding %s for %s",
|
|
fwupd_remote_get_report_uri (remote),
|
|
fwupd_remote_get_id (remote));
|
|
g_hash_table_insert (remote_id_uri_map,
|
|
(gpointer) fwupd_remote_get_id (remote),
|
|
(gpointer) fwupd_remote_get_report_uri (remote));
|
|
remote_automatic = fwupd_remote_get_automatic_reports (remote);
|
|
g_debug ("%s is %d", fwupd_remote_get_title (remote), remote_automatic);
|
|
if (remote_automatic && !all_automatic)
|
|
all_automatic = TRUE;
|
|
if (!remote_automatic && all_automatic) {
|
|
all_automatic = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* check that they can be reported */
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
FwupdDevice *dev = g_ptr_array_index (devices, i);
|
|
FwupdRelease *rel = fwupd_device_get_release_default (dev);
|
|
const gchar *remote_id;
|
|
const gchar *remote_uri;
|
|
|
|
if (!fu_util_filter_device (priv, dev))
|
|
continue;
|
|
if (fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_REPORTED))
|
|
continue;
|
|
if (!fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_SUPPORTED))
|
|
continue;
|
|
|
|
/* find the RemoteURI to use for the device */
|
|
remote_id = fwupd_release_get_remote_id (rel);
|
|
if (remote_id == NULL) {
|
|
g_debug ("%s has no RemoteID", fwupd_device_get_id (dev));
|
|
continue;
|
|
}
|
|
remote_uri = g_hash_table_lookup (remote_id_uri_map, remote_id);
|
|
if (remote_uri == NULL) {
|
|
g_debug ("%s has no RemoteURI", remote_id);
|
|
continue;
|
|
}
|
|
|
|
/* only send success and failure */
|
|
if (fwupd_device_get_update_state (dev) == FWUPD_UPDATE_STATE_FAILED) {
|
|
g_ptr_array_add (devices_failed, dev);
|
|
} else if (fwupd_device_get_update_state (dev) == FWUPD_UPDATE_STATE_SUCCESS) {
|
|
g_ptr_array_add (devices_success, dev);
|
|
} else {
|
|
g_debug ("ignoring %s with UpdateState %s",
|
|
fwupd_device_get_id (dev),
|
|
fwupd_update_state_to_string (fwupd_device_get_update_state (dev)));
|
|
}
|
|
}
|
|
|
|
/* nothing to do */
|
|
if (devices_failed->len == 0 && devices_success->len == 0) {
|
|
g_debug ("no unreported devices");
|
|
return TRUE;
|
|
}
|
|
|
|
g_debug ("All automatic: %d", all_automatic);
|
|
/* show the success and failures */
|
|
if (!priv->assume_yes && !all_automatic) {
|
|
|
|
/* delimit */
|
|
g_print ("________________________________________________\n");
|
|
|
|
/* failures */
|
|
if (devices_failed->len > 0) {
|
|
/* TRANSLATORS: a list of failed updates */
|
|
g_print ("\n%s\n\n", _("Devices that were not updated correctly:"));
|
|
for (guint i = 0; i < devices_failed->len; i++) {
|
|
FwupdDevice *dev = g_ptr_array_index (devices_failed, i);
|
|
FwupdRelease *rel = fwupd_device_get_release_default (dev);
|
|
g_print (" • %s (%s → %s)\n",
|
|
fwupd_device_get_name (dev),
|
|
fwupd_device_get_version (dev),
|
|
fwupd_release_get_version (rel));
|
|
}
|
|
}
|
|
|
|
/* success */
|
|
if (devices_success->len > 0) {
|
|
/* TRANSLATORS: a list of successful updates */
|
|
g_print ("\n%s\n\n", _("Devices that have been updated successfully:"));
|
|
for (guint i = 0; i < devices_success->len; i++) {
|
|
FwupdDevice *dev = g_ptr_array_index (devices_success, i);
|
|
FwupdRelease *rel = fwupd_device_get_release_default (dev);
|
|
g_print (" • %s (%s → %s)\n",
|
|
fwupd_device_get_name (dev),
|
|
fwupd_device_get_version (dev),
|
|
fwupd_release_get_version (rel));
|
|
}
|
|
}
|
|
|
|
/* ask for permission */
|
|
g_print ("\n%s\n%s (%s):\n",
|
|
/* TRANSLATORS: explain why we want to upload */
|
|
_("Uploading firmware reports helps hardware vendors"
|
|
" to quickly identify failing and successful updates"
|
|
" on real devices."),
|
|
/* TRANSLATORS: ask the user to upload */
|
|
_("Upload report now?"),
|
|
/* TRANSLATORS: metadata is downloaded from the Internet */
|
|
_("Requires internet connection"));
|
|
if (!fu_util_maybe_enable_automatic (priv, remotes, error))
|
|
return FALSE;
|
|
}
|
|
|
|
/* success */
|
|
return fu_util_report_history (priv, NULL, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_modify_remote_warning (FuUtilPrivate *priv, FwupdRemote *remote, GError **error)
|
|
{
|
|
const gchar *warning_markup = NULL;
|
|
g_autofree gchar *warning_plain = NULL;
|
|
|
|
/* get formatted text */
|
|
warning_markup = fwupd_remote_get_agreement (remote);
|
|
if (warning_markup == NULL)
|
|
return TRUE;
|
|
warning_plain = fu_util_convert_description (warning_markup, error);
|
|
if (warning_plain == NULL)
|
|
return FALSE;
|
|
|
|
/* show and ask user to confirm */
|
|
fu_util_warning_box (warning_plain, 80);
|
|
if (!priv->assume_yes) {
|
|
/* ask for permission */
|
|
g_print ("\n%s [Y|n]: ",
|
|
/* TRANSLATORS: should the remote still be enabled */
|
|
_("Agree and enable the remote?"));
|
|
if (!fu_util_prompt_for_boolean (TRUE)) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"Declined agreement");
|
|
return FALSE;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_modify_remote (FuUtilPrivate *priv,
|
|
const gchar *remote_id,
|
|
const gchar *key,
|
|
const gchar *value,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FwupdRemote) remote = NULL;
|
|
|
|
/* ensure the remote exists */
|
|
remote = fwupd_client_get_remote_by_id (priv->client, remote_id, NULL, error);
|
|
if (remote == NULL)
|
|
return FALSE;
|
|
|
|
/* show some kind of warning when enabling download-type remotes */
|
|
if (g_strcmp0 (key, "Enabled") == 0 && g_strcmp0 (value, "true") == 0) {
|
|
if (!fu_util_modify_remote_warning (priv, remote, error))
|
|
return FALSE;
|
|
}
|
|
return fwupd_client_modify_remote (priv->client,
|
|
remote_id, key, value,
|
|
NULL, error);
|
|
}
|
|
|
|
static void
|
|
fu_util_build_device_tree (FuUtilPrivate *priv, GNode *root, GPtrArray *devs, FwupdDevice *dev)
|
|
{
|
|
for (guint i = 0; i < devs->len; i++) {
|
|
FwupdDevice *dev_tmp = g_ptr_array_index (devs, i);
|
|
if (!fu_util_filter_device (priv, dev_tmp))
|
|
continue;
|
|
if (!priv->show_all_devices &&
|
|
!fu_util_is_interesting_device (dev_tmp))
|
|
continue;
|
|
if (fwupd_device_get_parent (dev_tmp) == dev) {
|
|
FwupdRelease *rel = fwupd_device_get_release_default (dev_tmp);
|
|
GNode *child = g_node_append_data (root, dev_tmp);
|
|
if (rel != NULL)
|
|
g_node_append_data (child, rel);
|
|
fu_util_build_device_tree (priv, child, devs, dev_tmp);
|
|
}
|
|
}
|
|
}
|
|
|
|
static gchar *
|
|
fu_util_get_tree_title (FuUtilPrivate *priv)
|
|
{
|
|
return g_strdup (fwupd_client_get_host_product (priv->client));
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_get_devices (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(GNode) root = g_node_new (NULL);
|
|
g_autoptr(GPtrArray) devs = NULL;
|
|
g_autofree gchar *title = fu_util_get_tree_title (priv);
|
|
|
|
/* get results from daemon */
|
|
devs = fwupd_client_get_devices (priv->client, NULL, error);
|
|
if (devs == NULL)
|
|
return FALSE;
|
|
|
|
/* print */
|
|
if (devs->len == 0) {
|
|
/* TRANSLATORS: nothing attached that can be upgraded */
|
|
g_print ("%s\n", _("No hardware detected with firmware update capability"));
|
|
return TRUE;
|
|
}
|
|
fu_util_build_device_tree (priv, root, devs, NULL);
|
|
fu_util_print_tree (root, title);
|
|
|
|
/* nag? */
|
|
if (!fu_util_perhaps_show_unreported (priv, error))
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gchar *
|
|
fu_util_download_if_required (FuUtilPrivate *priv, const gchar *perhapsfn, GError **error)
|
|
{
|
|
g_autofree gchar *filename = NULL;
|
|
g_autoptr(SoupURI) uri = NULL;
|
|
|
|
/* a local file */
|
|
if (g_file_test (perhapsfn, G_FILE_TEST_EXISTS))
|
|
return g_strdup (perhapsfn);
|
|
uri = soup_uri_new (perhapsfn);
|
|
if (uri == NULL)
|
|
return g_strdup (perhapsfn);
|
|
|
|
/* download the firmware to a cachedir */
|
|
filename = fu_util_get_user_cache_path (perhapsfn);
|
|
if (!fu_common_mkdir_parent (filename, error))
|
|
return NULL;
|
|
if (!fu_util_download_file (priv, uri, filename, NULL, error))
|
|
return NULL;
|
|
return g_steal_pointer (&filename);
|
|
}
|
|
|
|
static void
|
|
fu_util_display_current_message (FuUtilPrivate *priv)
|
|
{
|
|
if (priv->current_message == NULL) {
|
|
/* TRANSLATORS: success message */
|
|
g_print ("%s\n", _("Successfully installed firmware"));
|
|
return;
|
|
}
|
|
/* TRANSLATORS: success message */
|
|
g_print ("%s: %s\n", _("Successfully installed firmware"), priv->current_message);
|
|
g_clear_pointer (&priv->current_message, g_free);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_install (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
const gchar *id;
|
|
g_autofree gchar *filename = NULL;
|
|
|
|
/* handle both forms */
|
|
if (g_strv_length (values) == 1) {
|
|
id = FWUPD_DEVICE_ID_ANY;
|
|
} else if (g_strv_length (values) == 2) {
|
|
id = values[1];
|
|
} else {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments");
|
|
return FALSE;
|
|
}
|
|
|
|
priv->current_operation = FU_UTIL_OPERATION_INSTALL;
|
|
g_signal_connect (priv->client, "device-changed",
|
|
G_CALLBACK (fu_util_update_device_changed_cb), priv);
|
|
|
|
/* install with flags chosen by the user */
|
|
filename = fu_util_download_if_required (priv, values[0], error);
|
|
if (filename == NULL)
|
|
return FALSE;
|
|
|
|
if (!fwupd_client_install (priv->client, id, filename, priv->flags, NULL, error))
|
|
return FALSE;
|
|
|
|
fu_util_display_current_message (priv);
|
|
|
|
/* we don't want to ask anything */
|
|
if (priv->no_reboot_check) {
|
|
g_debug ("skipping reboot check");
|
|
return TRUE;
|
|
}
|
|
|
|
/* show reboot if needed */
|
|
return fu_util_prompt_complete (priv->completion_flags, TRUE, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_get_details (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(GPtrArray) array = NULL;
|
|
g_autoptr(GNode) root = g_node_new (NULL);
|
|
g_autofree gchar *title = fu_util_get_tree_title (priv);
|
|
|
|
/* check args */
|
|
if (g_strv_length (values) != 1) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments");
|
|
return FALSE;
|
|
}
|
|
array = fwupd_client_get_details (priv->client, values[0], NULL, error);
|
|
if (array == NULL)
|
|
return FALSE;
|
|
fu_util_build_device_tree (priv, root, array, NULL);
|
|
fu_util_print_tree (root, title);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_clear_history (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FuHistory) history = fu_history_new ();
|
|
return fu_history_remove_all (history, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_report_history_for_remote (FuUtilPrivate *priv,
|
|
const gchar *remote_id,
|
|
GPtrArray *devices,
|
|
GError **error)
|
|
{
|
|
JsonNode *json_root;
|
|
JsonObject *json_object;
|
|
const gchar *server_msg = NULL;
|
|
guint status_code;
|
|
const gchar *report_uri;
|
|
g_autofree gchar *data = NULL;
|
|
g_autofree gchar *sig = NULL;
|
|
g_autoptr(JsonParser) json_parser = NULL;
|
|
g_autoptr(SoupMessage) msg = NULL;
|
|
g_autoptr(FwupdRemote) remote = NULL;
|
|
|
|
/* convert to JSON */
|
|
data = fwupd_build_history_report_json (devices, error);
|
|
if (data == NULL)
|
|
return FALSE;
|
|
|
|
/* self sign data */
|
|
if (priv->sign) {
|
|
sig = fwupd_client_self_sign (priv->client, data,
|
|
FWUPD_SELF_SIGN_FLAG_ADD_TIMESTAMP,
|
|
priv->cancellable, error);
|
|
if (sig == NULL)
|
|
return FALSE;
|
|
}
|
|
|
|
remote = fwupd_client_get_remote_by_id (priv->client, remote_id,
|
|
NULL, error);
|
|
if (remote == NULL)
|
|
return FALSE;
|
|
report_uri = fwupd_remote_get_report_uri (remote);
|
|
|
|
/* ask for permission */
|
|
if (!priv->assume_yes && !fwupd_remote_get_automatic_reports (remote)) {
|
|
fu_util_print_data (_("Target"), report_uri);
|
|
fu_util_print_data (_("Payload"), data);
|
|
if (sig != NULL)
|
|
fu_util_print_data (_("Signature"), sig);
|
|
g_print ("%s [Y|n]: ", _("Proceed with upload?"));
|
|
if (!fu_util_prompt_for_boolean (TRUE)) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_PERMISSION_DENIED,
|
|
"User declined action");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* POST request */
|
|
if (sig != NULL) {
|
|
g_autoptr(SoupMultipart) mp = NULL;
|
|
mp = soup_multipart_new (SOUP_FORM_MIME_TYPE_MULTIPART);
|
|
soup_multipart_append_form_string (mp, "payload", data);
|
|
soup_multipart_append_form_string (mp, "signature", sig);
|
|
msg = soup_form_request_new_from_multipart (report_uri, mp);
|
|
} else {
|
|
msg = soup_message_new (SOUP_METHOD_POST, report_uri);
|
|
soup_message_set_request (msg, "application/json; charset=utf-8",
|
|
SOUP_MEMORY_COPY, data, strlen (data));
|
|
}
|
|
status_code = soup_session_send_message (priv->soup_session, msg);
|
|
g_debug ("server returned: %s", msg->response_body->data);
|
|
|
|
/* server returned nothing, and probably exploded in a ball of flames */
|
|
if (msg->response_body->length == 0) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_FILE,
|
|
"Failed to upload to %s: %s",
|
|
report_uri, soup_status_get_phrase (status_code));
|
|
return FALSE;
|
|
}
|
|
|
|
/* parse JSON reply */
|
|
json_parser = json_parser_new ();
|
|
if (!json_parser_load_from_data (json_parser,
|
|
msg->response_body->data,
|
|
msg->response_body->length,
|
|
error)) {
|
|
g_autofree gchar *str = g_strndup (msg->response_body->data,
|
|
msg->response_body->length);
|
|
g_prefix_error (error, "Failed to parse JSON response from '%s': ", str);
|
|
return FALSE;
|
|
}
|
|
json_root = json_parser_get_root (json_parser);
|
|
if (json_root == NULL) {
|
|
g_autofree gchar *str = g_strndup (msg->response_body->data,
|
|
msg->response_body->length);
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_PERMISSION_DENIED,
|
|
"JSON response was malformed: '%s'", str);
|
|
return FALSE;
|
|
}
|
|
json_object = json_node_get_object (json_root);
|
|
if (json_object == NULL) {
|
|
g_autofree gchar *str = g_strndup (msg->response_body->data,
|
|
msg->response_body->length);
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_PERMISSION_DENIED,
|
|
"JSON response object was malformed: '%s'", str);
|
|
return FALSE;
|
|
}
|
|
|
|
/* get any optional server message */
|
|
if (json_object_has_member (json_object, "msg"))
|
|
server_msg = json_object_get_string_member (json_object, "msg");
|
|
|
|
/* server reported failed */
|
|
if (!json_object_get_boolean_member (json_object, "success")) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_PERMISSION_DENIED,
|
|
"Server rejected report: %s",
|
|
server_msg != NULL ? server_msg : "unspecified");
|
|
return FALSE;
|
|
}
|
|
|
|
/* server wanted us to see the message */
|
|
if (server_msg != NULL) {
|
|
if (g_strstr_len (server_msg, -1, "known issue") != NULL &&
|
|
json_object_has_member (json_object, "uri")) {
|
|
g_print ("%s %s\n",
|
|
/* TRANSLATORS: the server sent the user a small message */
|
|
_("Update failure is a known issue, visit this URL for more information:"),
|
|
json_object_get_string_member (json_object, "uri"));
|
|
} else {
|
|
/* TRANSLATORS: the server sent the user a small message */
|
|
g_print ("%s %s\n", _("Upload message:"), server_msg);
|
|
}
|
|
}
|
|
|
|
/* fall back to HTTP status codes in case the server is offline */
|
|
if (!SOUP_STATUS_IS_SUCCESSFUL (status_code)) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_FILE,
|
|
"Failed to upload to %s: %s",
|
|
report_uri, soup_status_get_phrase (status_code));
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_report_history (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(GHashTable) remote_id_uri_map = NULL;
|
|
g_autoptr(GHashTable) report_map = NULL;
|
|
g_autoptr(GList) ids = NULL;
|
|
g_autoptr(GPtrArray) devices = NULL;
|
|
g_autoptr(GPtrArray) remotes = NULL;
|
|
g_autoptr(GString) str = g_string_new (NULL);
|
|
|
|
/* set up networking */
|
|
if (priv->soup_session == NULL) {
|
|
priv->soup_session = fu_util_setup_networking (error);
|
|
if (priv->soup_session == NULL)
|
|
return FALSE;
|
|
}
|
|
|
|
/* create a map of RemoteID to RemoteURI */
|
|
remotes = fwupd_client_get_remotes (priv->client, NULL, error);
|
|
if (remotes == NULL)
|
|
return FALSE;
|
|
remote_id_uri_map = g_hash_table_new (g_str_hash, g_str_equal);
|
|
for (guint i = 0; i < remotes->len; i++) {
|
|
FwupdRemote *remote = g_ptr_array_index (remotes, i);
|
|
if (fwupd_remote_get_id (remote) == NULL)
|
|
continue;
|
|
if (fwupd_remote_get_report_uri (remote) == NULL)
|
|
continue;
|
|
g_debug ("adding %s for %s",
|
|
fwupd_remote_get_report_uri (remote),
|
|
fwupd_remote_get_id (remote));
|
|
g_hash_table_insert (remote_id_uri_map,
|
|
(gpointer) fwupd_remote_get_id (remote),
|
|
(gpointer) fwupd_remote_get_report_uri (remote));
|
|
}
|
|
|
|
/* get all devices from the history database, then filter them,
|
|
* adding to a hash map of report-ids */
|
|
devices = fwupd_client_get_history (priv->client, NULL, error);
|
|
if (devices == NULL)
|
|
return FALSE;
|
|
report_map = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
g_free, (GDestroyNotify) g_ptr_array_unref);
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
FwupdDevice *dev = g_ptr_array_index (devices, i);
|
|
FwupdRelease *rel = fwupd_device_get_release_default (dev);
|
|
const gchar *remote_id;
|
|
const gchar *remote_uri;
|
|
GPtrArray *devices_tmp;
|
|
|
|
/* filter, if not forcing */
|
|
if (!fu_util_filter_device (priv, dev))
|
|
continue;
|
|
if ((priv->flags & FWUPD_INSTALL_FLAG_FORCE) == 0) {
|
|
if (fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_REPORTED))
|
|
continue;
|
|
if (!fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_SUPPORTED))
|
|
continue;
|
|
}
|
|
/* only send success and failure */
|
|
if (fwupd_device_get_update_state (dev) != FWUPD_UPDATE_STATE_FAILED &&
|
|
fwupd_device_get_update_state (dev) != FWUPD_UPDATE_STATE_SUCCESS) {
|
|
g_debug ("ignoring %s with UpdateState %s",
|
|
fwupd_device_get_id (dev),
|
|
fwupd_update_state_to_string (fwupd_device_get_update_state (dev)));
|
|
continue;
|
|
}
|
|
|
|
/* find the RemoteURI to use for the device */
|
|
remote_id = fwupd_release_get_remote_id (rel);
|
|
if (remote_id == NULL) {
|
|
g_debug ("%s has no RemoteID", fwupd_device_get_id (dev));
|
|
continue;
|
|
}
|
|
remote_uri = g_hash_table_lookup (remote_id_uri_map, remote_id);
|
|
if (remote_uri == NULL) {
|
|
g_debug ("%s has no RemoteURI", remote_id);
|
|
continue;
|
|
}
|
|
|
|
/* add this to the hash map */
|
|
devices_tmp = g_hash_table_lookup (report_map, remote_id);
|
|
if (devices_tmp == NULL) {
|
|
devices_tmp = g_ptr_array_new ();
|
|
g_hash_table_insert (report_map, g_strdup (remote_id), devices_tmp);
|
|
}
|
|
g_debug ("using %s for %s", remote_id, fwupd_device_get_id (dev));
|
|
g_ptr_array_add (devices_tmp, dev);
|
|
}
|
|
|
|
/* nothing to report */
|
|
if (g_hash_table_size (report_map) == 0) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED,
|
|
"No reports require uploading");
|
|
return FALSE;
|
|
}
|
|
|
|
/* process each uri */
|
|
ids = g_hash_table_get_keys (report_map);
|
|
for (GList *l = ids; l != NULL; l = l->next) {
|
|
const gchar *id = l->data;
|
|
GPtrArray *devices_tmp = g_hash_table_lookup (report_map, id);
|
|
if (!fu_util_report_history_for_remote (priv, id, devices_tmp, error))
|
|
return FALSE;
|
|
}
|
|
|
|
/* mark each device as reported */
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
FwupdDevice *dev = g_ptr_array_index (devices, i);
|
|
if (fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_REPORTED))
|
|
continue;
|
|
if (!fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_SUPPORTED))
|
|
continue;
|
|
g_debug ("setting flag on %s", fwupd_device_get_id (dev));
|
|
if (!fwupd_client_modify_device (priv->client,
|
|
fwupd_device_get_id (dev),
|
|
"Flags", "reported",
|
|
NULL, error))
|
|
return FALSE;
|
|
}
|
|
|
|
/* TRANSLATORS: success message -- where the user has uploaded
|
|
* success and/or failure reports to the remote server */
|
|
g_string_append_printf (str, ngettext ("Successfully uploaded %u report",
|
|
"Successfully uploaded %u reports",
|
|
g_hash_table_size (report_map)),
|
|
g_hash_table_size (report_map));
|
|
g_print ("%s\n", str->str);
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_get_history (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(GPtrArray) devices = NULL;
|
|
g_autoptr(GNode) root = g_node_new (NULL);
|
|
g_autofree gchar *title = fu_util_get_tree_title (priv);
|
|
|
|
/* get all devices from the history database */
|
|
devices = fwupd_client_get_history (priv->client, NULL, error);
|
|
if (devices == NULL)
|
|
return FALSE;
|
|
|
|
/* show each device */
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
g_autoptr(GPtrArray) rels = NULL;
|
|
FwupdDevice *dev = g_ptr_array_index (devices, i);
|
|
FwupdRelease *rel;
|
|
const gchar *remote;
|
|
GNode *child;
|
|
g_autoptr(GError) error_local = NULL;
|
|
|
|
if (!fu_util_filter_device (priv, dev))
|
|
continue;
|
|
child = g_node_append_data (root, dev);
|
|
|
|
rel = fwupd_device_get_release_default (dev);
|
|
if (rel == NULL)
|
|
continue;
|
|
remote = fwupd_release_get_remote_id (rel);
|
|
|
|
/* doesn't actually map to remote */
|
|
if (remote == NULL) {
|
|
g_node_append_data (child, rel);
|
|
continue;
|
|
}
|
|
|
|
/* try to lookup releases from client */
|
|
rels = fwupd_client_get_releases (priv->client,
|
|
fwupd_device_get_id (dev),
|
|
NULL, &error_local);
|
|
if (rels == NULL) {
|
|
g_debug ("failed to get releases for %s: %s",
|
|
fwupd_device_get_id (dev),
|
|
error_local->message);
|
|
g_node_append_data (child, rel);
|
|
continue;
|
|
}
|
|
|
|
/* map to a release in client */
|
|
for (guint j = 0; j < rels->len; j++) {
|
|
FwupdRelease *rel2 = g_ptr_array_index (rels, j);
|
|
if (g_strcmp0 (remote,
|
|
fwupd_release_get_remote_id (rel2)) != 0)
|
|
continue;
|
|
if (g_strcmp0 (fwupd_release_get_version (rel),
|
|
fwupd_release_get_version (rel2)) != 0)
|
|
continue;
|
|
g_node_append_data (child, g_object_ref (rel2));
|
|
rel = NULL;
|
|
break;
|
|
}
|
|
|
|
/* didn't match anything */
|
|
if (rels->len == 0 || rel != NULL) {
|
|
g_node_append_data (child, rel);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
fu_util_print_tree (root, title);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static FwupdDevice*
|
|
fu_util_get_device_or_prompt (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
FwupdDevice *dev = NULL;
|
|
|
|
/* get device to use */
|
|
if (g_strv_length (values) >= 1) {
|
|
g_autoptr(GError) error_local = NULL;
|
|
if (g_strv_length (values) > 1) {
|
|
for (guint i = 1; i < g_strv_length (values); i++)
|
|
g_debug ("Ignoring extra input %s", values[i]);
|
|
}
|
|
dev = fwupd_client_get_device_by_id (priv->client, values[0],
|
|
NULL, &error_local);
|
|
if (dev != NULL)
|
|
return dev;
|
|
g_print ("%s\n", error_local->message);
|
|
}
|
|
return fu_util_prompt_for_device (priv, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_clear_results (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FwupdDevice) dev = NULL;
|
|
|
|
dev = fu_util_get_device_or_prompt (priv, values, error);
|
|
if (dev == NULL)
|
|
return FALSE;
|
|
|
|
return fwupd_client_clear_results (priv->client, fwupd_device_get_id (dev), NULL, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_clear_offline (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FuHistory) history = fu_history_new ();
|
|
return fu_history_remove_all_with_state (history, FWUPD_UPDATE_STATE_PENDING, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_verify_update (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FwupdDevice) dev = NULL;
|
|
|
|
priv->filter_include |= FWUPD_DEVICE_FLAG_CAN_VERIFY;
|
|
dev = fu_util_get_device_or_prompt (priv, values, error);
|
|
if (dev == NULL)
|
|
return FALSE;
|
|
|
|
if (!fwupd_client_verify_update (priv->client, fwupd_device_get_id (dev), NULL, error)) {
|
|
g_prefix_error (error, "failed to verify update %s: ", fu_device_get_name (dev));
|
|
return FALSE;
|
|
}
|
|
/* TRANSLATORS: success message when user refreshes device checksums */
|
|
g_print ("%s\n", _("Successfully updated device checksums"));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_file_exists_with_checksum (const gchar *fn,
|
|
const gchar *checksum_expected,
|
|
GChecksumType checksum_type)
|
|
{
|
|
gsize len = 0;
|
|
g_autofree gchar *checksum_actual = NULL;
|
|
g_autofree gchar *data = NULL;
|
|
|
|
if (!g_file_get_contents (fn, &data, &len, NULL))
|
|
return FALSE;
|
|
checksum_actual = g_compute_checksum_for_data (checksum_type,
|
|
(guchar *) data, len);
|
|
return g_strcmp0 (checksum_expected, checksum_actual) == 0;
|
|
}
|
|
|
|
static void
|
|
fu_util_download_chunk_cb (SoupMessage *msg, SoupBuffer *chunk, gpointer user_data)
|
|
{
|
|
guint percentage;
|
|
goffset header_size;
|
|
goffset body_length;
|
|
FuUtilPrivate *priv = (FuUtilPrivate *) user_data;
|
|
|
|
/* if it's returning "Found" or an error, ignore the percentage */
|
|
if (msg->status_code != SOUP_STATUS_OK) {
|
|
g_debug ("ignoring status code %u (%s)",
|
|
msg->status_code, msg->reason_phrase);
|
|
return;
|
|
}
|
|
|
|
/* get data */
|
|
body_length = msg->response_body->length;
|
|
header_size = soup_message_headers_get_content_length (msg->response_headers);
|
|
|
|
/* size is not known */
|
|
if (header_size < body_length)
|
|
return;
|
|
|
|
/* calculate percentage */
|
|
percentage = (guint) ((100 * body_length) / header_size);
|
|
g_debug ("progress: %u%%", percentage);
|
|
fu_progressbar_update (priv->progressbar, FWUPD_STATUS_DOWNLOADING, percentage);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_download_file (FuUtilPrivate *priv,
|
|
SoupURI *uri,
|
|
const gchar *fn,
|
|
const gchar *checksum_expected,
|
|
GError **error)
|
|
{
|
|
GChecksumType checksum_type;
|
|
guint status_code;
|
|
g_autoptr(GError) error_local = NULL;
|
|
g_autofree gchar *checksum_actual = NULL;
|
|
g_autofree gchar *uri_str = NULL;
|
|
g_autoptr(SoupMessage) msg = NULL;
|
|
|
|
/* check if the file already exists with the right checksum */
|
|
checksum_type = fwupd_checksum_guess_kind (checksum_expected);
|
|
if (fu_util_file_exists_with_checksum (fn, checksum_expected, checksum_type)) {
|
|
g_debug ("skpping download as file already exists");
|
|
return TRUE;
|
|
}
|
|
|
|
/* set up networking */
|
|
if (priv->soup_session == NULL) {
|
|
priv->soup_session = fu_util_setup_networking (error);
|
|
if (priv->soup_session == NULL)
|
|
return FALSE;
|
|
}
|
|
|
|
/* download data */
|
|
uri_str = soup_uri_to_string (uri, FALSE);
|
|
g_debug ("downloading %s to %s", uri_str, fn);
|
|
msg = soup_message_new_from_uri (SOUP_METHOD_GET, uri);
|
|
if (msg == NULL) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_FILE,
|
|
"Failed to parse URI %s", uri_str);
|
|
return FALSE;
|
|
}
|
|
if (g_str_has_suffix (uri_str, ".asc") ||
|
|
g_str_has_suffix (uri_str, ".p7b") ||
|
|
g_str_has_suffix (uri_str, ".p7c")) {
|
|
/* TRANSLATORS: downloading new signing file */
|
|
g_print ("%s %s\n", _("Fetching signature"), uri_str);
|
|
} else if (g_str_has_suffix (uri_str, ".gz")) {
|
|
/* TRANSLATORS: downloading new metadata file */
|
|
g_print ("%s %s\n", _("Fetching metadata"), uri_str);
|
|
} else if (g_str_has_suffix (uri_str, ".cab")) {
|
|
/* TRANSLATORS: downloading new firmware file */
|
|
g_print ("%s %s\n", _("Fetching firmware"), uri_str);
|
|
} else {
|
|
/* TRANSLATORS: downloading unknown file */
|
|
g_print ("%s %s\n", _("Fetching file"), uri_str);
|
|
}
|
|
g_signal_connect (msg, "got-chunk",
|
|
G_CALLBACK (fu_util_download_chunk_cb), priv);
|
|
status_code = soup_session_send_message (priv->soup_session, msg);
|
|
g_print ("\n");
|
|
if (status_code == 429) {
|
|
g_autofree gchar *str = g_strndup (msg->response_body->data,
|
|
msg->response_body->length);
|
|
if (g_strcmp0 (str, "Too Many Requests") == 0) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_FILE,
|
|
/* TRANSLATORS: the server is rate-limiting downloads */
|
|
"%s", _("Failed to download due to server limit"));
|
|
return FALSE;
|
|
}
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_FILE,
|
|
"Failed to download due to server limit: %s", str);
|
|
return FALSE;
|
|
}
|
|
if (status_code != SOUP_STATUS_OK) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_FILE,
|
|
"Failed to download %s: %s",
|
|
uri_str, soup_status_get_phrase (status_code));
|
|
return FALSE;
|
|
}
|
|
|
|
/* verify checksum */
|
|
if (checksum_expected != NULL) {
|
|
checksum_actual = g_compute_checksum_for_data (checksum_type,
|
|
(guchar *) msg->response_body->data,
|
|
(gsize) msg->response_body->length);
|
|
if (g_strcmp0 (checksum_expected, checksum_actual) != 0) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_FILE,
|
|
"Checksum invalid, expected %s got %s",
|
|
checksum_expected, checksum_actual);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* save file */
|
|
if (!g_file_set_contents (fn,
|
|
msg->response_body->data,
|
|
msg->response_body->length,
|
|
&error_local)) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_WRITE,
|
|
"Failed to save file: %s",
|
|
error_local->message);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_download_metadata_for_remote (FuUtilPrivate *priv,
|
|
FwupdRemote *remote,
|
|
GError **error)
|
|
{
|
|
g_autofree gchar *basename_asc = NULL;
|
|
g_autofree gchar *basename_id_asc = NULL;
|
|
g_autofree gchar *basename_id = NULL;
|
|
g_autofree gchar *basename = NULL;
|
|
g_autofree gchar *filename = NULL;
|
|
g_autofree gchar *filename_asc = NULL;
|
|
g_autoptr(SoupURI) uri = NULL;
|
|
g_autoptr(SoupURI) uri_sig = NULL;
|
|
|
|
/* generate some plausible local filenames */
|
|
basename = g_path_get_basename (fwupd_remote_get_filename_cache (remote));
|
|
basename_id = g_strdup_printf ("%s-%s", fwupd_remote_get_id (remote), basename);
|
|
|
|
/* download the metadata */
|
|
filename = fu_util_get_user_cache_path (basename_id);
|
|
if (!fu_common_mkdir_parent (filename, error))
|
|
return FALSE;
|
|
uri = soup_uri_new (fwupd_remote_get_metadata_uri (remote));
|
|
if (!fu_util_download_file (priv, uri, filename, NULL, error))
|
|
return FALSE;
|
|
|
|
/* download the signature */
|
|
basename_asc = g_path_get_basename (fwupd_remote_get_filename_cache_sig (remote));
|
|
basename_id_asc = g_strdup_printf ("%s-%s", fwupd_remote_get_id (remote), basename_asc);
|
|
filename_asc = fu_util_get_user_cache_path (basename_id_asc);
|
|
uri_sig = soup_uri_new (fwupd_remote_get_metadata_uri_sig (remote));
|
|
if (!fu_util_download_file (priv, uri_sig, filename_asc, NULL, error))
|
|
return FALSE;
|
|
|
|
/* send all this to fwupd */
|
|
return fwupd_client_update_metadata (priv->client,
|
|
fwupd_remote_get_id (remote),
|
|
filename,
|
|
filename_asc,
|
|
NULL, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_download_metadata_enable_lvfs (FuUtilPrivate *priv, GError **error)
|
|
{
|
|
g_autoptr(FwupdRemote) remote = NULL;
|
|
|
|
/* is the LVFS available but disabled? */
|
|
remote = fwupd_client_get_remote_by_id (priv->client, "lvfs", NULL, error);
|
|
if (remote == NULL)
|
|
return TRUE;
|
|
g_print ("%s\n%s\n%s [Y|n]: ",
|
|
/* TRANSLATORS: explain why no metadata available */
|
|
_("No remotes are currently enabled so no metadata is available."),
|
|
/* TRANSLATORS: explain why no metadata available */
|
|
_("Metadata can be obtained from the Linux Vendor Firmware Service."),
|
|
/* TRANSLATORS: Turn on the remote */
|
|
_("Enable this remote?"));
|
|
if (!fu_util_prompt_for_boolean (TRUE))
|
|
return TRUE;
|
|
if (!fu_util_modify_remote (priv, "lvfs", "Enabled", "true", error))
|
|
return FALSE;
|
|
|
|
/* refresh the newly-enabled remote */
|
|
return fu_util_download_metadata_for_remote (priv, remote, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_check_oldest_remote (FuUtilPrivate *priv, guint64 *age_oldest, GError **error)
|
|
{
|
|
g_autoptr(GPtrArray) remotes = NULL;
|
|
|
|
/* get the age of the oldest enabled remotes */
|
|
remotes = fwupd_client_get_remotes (priv->client, NULL, error);
|
|
if (remotes == NULL)
|
|
return FALSE;
|
|
for (guint i = 0; i < remotes->len; i++) {
|
|
FwupdRemote *remote = g_ptr_array_index (remotes, i);
|
|
if (!fwupd_remote_get_enabled (remote))
|
|
continue;
|
|
if (fwupd_remote_get_kind (remote) != FWUPD_REMOTE_KIND_DOWNLOAD)
|
|
continue;
|
|
if (fwupd_remote_get_age (remote) > *age_oldest)
|
|
*age_oldest = fwupd_remote_get_age (remote);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_download_metadata (FuUtilPrivate *priv, GError **error)
|
|
{
|
|
gboolean download_remote_enabled = FALSE;
|
|
guint devices_supported_cnt = 0;
|
|
g_autoptr(GPtrArray) devs = NULL;
|
|
g_autoptr(GPtrArray) remotes = NULL;
|
|
g_autoptr(GString) str = g_string_new (NULL);
|
|
|
|
/* metadata refreshed recently */
|
|
if ((priv->flags & FWUPD_INSTALL_FLAG_FORCE) == 0) {
|
|
guint64 age_oldest = 0;
|
|
const guint64 age_limit_hours = 24;
|
|
|
|
if (!fu_util_check_oldest_remote (priv, &age_oldest, error))
|
|
return FALSE;
|
|
if (age_oldest < 60 * 60 * age_limit_hours) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
/* TRANSLATORS: error message for a user who ran fwupdmgr refresh recently
|
|
%1 is an already translated timestamp such as 6 hours or 15 seconds */
|
|
"Firmware metadata last refresh: %s ago. "
|
|
"Use --force to refresh again.",
|
|
fu_util_time_to_str (age_oldest));
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
remotes = fwupd_client_get_remotes (priv->client, NULL, error);
|
|
if (remotes == NULL)
|
|
return FALSE;
|
|
for (guint i = 0; i < remotes->len; i++) {
|
|
FwupdRemote *remote = g_ptr_array_index (remotes, i);
|
|
if (!fwupd_remote_get_enabled (remote))
|
|
continue;
|
|
if (fwupd_remote_get_kind (remote) != FWUPD_REMOTE_KIND_DOWNLOAD)
|
|
continue;
|
|
download_remote_enabled = TRUE;
|
|
if (!fu_util_download_metadata_for_remote (priv, remote, error))
|
|
return FALSE;
|
|
}
|
|
|
|
/* no web remote is declared; try to enable LVFS */
|
|
if (!download_remote_enabled) {
|
|
/* we don't want to ask anything */
|
|
if (priv->no_metadata_check) {
|
|
g_debug ("skipping metadata check");
|
|
return TRUE;
|
|
}
|
|
|
|
if (!fu_util_download_metadata_enable_lvfs (priv, error))
|
|
return FALSE;
|
|
}
|
|
|
|
/* get devices from daemon */
|
|
devs = fwupd_client_get_devices (priv->client, NULL, error);
|
|
if (devs == NULL)
|
|
return FALSE;
|
|
|
|
/* get results */
|
|
for (guint i = 0; i < devs->len; i++) {
|
|
FwupdDevice *dev = g_ptr_array_index (devs, i);
|
|
if (fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_SUPPORTED))
|
|
devices_supported_cnt++;
|
|
}
|
|
|
|
/* TRANSLATORS: success message -- where 'metadata' is information
|
|
* about available firmware on the remote server */
|
|
g_string_append (str, _("Successfully downloaded new metadata: "));
|
|
|
|
/* TRANSLATORS: how many local devices can expect updates now */
|
|
g_string_append_printf (str, ngettext ("%u local device supported",
|
|
"%u local devices supported",
|
|
devices_supported_cnt),
|
|
devices_supported_cnt);
|
|
g_print ("%s\n", str->str);
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_refresh (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
if (g_strv_length (values) == 0)
|
|
return fu_util_download_metadata (priv, error);
|
|
if (g_strv_length (values) != 3) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments");
|
|
return FALSE;
|
|
}
|
|
|
|
/* open file */
|
|
if (!fwupd_client_update_metadata (priv->client,
|
|
values[2],
|
|
values[0],
|
|
values[1],
|
|
NULL,
|
|
error))
|
|
return FALSE;
|
|
|
|
/* TRANSLATORS: success message -- the user can do this by-hand too */
|
|
g_print ("%s\n", _("Successfully refreshed metadata manually"));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_get_results (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autofree gchar *tmp = NULL;
|
|
g_autoptr(FwupdDevice) dev = NULL;
|
|
g_autoptr(FwupdDevice) rel = NULL;
|
|
|
|
dev = fu_util_get_device_or_prompt (priv, values, error);
|
|
if (dev == NULL)
|
|
return FALSE;
|
|
|
|
rel = fwupd_client_get_results (priv->client, fwupd_device_get_id (dev), NULL, error);
|
|
if (rel == NULL)
|
|
return FALSE;
|
|
tmp = fu_util_device_to_string (rel, 0);
|
|
g_print ("%s", tmp);
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_get_releases (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FwupdDevice) dev = NULL;
|
|
g_autoptr(GPtrArray) rels = NULL;
|
|
g_autoptr(GNode) root = g_node_new (NULL);
|
|
g_autofree gchar *title = fu_util_get_tree_title (priv);
|
|
|
|
priv->filter_include |= FWUPD_DEVICE_FLAG_SUPPORTED;
|
|
dev = fu_util_get_device_or_prompt (priv, values, error);
|
|
if (dev == NULL)
|
|
return FALSE;
|
|
|
|
/* get the releases for this device */
|
|
rels = fwupd_client_get_releases (priv->client, fwupd_device_get_id (dev), NULL, error);
|
|
if (rels == NULL)
|
|
return FALSE;
|
|
|
|
if (rels->len == 0) {
|
|
/* TRANSLATORS: no repositories to download from */
|
|
g_print ("%s\n", _("No releases available"));
|
|
return TRUE;
|
|
}
|
|
|
|
for (guint i = 0; i < rels->len; i++) {
|
|
FwupdRelease *rel = g_ptr_array_index (rels, i);
|
|
g_node_append_data (root, rel);
|
|
}
|
|
fu_util_print_tree (root, title);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static FwupdRelease *
|
|
fu_util_prompt_for_release (FuUtilPrivate *priv, GPtrArray *rels, GError **error)
|
|
{
|
|
FwupdRelease *rel;
|
|
guint idx;
|
|
|
|
/* nothing */
|
|
if (rels->len == 0) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"No supported releases");
|
|
return NULL;
|
|
}
|
|
|
|
/* exactly one */
|
|
if (rels->len == 1) {
|
|
rel = g_ptr_array_index (rels, 0);
|
|
return g_object_ref (rel);
|
|
}
|
|
|
|
/* TRANSLATORS: get interactive prompt */
|
|
g_print ("%s\n", _("Choose a release:"));
|
|
/* TRANSLATORS: this is to abort the interactive prompt */
|
|
g_print ("0.\t%s\n", _("Cancel"));
|
|
for (guint i = 0; i < rels->len; i++) {
|
|
const gchar *desc_tmp;
|
|
g_autofree gchar *desc = NULL;
|
|
|
|
rel = g_ptr_array_index (rels, i);
|
|
|
|
/* no description provided */
|
|
desc_tmp = fwupd_release_get_description (rel);
|
|
if (desc_tmp == NULL) {
|
|
g_print ("%u.\t%s\n", i + 1, fwupd_release_get_version (rel));
|
|
continue;
|
|
}
|
|
|
|
/* remove markup, and fall back if we fail */
|
|
desc = fu_util_convert_description (desc_tmp, NULL);
|
|
if (desc == NULL)
|
|
desc = g_strdup (desc_tmp);
|
|
g_print ("%u.\t%s (%s)\n", i + 1, fwupd_release_get_version (rel), desc);
|
|
}
|
|
idx = fu_util_prompt_for_number (rels->len);
|
|
if (idx == 0) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"Request canceled");
|
|
return NULL;
|
|
}
|
|
rel = g_ptr_array_index (rels, idx - 1);
|
|
return g_object_ref (rel);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_verify (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FwupdDevice) dev = NULL;
|
|
|
|
priv->filter_include |= FWUPD_DEVICE_FLAG_CAN_VERIFY;
|
|
dev = fu_util_get_device_or_prompt (priv, values, error);
|
|
if (dev == NULL)
|
|
return FALSE;
|
|
|
|
if (!fwupd_client_verify (priv->client, fwupd_device_get_id (dev), NULL, error)) {
|
|
g_prefix_error (error, "failed to verify %s: ", fu_device_get_name (dev));
|
|
return FALSE;
|
|
}
|
|
/* TRANSLATORS: success message when user verified device checksums */
|
|
g_print ("%s\n", _("Successfully verified device checksums"));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_unlock (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FwupdDevice) dev = NULL;
|
|
|
|
priv->filter_include |= FWUPD_DEVICE_FLAG_LOCKED;
|
|
dev = fu_util_get_device_or_prompt (priv, values, error);
|
|
if (dev == NULL)
|
|
return FALSE;
|
|
|
|
if (fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_NEEDS_SHUTDOWN))
|
|
priv->completion_flags |= FWUPD_DEVICE_FLAG_NEEDS_SHUTDOWN;
|
|
if (fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_NEEDS_REBOOT))
|
|
priv->completion_flags |= FWUPD_DEVICE_FLAG_NEEDS_REBOOT;
|
|
|
|
if (!fwupd_client_unlock (priv->client, fwupd_device_get_id (dev), NULL, error))
|
|
return FALSE;
|
|
|
|
return fu_util_prompt_complete (priv->completion_flags, TRUE, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_perhaps_refresh_remotes (FuUtilPrivate *priv, GError **error)
|
|
{
|
|
guint64 age_oldest = 0;
|
|
const guint64 age_limit_days = 30;
|
|
|
|
/* we don't want to ask anything */
|
|
if (priv->no_metadata_check) {
|
|
g_debug ("skipping metadata check");
|
|
return TRUE;
|
|
}
|
|
|
|
if (!fu_util_check_oldest_remote (priv, &age_oldest, error))
|
|
return FALSE;
|
|
|
|
/* metadata is new enough */
|
|
if (age_oldest < 60 * 60 * 24 * age_limit_days)
|
|
return TRUE;
|
|
|
|
/* ask for permission */
|
|
if (!priv->assume_yes) {
|
|
/* TRANSLATORS: the metadata is very out of date; %u is a number > 1 */
|
|
g_print (ngettext("Firmware metadata has not been updated for %u"
|
|
" day and may not be up to date.",
|
|
"Firmware metadata has not been updated for %u"
|
|
" days and may not be up to date.",
|
|
(gint) age_limit_days), (guint) age_limit_days);
|
|
g_print ("\n\n");
|
|
g_print ("%s (%s) [y|N]: ",
|
|
/* TRANSLATORS: ask the user if we can update the metadata */
|
|
_("Update now?"),
|
|
/* TRANSLATORS: metadata is downloaded from the Internet */
|
|
_("Requires internet connection"));
|
|
if (!fu_util_prompt_for_boolean (FALSE))
|
|
return TRUE;
|
|
}
|
|
|
|
/* downloads new metadata */
|
|
return fu_util_download_metadata (priv, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_get_updates (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(GPtrArray) devices = NULL;
|
|
gboolean supported = FALSE;
|
|
g_autoptr(GNode) root = g_node_new (NULL);
|
|
g_autofree gchar *title = fu_util_get_tree_title (priv);
|
|
|
|
/* are the remotes very old */
|
|
if (!fu_util_perhaps_refresh_remotes (priv, error))
|
|
return FALSE;
|
|
|
|
/* get devices from daemon */
|
|
devices = fwupd_client_get_devices (priv->client, NULL, error);
|
|
if (devices == NULL)
|
|
return FALSE;
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
FwupdDevice *dev = g_ptr_array_index (devices, i);
|
|
g_autofree gchar *upgrade_str = NULL;
|
|
g_autoptr(GPtrArray) rels = NULL;
|
|
g_autoptr(GError) error_local = NULL;
|
|
GNode *child;
|
|
|
|
/* not going to have results, so save a D-Bus round-trip */
|
|
if (!fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_SUPPORTED))
|
|
continue;
|
|
if (!fu_util_filter_device (priv, dev))
|
|
continue;
|
|
supported = TRUE;
|
|
|
|
/* get the releases for this device and filter for validity */
|
|
rels = fwupd_client_get_upgrades (priv->client,
|
|
fwupd_device_get_id (dev),
|
|
NULL, &error_local);
|
|
if (rels == NULL) {
|
|
/* TRANSLATORS: message letting the user know no device upgrade available
|
|
* %1 is the device name */
|
|
upgrade_str = g_strdup_printf (_("No upgrades for %s"),
|
|
fwupd_device_get_name (dev));
|
|
g_printerr ("%s: %s\n", upgrade_str, error_local->message);
|
|
continue;
|
|
}
|
|
child = g_node_append_data (root, dev);
|
|
|
|
/* add all releases */
|
|
for (guint j = 0; j < rels->len; j++) {
|
|
FwupdRelease *rel = g_ptr_array_index (rels, j);
|
|
g_node_append_data (child, g_object_ref (rel));
|
|
}
|
|
}
|
|
|
|
if (g_node_n_nodes (root, G_TRAVERSE_ALL) > 1)
|
|
fu_util_print_tree (root, title);
|
|
|
|
/* nag? */
|
|
if (!fu_util_perhaps_show_unreported (priv, error))
|
|
return FALSE;
|
|
|
|
/* no devices supported by LVFS or all are filtered */
|
|
if (!supported) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"No updatable devices");
|
|
return FALSE;
|
|
}
|
|
|
|
/* success */
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_get_remotes (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(GNode) root = g_node_new (NULL);
|
|
g_autoptr(GPtrArray) remotes = NULL;
|
|
g_autofree gchar *title = fu_util_get_tree_title (priv);
|
|
|
|
remotes = fwupd_client_get_remotes (priv->client, NULL, error);
|
|
if (remotes == NULL)
|
|
return FALSE;
|
|
|
|
if (remotes->len == 0) {
|
|
/* TRANSLATORS: no repositories to download from */
|
|
g_print ("%s\n", _("No remotes available"));
|
|
return TRUE;
|
|
}
|
|
|
|
for (guint i = 0; i < remotes->len; i++) {
|
|
FwupdRemote *remote_tmp = g_ptr_array_index (remotes, i);
|
|
g_node_append_data (root, remote_tmp);
|
|
}
|
|
fu_util_print_tree (root, title);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_update_device_with_release (FuUtilPrivate *priv,
|
|
FwupdDevice *dev,
|
|
FwupdRelease *rel,
|
|
GError **error)
|
|
{
|
|
GPtrArray *checksums;
|
|
const gchar *remote_id;
|
|
const gchar *uri_tmp;
|
|
g_autofree gchar *fn = NULL;
|
|
g_autofree gchar *uri_str = NULL;
|
|
g_autoptr(SoupURI) uri = NULL;
|
|
|
|
if (!priv->no_safety_check && !priv->assume_yes) {
|
|
if (!fu_util_prompt_warning (dev,
|
|
fu_util_get_tree_title (priv),
|
|
error))
|
|
return FALSE;
|
|
}
|
|
|
|
/* work out what remote-specific URI fields this should use */
|
|
uri_tmp = fwupd_release_get_uri (rel);
|
|
remote_id = fwupd_release_get_remote_id (rel);
|
|
if (remote_id != NULL) {
|
|
g_autoptr(FwupdRemote) remote = NULL;
|
|
remote = fwupd_client_get_remote_by_id (priv->client,
|
|
remote_id,
|
|
NULL,
|
|
error);
|
|
if (remote == NULL)
|
|
return FALSE;
|
|
|
|
/* local and directory remotes have the firmware already */
|
|
if (fwupd_remote_get_kind (remote) == FWUPD_REMOTE_KIND_LOCAL) {
|
|
const gchar *fn_cache = fwupd_remote_get_filename_cache (remote);
|
|
g_autofree gchar *path = g_path_get_dirname (fn_cache);
|
|
|
|
fn = g_build_filename (path, uri_tmp, NULL);
|
|
} else if (fwupd_remote_get_kind (remote) == FWUPD_REMOTE_KIND_DIRECTORY) {
|
|
fn = g_strdup (uri_tmp + 7);
|
|
}
|
|
/* install with flags chosen by the user */
|
|
if (fn != NULL) {
|
|
return fwupd_client_install (priv->client,
|
|
fwupd_device_get_id (dev),
|
|
fn, priv->flags, NULL, error);
|
|
}
|
|
|
|
uri_str = fwupd_remote_build_firmware_uri (remote, uri_tmp, error);
|
|
if (uri_str == NULL)
|
|
return FALSE;
|
|
} else {
|
|
uri_str = g_strdup (uri_tmp);
|
|
}
|
|
|
|
/* download file */
|
|
g_print ("Downloading %s for %s...\n",
|
|
fwupd_release_get_version (rel),
|
|
fwupd_device_get_name (dev));
|
|
fn = fu_util_get_user_cache_path (uri_str);
|
|
if (!fu_common_mkdir_parent (fn, error))
|
|
return FALSE;
|
|
checksums = fwupd_release_get_checksums (rel);
|
|
uri = soup_uri_new (uri_str);
|
|
if (!fu_util_download_file (priv, uri, fn,
|
|
fwupd_checksum_get_best (checksums),
|
|
error))
|
|
return FALSE;
|
|
/* if the device specifies ONLY_OFFLINE automatically set this flag */
|
|
if (fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_ONLY_OFFLINE))
|
|
priv->flags |= FWUPD_INSTALL_FLAG_OFFLINE;
|
|
return fwupd_client_install (priv->client,
|
|
fwupd_device_get_id (dev), fn,
|
|
priv->flags, NULL, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_maybe_send_reports (FuUtilPrivate *priv, const gchar *remote_id,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FwupdRemote) remote = NULL;
|
|
g_autoptr(GError) error_local = NULL;
|
|
if (remote_id == NULL) {
|
|
g_debug ("not sending reports, no remote");
|
|
return TRUE;
|
|
}
|
|
remote = fwupd_client_get_remote_by_id (priv->client,
|
|
remote_id,
|
|
NULL,
|
|
error);
|
|
if (remote == NULL)
|
|
return FALSE;
|
|
if (fwupd_remote_get_automatic_reports (remote)) {
|
|
if (!fu_util_report_history (priv, NULL, &error_local))
|
|
if (!g_error_matches (error_local,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED))
|
|
g_warning ("%s", error_local->message);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_update_all (FuUtilPrivate *priv, GError **error)
|
|
{
|
|
g_autoptr(GPtrArray) devices = NULL;
|
|
gboolean supported = FALSE;
|
|
|
|
/* get devices from daemon */
|
|
devices = fwupd_client_get_devices (priv->client, NULL, error);
|
|
if (devices == NULL)
|
|
return FALSE;
|
|
priv->current_operation = FU_UTIL_OPERATION_UPDATE;
|
|
g_signal_connect (priv->client, "device-changed",
|
|
G_CALLBACK (fu_util_update_device_changed_cb), priv);
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
FwupdDevice *dev = g_ptr_array_index (devices, i);
|
|
FwupdRelease *rel;
|
|
const gchar *remote_id;
|
|
g_autofree gchar *upgrade_str = NULL;
|
|
g_autoptr(GPtrArray) rels = NULL;
|
|
g_autoptr(GError) error_local = NULL;
|
|
|
|
/* not going to have results, so save a D-Bus round-trip */
|
|
if (!fwupd_device_has_flag (dev, FWUPD_DEVICE_FLAG_SUPPORTED))
|
|
continue;
|
|
if (!fu_util_filter_device (priv, dev))
|
|
continue;
|
|
supported = TRUE;
|
|
|
|
/* get the releases for this device and filter for validity */
|
|
rels = fwupd_client_get_upgrades (priv->client,
|
|
fwupd_device_get_id (dev),
|
|
NULL, &error_local);
|
|
if (rels == NULL) {
|
|
/* TRANSLATORS: message letting the user know no device upgrade available
|
|
* %1 is the device name */
|
|
upgrade_str = g_strdup_printf (_("No upgrades for %s"),
|
|
fwupd_device_get_name (dev));
|
|
g_printerr ("%s: %s\n", upgrade_str, error_local->message);
|
|
continue;
|
|
}
|
|
rel = g_ptr_array_index (rels, 0);
|
|
/* TRANSLATORS: message letting the user know an upgrade is available
|
|
* %1 is the device name and %2 and %3 are version strings */
|
|
upgrade_str = g_strdup_printf (_("Upgrade available for %s from %s to %s"),
|
|
fwupd_device_get_name (dev),
|
|
fwupd_device_get_version (dev),
|
|
fwupd_release_get_version (rel));
|
|
g_print ("%s\n", upgrade_str);
|
|
if (!fu_util_update_device_with_release (priv, dev, rel, error))
|
|
return FALSE;
|
|
|
|
fu_util_display_current_message (priv);
|
|
|
|
/* send report if we're supposed to */
|
|
remote_id = fwupd_release_get_remote_id (rel);
|
|
if (!fu_util_maybe_send_reports (priv, remote_id, error))
|
|
return FALSE;
|
|
}
|
|
|
|
/* no devices supported by LVFS or all are filtered */
|
|
if (!supported) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"No updatable devices");
|
|
return FALSE;
|
|
}
|
|
|
|
/* we don't want to ask anything */
|
|
if (priv->no_reboot_check) {
|
|
g_debug ("skipping reboot check");
|
|
return TRUE;
|
|
}
|
|
|
|
return fu_util_prompt_complete (priv->completion_flags, TRUE, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_update_by_id (FuUtilPrivate *priv, const gchar *device_id, GError **error)
|
|
{
|
|
FwupdRelease *rel;
|
|
const gchar *remote_id;
|
|
g_autoptr(FwupdDevice) dev = NULL;
|
|
g_autoptr(GPtrArray) rels = NULL;
|
|
|
|
/* do not allow a partial device-id */
|
|
dev = fwupd_client_get_device_by_id (priv->client, device_id, NULL, error);
|
|
if (dev == NULL)
|
|
return FALSE;
|
|
|
|
/* get devices from daemon */
|
|
priv->current_operation = FU_UTIL_OPERATION_UPDATE;
|
|
g_signal_connect (priv->client, "device-changed",
|
|
G_CALLBACK (fu_util_update_device_changed_cb), priv);
|
|
|
|
/* get the releases for this device and filter for validity */
|
|
rels = fwupd_client_get_upgrades (priv->client,
|
|
fwupd_device_get_id (dev),
|
|
NULL, error);
|
|
if (rels == NULL)
|
|
return FALSE;
|
|
rel = g_ptr_array_index (rels, 0);
|
|
if (!fu_util_update_device_with_release (priv, dev, rel, error))
|
|
return FALSE;
|
|
fu_util_display_current_message (priv);
|
|
|
|
/* send report if we're supposed to */
|
|
remote_id = fwupd_release_get_remote_id (rel);
|
|
if (!fu_util_maybe_send_reports (priv, remote_id, error))
|
|
return FALSE;
|
|
|
|
/* we don't want to ask anything */
|
|
if (priv->no_reboot_check) {
|
|
g_debug ("skipping reboot check");
|
|
return TRUE;
|
|
}
|
|
|
|
/* the update needs the user to restart the computer */
|
|
return fu_util_prompt_complete (priv->completion_flags, TRUE, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_update (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
if (priv->flags & FWUPD_INSTALL_FLAG_ALLOW_OLDER) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"--allow-older is not supported for this command");
|
|
return FALSE;
|
|
}
|
|
|
|
if (priv->flags & FWUPD_INSTALL_FLAG_ALLOW_REINSTALL) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"--allow-reinstall is not supported for this command");
|
|
return FALSE;
|
|
}
|
|
|
|
if (g_strv_length (values) == 0)
|
|
return fu_util_update_all (priv, error);
|
|
if (g_strv_length (values) == 1)
|
|
return fu_util_update_by_id (priv, values[0], error);
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments");
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_remote_modify (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
if (g_strv_length (values) < 3) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments");
|
|
return FALSE;
|
|
}
|
|
if (!fu_util_modify_remote (priv, values[0], values[1], values[2], error))
|
|
return FALSE;
|
|
|
|
/* TRANSLATORS: success message for a per-remote setting change */
|
|
g_print ("%s\n", _("Successfully modified remote"));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_remote_enable (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
if (g_strv_length (values) != 1) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments");
|
|
return FALSE;
|
|
}
|
|
if (!fu_util_modify_remote (priv, values[0], "Enabled", "true", error))
|
|
return FALSE;
|
|
|
|
/* TRANSLATORS: success message */
|
|
g_print ("%s\n", _("Successfully enabled remote"));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_remote_disable (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
if (g_strv_length (values) != 1) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments");
|
|
return FALSE;
|
|
}
|
|
if (!fu_util_modify_remote (priv, values[0], "Enabled", "false", error))
|
|
return FALSE;
|
|
|
|
/* TRANSLATORS: success message */
|
|
g_print ("%s\n", _("Successfully disabled remote"));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_downgrade (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
const gchar *remote_id;
|
|
g_autoptr(FwupdDevice) dev = NULL;
|
|
g_autoptr(FwupdRelease) rel = NULL;
|
|
g_autoptr(GPtrArray) rels = NULL;
|
|
|
|
if (priv->flags & FWUPD_INSTALL_FLAG_ALLOW_REINSTALL) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"--allow-reinstall is not supported for this command");
|
|
return FALSE;
|
|
}
|
|
|
|
priv->filter_include |= FWUPD_DEVICE_FLAG_SUPPORTED;
|
|
dev = fu_util_get_device_or_prompt (priv, values, error);
|
|
if (dev == NULL)
|
|
return FALSE;
|
|
|
|
/* get the releases for this device and filter for validity */
|
|
rels = fwupd_client_get_downgrades (priv->client,
|
|
fwupd_device_get_id (dev),
|
|
NULL, error);
|
|
if (rels == NULL) {
|
|
/* TRANSLATORS: message letting the user know no device downgrade available
|
|
* %1 is the device name */
|
|
g_autofree gchar *downgrade_str = g_strdup_printf (_("No downgrades for %s"),
|
|
fwupd_device_get_name (dev));
|
|
g_prefix_error (error, "%s: ", downgrade_str);
|
|
return FALSE;
|
|
}
|
|
|
|
/* get the chosen release */
|
|
rel = fu_util_prompt_for_release (priv, rels, error);
|
|
if (rel == NULL)
|
|
return FALSE;
|
|
|
|
/* update the console if composite devices are also updated */
|
|
priv->current_operation = FU_UTIL_OPERATION_DOWNGRADE;
|
|
g_signal_connect (priv->client, "device-changed",
|
|
G_CALLBACK (fu_util_update_device_changed_cb), priv);
|
|
priv->flags |= FWUPD_INSTALL_FLAG_ALLOW_OLDER;
|
|
if (!fu_util_update_device_with_release (priv, dev, rel, error))
|
|
return FALSE;
|
|
|
|
/* TRANSLATORS: success message where we made the firmware on the
|
|
* device older than it was before */
|
|
g_print ("%s\n", _("Successfully downgraded device"));
|
|
|
|
/* send report if we're supposed to */
|
|
remote_id = fwupd_release_get_remote_id (rel);
|
|
if (!fu_util_maybe_send_reports (priv, remote_id, error))
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_reinstall (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
const gchar *remote_id;
|
|
g_autoptr(FwupdRelease) rel = NULL;
|
|
g_autoptr(GPtrArray) rels = NULL;
|
|
g_autoptr(FwupdDevice) dev = NULL;
|
|
|
|
priv->filter_include |= FWUPD_DEVICE_FLAG_SUPPORTED;
|
|
dev = fu_util_get_device_or_prompt (priv, values, error);
|
|
if (dev == NULL)
|
|
return FALSE;
|
|
|
|
/* try to lookup/match release from client */
|
|
rels = fwupd_client_get_releases (priv->client, fwupd_device_get_id (dev),
|
|
NULL, error);
|
|
if (rels == NULL)
|
|
return FALSE;
|
|
for (guint j = 0; j < rels->len; j++) {
|
|
FwupdRelease *rel_tmp = g_ptr_array_index (rels, j);
|
|
if (fu_common_vercmp (fwupd_release_get_version (rel_tmp),
|
|
fu_device_get_version (dev)) == 0) {
|
|
rel = g_object_ref (rel_tmp);
|
|
break;
|
|
}
|
|
}
|
|
if (rel == NULL) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED,
|
|
"Unable to locate release for %s version %s",
|
|
fu_device_get_name (dev),
|
|
fu_device_get_version (dev));
|
|
return FALSE;
|
|
}
|
|
|
|
/* update the console if composite devices are also updated */
|
|
priv->current_operation = FU_UTIL_OPERATION_INSTALL;
|
|
g_signal_connect (priv->client, "device-changed",
|
|
G_CALLBACK (fu_util_update_device_changed_cb), priv);
|
|
priv->flags |= FWUPD_INSTALL_FLAG_ALLOW_REINSTALL;
|
|
if (!fu_util_update_device_with_release (priv, dev, rel, error))
|
|
return FALSE;
|
|
fu_util_display_current_message (priv);
|
|
|
|
/* send report if we're supposed to */
|
|
remote_id = fwupd_release_get_remote_id (rel);
|
|
if (!fu_util_maybe_send_reports (priv, remote_id, error))
|
|
return FALSE;
|
|
|
|
/* we don't want to ask anything */
|
|
if (priv->no_reboot_check) {
|
|
g_debug ("skipping reboot check");
|
|
return TRUE;
|
|
}
|
|
|
|
return fu_util_prompt_complete (priv->completion_flags, TRUE, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_activate (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(GPtrArray) devices = NULL;
|
|
gboolean has_pending = FALSE;
|
|
|
|
/* handle both forms */
|
|
if (g_strv_length (values) == 0) {
|
|
/* activate anything with _NEEDS_ACTIVATION */
|
|
devices = fwupd_client_get_devices (priv->client, NULL, error);
|
|
if (devices == NULL)
|
|
return FALSE;
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
FuDevice *device = g_ptr_array_index (devices, i);
|
|
if (fu_device_has_flag (device, FWUPD_DEVICE_FLAG_NEEDS_ACTIVATION)) {
|
|
has_pending = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
} else if (g_strv_length (values) == 1) {
|
|
FwupdDevice *device = fwupd_client_get_device_by_id (priv->client,
|
|
values[0],
|
|
NULL,
|
|
error);
|
|
if (device == NULL)
|
|
return FALSE;
|
|
devices = g_ptr_array_new_with_free_func ((GDestroyNotify) g_object_unref);
|
|
g_ptr_array_add (devices, device);
|
|
if (fwupd_device_has_flag (device, FWUPD_DEVICE_FLAG_NEEDS_ACTIVATION))
|
|
has_pending = TRUE;
|
|
} else {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments");
|
|
return FALSE;
|
|
}
|
|
|
|
/* nothing to do */
|
|
if (!has_pending) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"No firmware to activate");
|
|
return FALSE;
|
|
}
|
|
|
|
/* activate anything with _NEEDS_ACTIVATION */
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
FwupdDevice *device = g_ptr_array_index (devices, i);
|
|
if (!fu_util_filter_device (priv, device))
|
|
continue;
|
|
if (!fu_device_has_flag (device, FWUPD_DEVICE_FLAG_NEEDS_ACTIVATION))
|
|
continue;
|
|
/* TRANSLATORS: shown when shutting down to switch to the new version */
|
|
g_print ("%s %s…\n", _("Activating firmware update for"),
|
|
fwupd_device_get_name (device));
|
|
if (!fwupd_client_activate (priv->client, NULL,
|
|
fwupd_device_get_id (device), error))
|
|
return FALSE;
|
|
}
|
|
|
|
/* TRANSLATORS: success message -- where activation is making the new
|
|
* firmware take effect, usually after updating offline */
|
|
g_print ("%s\n", _("Successfully activated all devices"));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_set_approved_firmware (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_auto(GStrv) checksums = NULL;
|
|
|
|
/* check args */
|
|
if (g_strv_length (values) != 1) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments: list of checksums expected");
|
|
return FALSE;
|
|
}
|
|
|
|
/* call into daemon */
|
|
checksums = g_strsplit (values[0], ",", -1);
|
|
return fwupd_client_set_approved_firmware (priv->client,
|
|
checksums,
|
|
priv->cancellable,
|
|
error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_get_approved_firmware (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_auto(GStrv) checksums = NULL;
|
|
|
|
/* check args */
|
|
if (g_strv_length (values) != 0) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments: none expected");
|
|
return FALSE;
|
|
}
|
|
|
|
/* call into daemon */
|
|
checksums = fwupd_client_get_approved_firmware (priv->client,
|
|
priv->cancellable,
|
|
error);
|
|
if (checksums == NULL)
|
|
return FALSE;
|
|
if (g_strv_length (checksums) == 0) {
|
|
/* TRANSLATORS: approved firmware has been checked by
|
|
* the domain administrator */
|
|
g_print ("%s\n", _("There is no approved firmware."));
|
|
} else {
|
|
/* TRANSLATORS: approved firmware has been checked by
|
|
* the domain administrator */
|
|
g_print ("%s\n", ngettext ("Approved firmware:",
|
|
"Approved firmware:",
|
|
g_strv_length (checksums)));
|
|
for (guint i = 0; checksums[i] != NULL; i++)
|
|
g_print (" * %s\n", checksums[i]);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_modify_config (FuUtilPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
/* check args */
|
|
if (g_strv_length (values) != 2) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_ARGS,
|
|
"Invalid arguments: KEY VALUE expected");
|
|
return FALSE;
|
|
}
|
|
if (!fwupd_client_modify_config (priv->client,
|
|
values[0], values[1],
|
|
priv->cancellable,
|
|
error))
|
|
return FALSE;
|
|
if (!priv->assume_yes) {
|
|
g_print ("%s [Y|n]: ",
|
|
/* TRANSLATORS: configuration changes only take effect on restart */
|
|
_("Restart the daemon to make the change effective?"));
|
|
if (!fu_util_prompt_for_boolean (FALSE))
|
|
return TRUE;
|
|
}
|
|
#ifdef HAVE_SYSTEMD
|
|
if (!fu_systemd_unit_stop (fu_util_get_systemd_unit (), error))
|
|
return FALSE;
|
|
#endif
|
|
/* TRANSLATORS: success message -- a per-system setting value */
|
|
g_print ("%s\n", _("Successfully modified configuration value"));
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
fu_util_ignore_cb (const gchar *log_domain, GLogLevelFlags log_level,
|
|
const gchar *message, gpointer user_data)
|
|
{
|
|
}
|
|
|
|
#ifdef HAVE_GIO_UNIX
|
|
static gboolean
|
|
fu_util_sigint_cb (gpointer user_data)
|
|
{
|
|
FuUtilPrivate *priv = (FuUtilPrivate *) user_data;
|
|
g_debug ("Handling SIGINT");
|
|
g_cancellable_cancel (priv->cancellable);
|
|
return FALSE;
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
fu_util_private_free (FuUtilPrivate *priv)
|
|
{
|
|
if (priv->client != NULL)
|
|
g_object_unref (priv->client);
|
|
if (priv->current_device != NULL)
|
|
g_object_unref (priv->current_device);
|
|
if (priv->soup_session != NULL)
|
|
g_object_unref (priv->soup_session);
|
|
g_free (priv->current_message);
|
|
g_main_loop_unref (priv->loop);
|
|
g_object_unref (priv->cancellable);
|
|
g_object_unref (priv->progressbar);
|
|
g_option_context_free (priv->context);
|
|
g_free (priv);
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_check_daemon_version (FuUtilPrivate *priv, GError **error)
|
|
{
|
|
g_autofree gchar *client = fu_util_get_client_version ();
|
|
const gchar *daemon = fwupd_client_get_daemon_version (priv->client);
|
|
|
|
if (g_strcmp0 (daemon, client) != 0) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED,
|
|
/* TRANSLATORS: error message */
|
|
_("Unsupported daemon version %s, client version is %s"),
|
|
daemon, client);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_util_check_polkit_actions (GError **error)
|
|
{
|
|
g_autofree gchar *directory = fu_common_get_path (FU_PATH_KIND_POLKIT_ACTIONS);
|
|
g_autofree gchar *filename = g_build_filename (directory,
|
|
"org.freedesktop.fwupd.policy",
|
|
NULL);
|
|
if (!g_file_test (filename, G_FILE_TEST_IS_REGULAR)) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_AUTH_FAILED,
|
|
"PolicyKit files are missing, see https://github.com/fwupd/fwupd/wiki/PolicyKit-files-are-missing");
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
fu_util_display_help (FuUtilPrivate *priv)
|
|
{
|
|
g_autofree gchar *tmp = NULL;
|
|
tmp = g_option_context_get_help (priv->context, TRUE, NULL);
|
|
g_printerr ("%s\n", tmp);
|
|
}
|
|
|
|
#pragma clang diagnostic push
|
|
#pragma clang diagnostic ignored "-Wunused-function"
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC(FuUtilPrivate, fu_util_private_free)
|
|
#pragma clang diagnostic pop
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
gboolean force = FALSE;
|
|
gboolean allow_older = FALSE;
|
|
gboolean allow_reinstall = FALSE;
|
|
gboolean no_history = FALSE;
|
|
gboolean offline = FALSE;
|
|
gboolean ret;
|
|
gboolean verbose = FALSE;
|
|
gboolean version = FALSE;
|
|
g_autoptr(FuUtilPrivate) priv = g_new0 (FuUtilPrivate, 1);
|
|
g_autoptr(GError) error = NULL;
|
|
g_autoptr(GPtrArray) cmd_array = fu_util_cmd_array_new ();
|
|
g_autofree gchar *cmd_descriptions = NULL;
|
|
g_autofree gchar *filter = NULL;
|
|
g_autofree gchar *log = NULL;
|
|
const GOptionEntry options[] = {
|
|
{ "verbose", 'v', 0, G_OPTION_ARG_NONE, &verbose,
|
|
/* TRANSLATORS: command line option */
|
|
_("Show extra debugging information"), NULL },
|
|
{ "version", '\0', 0, G_OPTION_ARG_NONE, &version,
|
|
/* TRANSLATORS: command line option */
|
|
_("Show client and daemon versions"), NULL },
|
|
{ "offline", '\0', 0, G_OPTION_ARG_NONE, &offline,
|
|
/* TRANSLATORS: command line option */
|
|
_("Schedule installation for next reboot when possible"), NULL },
|
|
{ "allow-reinstall", '\0', 0, G_OPTION_ARG_NONE, &allow_reinstall,
|
|
/* TRANSLATORS: command line option */
|
|
_("Allow reinstalling existing firmware versions"), NULL },
|
|
{ "allow-older", '\0', 0, G_OPTION_ARG_NONE, &allow_older,
|
|
/* TRANSLATORS: command line option */
|
|
_("Allow downgrading firmware versions"), NULL },
|
|
{ "force", '\0', 0, G_OPTION_ARG_NONE, &force,
|
|
/* TRANSLATORS: command line option */
|
|
_("Override warnings and force the action"), NULL },
|
|
{ "assume-yes", 'y', 0, G_OPTION_ARG_NONE, &priv->assume_yes,
|
|
/* TRANSLATORS: command line option */
|
|
_("Answer yes to all questions"), NULL },
|
|
{ "sign", '\0', 0, G_OPTION_ARG_NONE, &priv->sign,
|
|
/* TRANSLATORS: command line option */
|
|
_("Sign the uploaded data with the client certificate"), NULL },
|
|
{ "no-unreported-check", '\0', 0, G_OPTION_ARG_NONE, &priv->no_unreported_check,
|
|
/* TRANSLATORS: command line option */
|
|
_("Do not check for unreported history"), NULL },
|
|
{ "no-metadata-check", '\0', 0, G_OPTION_ARG_NONE, &priv->no_metadata_check,
|
|
/* TRANSLATORS: command line option */
|
|
_("Do not check for old metadata"), NULL },
|
|
{ "no-reboot-check", '\0', 0, G_OPTION_ARG_NONE, &priv->no_reboot_check,
|
|
/* TRANSLATORS: command line option */
|
|
_("Do not check for reboot after update"), NULL },
|
|
{ "no-safety-check", '\0', 0, G_OPTION_ARG_NONE, &priv->no_safety_check,
|
|
/* TRANSLATORS: command line option */
|
|
_("Do not perform device safety checks"), NULL },
|
|
{ "no-history", '\0', 0, G_OPTION_ARG_NONE, &no_history,
|
|
/* TRANSLATORS: command line option */
|
|
_("Do not write to the history database"), NULL },
|
|
{ "show-all-devices", '\0', 0, G_OPTION_ARG_NONE, &priv->show_all_devices,
|
|
/* TRANSLATORS: command line option */
|
|
_("Show devices that are not updatable"), NULL },
|
|
{ "disable-ssl-strict", '\0', 0, G_OPTION_ARG_NONE, &priv->disable_ssl_strict,
|
|
/* TRANSLATORS: command line option */
|
|
_("Ignore SSL strict checks when downloading files"), NULL },
|
|
{ "filter", '\0', 0, G_OPTION_ARG_STRING, &filter,
|
|
/* TRANSLATORS: command line option */
|
|
_("Filter with a set of device flags using a ~ prefix to "
|
|
"exclude, e.g. 'internal,~needs-reboot'"), NULL },
|
|
{ NULL}
|
|
};
|
|
|
|
setlocale (LC_ALL, "");
|
|
|
|
bindtextdomain (GETTEXT_PACKAGE, FWUPD_LOCALEDIR);
|
|
bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
|
|
textdomain (GETTEXT_PACKAGE);
|
|
|
|
/* ensure D-Bus errors are registered */
|
|
fwupd_error_quark ();
|
|
|
|
/* create helper object */
|
|
priv->loop = g_main_loop_new (NULL, FALSE);
|
|
priv->progressbar = fu_progressbar_new ();
|
|
|
|
/* add commands */
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"get-devices,get-topology",
|
|
NULL,
|
|
/* TRANSLATORS: command description */
|
|
_("Get all devices that support firmware updates"),
|
|
fu_util_get_devices);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"get-history",
|
|
NULL,
|
|
/* TRANSLATORS: command description */
|
|
_("Show history of firmware updates"),
|
|
fu_util_get_history);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"clear-history",
|
|
NULL,
|
|
/* TRANSLATORS: command description */
|
|
_("Erase all firmware update history"),
|
|
fu_util_clear_history);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"report-history",
|
|
NULL,
|
|
/* TRANSLATORS: command description */
|
|
_("Share firmware history with the developers"),
|
|
fu_util_report_history);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"install",
|
|
"FILE [DEVICE-ID]",
|
|
/* TRANSLATORS: command description */
|
|
_("Install a firmware file on this hardware"),
|
|
fu_util_install);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"get-details",
|
|
"FILE",
|
|
/* TRANSLATORS: command description */
|
|
_("Gets details about a firmware file"),
|
|
fu_util_get_details);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"get-updates,get-upgrades",
|
|
NULL,
|
|
/* TRANSLATORS: command description */
|
|
_("Gets the list of updates for connected hardware"),
|
|
fu_util_get_updates);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"update,upgrade",
|
|
NULL,
|
|
/* TRANSLATORS: command description */
|
|
_("Updates all firmware to latest versions available"),
|
|
fu_util_update);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"verify",
|
|
"[DEVICE-ID]",
|
|
/* TRANSLATORS: command description */
|
|
_("Checks cryptographic hash matches firmware"),
|
|
fu_util_verify);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"unlock",
|
|
"DEVICE-ID",
|
|
/* TRANSLATORS: command description */
|
|
_("Unlocks the device for firmware access"),
|
|
fu_util_unlock);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"clear-results",
|
|
"DEVICE-ID",
|
|
/* TRANSLATORS: command description */
|
|
_("Clears the results from the last update"),
|
|
fu_util_clear_results);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"clear-offline",
|
|
NULL,
|
|
/* TRANSLATORS: command description */
|
|
_("Clears any updates scheduled to be updated offline"),
|
|
fu_util_clear_offline);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"get-results",
|
|
"DEVICE-ID",
|
|
/* TRANSLATORS: command description */
|
|
_("Gets the results from the last update"),
|
|
fu_util_get_results);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"get-releases",
|
|
"[DEVICE-ID]",
|
|
/* TRANSLATORS: command description */
|
|
_("Gets the releases for a device"),
|
|
fu_util_get_releases);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"get-remotes",
|
|
NULL,
|
|
/* TRANSLATORS: command description */
|
|
_("Gets the configured remotes"),
|
|
fu_util_get_remotes);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"downgrade",
|
|
"[DEVICE-ID]",
|
|
/* TRANSLATORS: command description */
|
|
_("Downgrades the firmware on a device"),
|
|
fu_util_downgrade);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"refresh",
|
|
"[FILE FILE_SIG REMOTE-ID]",
|
|
/* TRANSLATORS: command description */
|
|
_("Refresh metadata from remote server"),
|
|
fu_util_refresh);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"verify-update",
|
|
"[DEVICE-ID]",
|
|
/* TRANSLATORS: command description */
|
|
_("Update the stored cryptographic hash with current ROM contents"),
|
|
fu_util_verify_update);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"modify-remote",
|
|
"REMOTE-ID KEY VALUE",
|
|
/* TRANSLATORS: command description */
|
|
_("Modifies a given remote"),
|
|
fu_util_remote_modify);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"enable-remote",
|
|
"REMOTE-ID",
|
|
/* TRANSLATORS: command description */
|
|
_("Enables a given remote"),
|
|
fu_util_remote_enable);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"disable-remote",
|
|
"REMOTE-ID",
|
|
/* TRANSLATORS: command description */
|
|
_("Disables a given remote"),
|
|
fu_util_remote_disable);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"activate",
|
|
"[DEVICE-ID]",
|
|
/* TRANSLATORS: command description */
|
|
_("Activate devices"),
|
|
fu_util_activate);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"get-approved-firmware",
|
|
NULL,
|
|
/* TRANSLATORS: firmware approved by the admin */
|
|
_("Gets the list of approved firmware."),
|
|
fu_util_get_approved_firmware);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"set-approved-firmware",
|
|
"CHECKSUM1[,CHECKSUM2][,CHECKSUM3]",
|
|
/* TRANSLATORS: firmware approved by the admin */
|
|
_("Sets the list of approved firmware."),
|
|
fu_util_set_approved_firmware);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"modify-config",
|
|
"KEY,VALUE",
|
|
/* TRANSLATORS: sets something in daemon.conf */
|
|
_("Modifies a daemon configuration value."),
|
|
fu_util_modify_config);
|
|
fu_util_cmd_array_add (cmd_array,
|
|
"reinstall",
|
|
"[DEVICE-ID]",
|
|
/* TRANSLATORS: command description */
|
|
_("Reinstall current firmware on the device."),
|
|
fu_util_reinstall);
|
|
|
|
|
|
/* do stuff on ctrl+c */
|
|
priv->cancellable = g_cancellable_new ();
|
|
#ifdef HAVE_GIO_UNIX
|
|
g_unix_signal_add_full (G_PRIORITY_DEFAULT,
|
|
SIGINT, fu_util_sigint_cb,
|
|
priv, NULL);
|
|
#endif
|
|
|
|
/* sort by command name */
|
|
fu_util_cmd_array_sort (cmd_array);
|
|
|
|
/* get a list of the commands */
|
|
priv->context = g_option_context_new (NULL);
|
|
cmd_descriptions = fu_util_cmd_array_to_string (cmd_array);
|
|
g_option_context_set_summary (priv->context, cmd_descriptions);
|
|
g_option_context_set_description (priv->context,
|
|
"This tool allows an administrator to query and control the "
|
|
"fwupd daemon, allowing them to perform actions such as "
|
|
"installing or downgrading firmware.");
|
|
|
|
/* TRANSLATORS: program name */
|
|
g_set_application_name (_("Firmware Utility"));
|
|
g_option_context_add_main_entries (priv->context, options, NULL);
|
|
ret = g_option_context_parse (priv->context, &argc, &argv, &error);
|
|
if (!ret) {
|
|
/* TRANSLATORS: the user didn't read the man page */
|
|
g_print ("%s: %s\n", _("Failed to parse arguments"),
|
|
error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
/* allow disabling SSL strict mode for broken corporate proxies */
|
|
if (priv->disable_ssl_strict) {
|
|
/* TRANSLATORS: try to help */
|
|
g_printerr ("%s\n", _("WARNING: Ignoring SSL strict checks, "
|
|
"to do this automatically in the future "
|
|
"export DISABLE_SSL_STRICT in your environment"));
|
|
g_setenv ("DISABLE_SSL_STRICT", "1", TRUE);
|
|
}
|
|
|
|
/* non-TTY consoles cannot answer questions */
|
|
if (isatty (fileno (stdout)) == 0) {
|
|
priv->no_unreported_check = TRUE;
|
|
priv->no_metadata_check = TRUE;
|
|
priv->no_reboot_check = TRUE;
|
|
priv->no_safety_check = TRUE;
|
|
fu_progressbar_set_interactive (priv->progressbar, FALSE);
|
|
}
|
|
|
|
/* parse filter flags */
|
|
if (filter != NULL) {
|
|
if (!fu_util_parse_filter_flags (filter,
|
|
&priv->filter_include,
|
|
&priv->filter_exclude,
|
|
&error)) {
|
|
/* TRANSLATORS: the user didn't read the man page */
|
|
g_print ("%s: %s\n", _("Failed to parse flags for --filter"),
|
|
error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
|
|
/* set verbose? */
|
|
if (verbose) {
|
|
g_setenv ("G_MESSAGES_DEBUG", "all", FALSE);
|
|
g_setenv ("FWUPD_VERBOSE", "1", FALSE);
|
|
} else {
|
|
g_log_set_handler (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG,
|
|
fu_util_ignore_cb, NULL);
|
|
}
|
|
|
|
/* set flags */
|
|
if (offline)
|
|
priv->flags |= FWUPD_INSTALL_FLAG_OFFLINE;
|
|
if (allow_reinstall)
|
|
priv->flags |= FWUPD_INSTALL_FLAG_ALLOW_REINSTALL;
|
|
if (allow_older)
|
|
priv->flags |= FWUPD_INSTALL_FLAG_ALLOW_OLDER;
|
|
if (force)
|
|
priv->flags |= FWUPD_INSTALL_FLAG_FORCE;
|
|
if (no_history)
|
|
priv->flags |= FWUPD_INSTALL_FLAG_NO_HISTORY;
|
|
|
|
/* connect to the daemon */
|
|
priv->client = fwupd_client_new ();
|
|
g_signal_connect (priv->client, "notify::percentage",
|
|
G_CALLBACK (fu_util_client_notify_cb), priv);
|
|
g_signal_connect (priv->client, "notify::status",
|
|
G_CALLBACK (fu_util_client_notify_cb), priv);
|
|
|
|
/* just show versions and exit */
|
|
if (version) {
|
|
g_autofree gchar *version_str = fu_util_get_versions();
|
|
g_print ("%s\n", version_str);
|
|
if (!fwupd_client_connect (priv->client, priv->cancellable, &error)) {
|
|
g_printerr ("Failed to connect to daemon: %s\n",
|
|
error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
g_print ("daemon version:\t%s\n",
|
|
fwupd_client_get_daemon_version (priv->client));
|
|
return EXIT_SUCCESS;
|
|
}
|
|
|
|
/* show a warning if the daemon is tainted */
|
|
if (!fwupd_client_connect (priv->client, priv->cancellable, &error)) {
|
|
g_printerr ("Failed to connect to daemon: %s\n",
|
|
error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
if (fwupd_client_get_tainted (priv->client)) {
|
|
g_printerr ("WARNING: The daemon has loaded 3rd party code and "
|
|
"is no longer supported by the upstream developers!\n");
|
|
}
|
|
|
|
/* check that we have at least this version daemon running */
|
|
if ((priv->flags & FWUPD_INSTALL_FLAG_FORCE) == 0 &&
|
|
!fu_util_check_daemon_version (priv, &error)) {
|
|
g_printerr ("%s\n", error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
#ifdef HAVE_SYSTEMD
|
|
/* make sure the correct daemon is in use */
|
|
if ((priv->flags & FWUPD_INSTALL_FLAG_FORCE) == 0 &&
|
|
!fwupd_client_get_daemon_interactive (priv->client) &&
|
|
!fu_util_using_correct_daemon (&error)) {
|
|
g_printerr ("%s\n", error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
#endif
|
|
|
|
/* make sure polkit actions were installed */
|
|
if (!fu_util_check_polkit_actions (&error)) {
|
|
g_printerr ("%s\n", error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
/* run the specified command */
|
|
ret = fu_util_cmd_array_run (cmd_array, priv, argv[1], (gchar**) &argv[2], &error);
|
|
if (!ret) {
|
|
ret = EXIT_FAILURE;
|
|
g_printerr ("%s\n", error->message);
|
|
if (g_error_matches (error, FWUPD_ERROR, FWUPD_ERROR_INVALID_ARGS))
|
|
fu_util_display_help (priv);
|
|
else if (g_error_matches (error, FWUPD_ERROR, FWUPD_ERROR_NOTHING_TO_DO))
|
|
ret = EXIT_NOTHING_TO_DO;
|
|
} else {
|
|
ret = EXIT_SUCCESS;
|
|
}
|
|
|
|
return ret;
|
|
}
|