fwupd/libfwupd/fwupd-report.c
Josh Soref 67deecde31
Lots of spelling fixes
Fixes:

* activate
* active
* additional
* and
* approaches
* attestation
* bootloader
* calculate
* capability
* children
* close
* compatible
* continuous
* convenience
* critical
* data
* delivery
* disabled
* disambiguate
* documented
* elapse
* emergency
* erasable
* expectations
* filesystem
* from
* haptic
* ignorable
* images
* infinity
* information
* information
* inhibited
* insufficient
* interrupt
* into
* limited
* management
* manifest
* maximum
* memory
* metadata
* mismatch
* model
* nonexistent
* not
* objects
* offset
* omissions
* ota
* past
* perform
* peripherals
* predictable
* product
* quarterly
* quirk
* quirks
* recognize
* release
* requests
* revocation
* sanitized
* sector
* status
* the
* the update
* timeout
* transfer
* transfers
* typically
* unspecified
* upgrade
* which
* will
* wireless
2022-12-29 13:57:31 +00:00

710 lines
17 KiB
C

/*
* Copyright (C) 2022 Richard Hughes <richard@hughsie.com>
*
* SPDX-License-Identifier: LGPL-2.1+
*/
#include "config.h"
#include <gio/gio.h>
#include <string.h>
#include "fwupd-common-private.h"
#include "fwupd-enums-private.h"
#include "fwupd-error.h"
#include "fwupd-report-private.h"
/**
* FwupdReport:
*
* A firmware report from a vendor.
*
* This is the LVFS formatted report that the fwupd user consumes, NOT the thing that gets uploaded.
*
* See also: [class@FwupdRelease]
*/
typedef struct {
guint64 created;
gchar *version_old;
gchar *vendor;
guint32 vendor_id;
gchar *device_name;
gchar *distro_id;
gchar *distro_version;
GHashTable *metadata;
gchar *distro_variant;
} FwupdReportPrivate;
enum { PROP_0, PROP_REMOTE_ID, PROP_LAST };
G_DEFINE_TYPE_WITH_PRIVATE(FwupdReport, fwupd_report, G_TYPE_OBJECT)
#define GET_PRIVATE(o) (fwupd_report_get_instance_private(o))
/**
* fwupd_report_get_created:
* @self: a #FwupdReport
*
* Gets when the report was created.
*
* Returns: UTC timestamp in UNIX format, or 0 if unset
*
* Since: 1.8.8
**/
guint64
fwupd_report_get_created(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), 0);
return priv->created;
}
/**
* fwupd_report_set_created:
* @self: a #FwupdReport
* @created: UTC timestamp in UNIX format
*
* Sets when the report was created.
*
* Since: 1.8.8
**/
void
fwupd_report_set_created(FwupdReport *self, guint64 created)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_if_fail(FWUPD_IS_REPORT(self));
priv->created = created;
}
/**
* fwupd_report_get_version_old:
* @self: a #FwupdReport
*
* Gets the old version, i.e. what the upser was upgrading *from*.
*
* Returns: the version, or %NULL if unset
*
* Since: 1.8.8
**/
const gchar *
fwupd_report_get_version_old(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
return priv->version_old;
}
/**
* fwupd_report_set_version_old:
* @self: a #FwupdReport
* @version_old: (nullable): the version, e.g. `1.2.3`
*
* Sets the old version, i.e. what the upser was upgrading *from*.
*
* Since: 1.8.8
**/
void
fwupd_report_set_version_old(FwupdReport *self, const gchar *version_old)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_if_fail(FWUPD_IS_REPORT(self));
/* not changed */
if (g_strcmp0(priv->version_old, version_old) == 0)
return;
g_free(priv->version_old);
priv->version_old = g_strdup(version_old);
}
/**
* fwupd_report_get_vendor:
* @self: a #FwupdReport
*
* Gets the vendor that uploaded the test result.
*
* Returns: the test vendor, or %NULL if unset
*
* Since: 1.8.8
**/
const gchar *
fwupd_report_get_vendor(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
return priv->vendor;
}
/**
* fwupd_report_set_vendor:
* @self: a #FwupdReport
* @vendor: (nullable): the vendor name
*
* Sets the vendor that uploaded the test result.
*
* Since: 1.8.8
**/
void
fwupd_report_set_vendor(FwupdReport *self, const gchar *vendor)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_if_fail(FWUPD_IS_REPORT(self));
/* not changed */
if (g_strcmp0(priv->vendor, vendor) == 0)
return;
g_free(priv->vendor);
priv->vendor = g_strdup(vendor);
}
/**
* fwupd_report_get_vendor_id:
* @self: a #FwupdReport
*
* Gets the vendor identifier. The mapping is only known on the remote server, and this can be
* useful to filter on different QA teams that work for the same OEM.
*
* Returns: the vendor ID, or 0 if unset
*
* Since: 1.8.8
**/
guint32
fwupd_report_get_vendor_id(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), 0);
return priv->vendor_id;
}
/**
* fwupd_report_set_vendor_id:
* @self: a #FwupdReport
* @vendor_id: the vendor ID, or 0
*
* Sets the vendor identifier. The mapping is only known on the remote server, and this can be
* useful to filter on different QA teams that work for the same OEM.
*
* Since: 1.8.8
**/
void
fwupd_report_set_vendor_id(FwupdReport *self, guint32 vendor_id)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_if_fail(FWUPD_IS_REPORT(self));
priv->vendor_id = vendor_id;
}
/**
* fwupd_report_get_device_name:
* @self: a #FwupdReport
*
* Gets the name of the device the update was performed on.
*
* Returns: the name, or %NULL if unset
*
* Since: 1.8.8
**/
const gchar *
fwupd_report_get_device_name(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
return priv->device_name;
}
/**
* fwupd_report_set_device_name:
* @self: a #FwupdReport
* @device_name: (nullable): the name, e.g. `LENOVO ThinkPad P1 Gen 3`
*
* Sets the name of the device the update was performed on.
*
* Since: 1.8.8
**/
void
fwupd_report_set_device_name(FwupdReport *self, const gchar *device_name)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_if_fail(FWUPD_IS_REPORT(self));
/* not changed */
if (g_strcmp0(priv->device_name, device_name) == 0)
return;
g_free(priv->device_name);
priv->device_name = g_strdup(device_name);
}
/**
* fwupd_report_get_distro_id:
* @self: a #FwupdReport
*
* Gets the distribution name.
*
* Returns: the name, or %NULL if unset
*
* Since: 1.8.8
**/
const gchar *
fwupd_report_get_distro_id(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
return priv->distro_id;
}
/**
* fwupd_report_set_distro_id:
* @self: a #FwupdReport
* @distro_id: (nullable): the name, e.g. `fedora`
*
* Sets the distribution name.
*
* Since: 1.8.8
**/
void
fwupd_report_set_distro_id(FwupdReport *self, const gchar *distro_id)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_if_fail(FWUPD_IS_REPORT(self));
/* not changed */
if (g_strcmp0(priv->distro_id, distro_id) == 0)
return;
g_free(priv->distro_id);
priv->distro_id = g_strdup(distro_id);
}
/**
* fwupd_report_get_distro_variant:
* @self: a #FwupdReport
*
* Gets the distribution variant.
*
* Returns: variant, or %NULL if unset
*
* Since: 1.8.8
**/
const gchar *
fwupd_report_get_distro_variant(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
return priv->distro_variant;
}
/**
* fwupd_report_set_distro_variant:
* @self: a #FwupdReport
* @distro_variant: (nullable): the variant, e.g. `workstation`
*
* Sets the distribution variant.
*
* Since: 1.8.8
**/
void
fwupd_report_set_distro_variant(FwupdReport *self, const gchar *distro_variant)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_if_fail(FWUPD_IS_REPORT(self));
/* not changed */
if (g_strcmp0(priv->distro_variant, distro_variant) == 0)
return;
g_free(priv->distro_variant);
priv->distro_variant = g_strdup(distro_variant);
}
/**
* fwupd_report_get_distro_version:
* @self: a #FwupdReport
*
* Gets the distribution version.
*
* Returns: a string, or %NULL if unset
*
* Since: 1.8.8
**/
const gchar *
fwupd_report_get_distro_version(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
return priv->distro_version;
}
/**
* fwupd_report_set_distro_version:
* @self: a #FwupdReport
* @distro_version: (nullable): a string
*
* Sets the distribution version.
*
* Since: 1.8.8
**/
void
fwupd_report_set_distro_version(FwupdReport *self, const gchar *distro_version)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_if_fail(FWUPD_IS_REPORT(self));
/* not changed */
if (g_strcmp0(priv->distro_version, distro_version) == 0)
return;
g_free(priv->distro_version);
priv->distro_version = g_strdup(distro_version);
}
/**
* fwupd_report_get_metadata:
* @self: a #FwupdReport
*
* Gets the report metadata.
*
* Returns: (transfer none): the metadata, which may be empty
*
* Since: 1.8.8
**/
GHashTable *
fwupd_report_get_metadata(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
return priv->metadata;
}
/**
* fwupd_report_add_metadata_item:
* @self: a #FwupdReport
* @key: (not nullable): the key
* @value: (not nullable): the value
*
* Sets a report metadata item.
*
* Since: 1.8.8
**/
void
fwupd_report_add_metadata_item(FwupdReport *self, const gchar *key, const gchar *value)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_if_fail(FWUPD_IS_REPORT(self));
g_return_if_fail(key != NULL);
g_return_if_fail(value != NULL);
g_hash_table_insert(priv->metadata, g_strdup(key), g_strdup(value));
}
/**
* fwupd_report_get_metadata_item:
* @self: a #FwupdReport
* @key: (not nullable): the key
*
* Gets a report metadata item.
*
* Returns: the value, or %NULL if unset
*
* Since: 1.8.8
**/
const gchar *
fwupd_report_get_metadata_item(FwupdReport *self, const gchar *key)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
g_return_val_if_fail(key != NULL, NULL);
return g_hash_table_lookup(priv->metadata, key);
}
/**
* fwupd_report_to_variant:
* @self: a #FwupdReport
*
* Serialize the report data.
*
* Returns: the serialized data, or %NULL for error
*
* Since: 1.8.8
**/
GVariant *
fwupd_report_to_variant(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
GVariantBuilder builder;
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
/* create an array with all the metadata in */
g_variant_builder_init(&builder, G_VARIANT_TYPE_VARDICT);
if (priv->distro_id != NULL) {
g_variant_builder_add(&builder,
"{sv}",
FWUPD_RESULT_KEY_DISTRO_ID,
g_variant_new_string(priv->distro_id));
}
if (priv->distro_variant != NULL) {
g_variant_builder_add(&builder,
"{sv}",
FWUPD_RESULT_KEY_DISTRO_VARIANT,
g_variant_new_string(priv->distro_variant));
}
if (priv->distro_version != NULL) {
g_variant_builder_add(&builder,
"{sv}",
FWUPD_RESULT_KEY_DISTRO_VERSION,
g_variant_new_string(priv->distro_version));
}
if (priv->vendor != NULL) {
g_variant_builder_add(&builder,
"{sv}",
FWUPD_RESULT_KEY_VENDOR,
g_variant_new_string(priv->vendor));
}
if (priv->device_name != NULL) {
g_variant_builder_add(&builder,
"{sv}",
FWUPD_RESULT_KEY_DEVICE_NAME,
g_variant_new_string(priv->device_name));
}
if (priv->created != 0) {
g_variant_builder_add(&builder,
"{sv}",
FWUPD_RESULT_KEY_CREATED,
g_variant_new_uint64(priv->created));
}
if (priv->version_old != NULL) {
g_variant_builder_add(&builder,
"{sv}",
FWUPD_RESULT_KEY_VERSION_OLD,
g_variant_new_string(priv->version_old));
}
if (priv->vendor_id > 0) {
g_variant_builder_add(&builder,
"{sv}",
FWUPD_RESULT_KEY_VENDOR_ID,
g_variant_new_uint32(priv->vendor_id));
}
if (g_hash_table_size(priv->metadata) > 0) {
g_variant_builder_add(&builder,
"{sv}",
FWUPD_RESULT_KEY_METADATA,
fwupd_hash_kv_to_variant(priv->metadata));
}
return g_variant_new("a{sv}", &builder);
}
static void
fwupd_report_from_key_value(FwupdReport *self, const gchar *key, GVariant *value)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
if (g_strcmp0(key, FWUPD_RESULT_KEY_DISTRO_ID) == 0) {
fwupd_report_set_distro_id(self, g_variant_get_string(value, NULL));
return;
}
if (g_strcmp0(key, FWUPD_RESULT_KEY_DISTRO_VARIANT) == 0) {
fwupd_report_set_distro_variant(self, g_variant_get_string(value, NULL));
return;
}
if (g_strcmp0(key, FWUPD_RESULT_KEY_DISTRO_VERSION) == 0) {
fwupd_report_set_distro_version(self, g_variant_get_string(value, NULL));
return;
}
if (g_strcmp0(key, FWUPD_RESULT_KEY_VENDOR) == 0) {
fwupd_report_set_vendor(self, g_variant_get_string(value, NULL));
return;
}
if (g_strcmp0(key, FWUPD_RESULT_KEY_VENDOR_ID) == 0) {
fwupd_report_set_vendor_id(self, g_variant_get_uint32(value));
return;
}
if (g_strcmp0(key, FWUPD_RESULT_KEY_DEVICE_NAME) == 0) {
fwupd_report_set_device_name(self, g_variant_get_string(value, NULL));
return;
}
if (g_strcmp0(key, FWUPD_RESULT_KEY_CREATED) == 0) {
fwupd_report_set_created(self, g_variant_get_uint64(value));
return;
}
if (g_strcmp0(key, FWUPD_RESULT_KEY_VERSION_OLD) == 0) {
fwupd_report_set_version_old(self, g_variant_get_string(value, NULL));
return;
}
if (g_strcmp0(key, FWUPD_RESULT_KEY_METADATA) == 0) {
g_hash_table_unref(priv->metadata);
priv->metadata = fwupd_variant_to_hash_kv(value);
return;
}
}
/**
* fwupd_report_to_json:
* @self: a #FwupdReport
* @builder: a JSON builder
*
* Adds a fwupd report to a JSON builder
*
* Since: 1.8.8
**/
void
fwupd_report_to_json(FwupdReport *self, JsonBuilder *builder)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_autoptr(GList) keys = NULL;
g_return_if_fail(FWUPD_IS_REPORT(self));
g_return_if_fail(builder != NULL);
fwupd_common_json_add_string(builder, FWUPD_RESULT_KEY_DEVICE_NAME, priv->device_name);
fwupd_common_json_add_string(builder, FWUPD_RESULT_KEY_DISTRO_ID, priv->distro_id);
fwupd_common_json_add_string(builder,
FWUPD_RESULT_KEY_DISTRO_VARIANT,
priv->distro_variant);
fwupd_common_json_add_string(builder,
FWUPD_RESULT_KEY_DISTRO_VERSION,
priv->distro_version);
fwupd_common_json_add_string(builder, FWUPD_RESULT_KEY_VERSION_OLD, priv->version_old);
fwupd_common_json_add_string(builder, FWUPD_RESULT_KEY_VENDOR, priv->vendor);
if (priv->vendor_id > 0)
fwupd_common_json_add_int(builder, FWUPD_RESULT_KEY_VENDOR_ID, priv->vendor_id);
/* metadata */
keys = g_hash_table_get_keys(priv->metadata);
for (GList *l = keys; l != NULL; l = l->next) {
const gchar *key = l->data;
const gchar *value = g_hash_table_lookup(priv->metadata, key);
fwupd_common_json_add_string(builder, key, value);
}
}
/**
* fwupd_report_to_string:
* @self: a #FwupdReport
*
* Builds a text representation of the object.
*
* Returns: text, or %NULL for invalid
*
* Since: 1.8.8
**/
gchar *
fwupd_report_to_string(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
GString *str;
g_autoptr(GList) keys = NULL;
g_return_val_if_fail(FWUPD_IS_REPORT(self), NULL);
str = g_string_new("");
fwupd_pad_kv_str(str, FWUPD_RESULT_KEY_DEVICE_NAME, priv->device_name);
fwupd_pad_kv_str(str, FWUPD_RESULT_KEY_DISTRO_ID, priv->distro_id);
fwupd_pad_kv_str(str, FWUPD_RESULT_KEY_DISTRO_VARIANT, priv->distro_variant);
fwupd_pad_kv_str(str, FWUPD_RESULT_KEY_DISTRO_VERSION, priv->distro_version);
fwupd_pad_kv_str(str, FWUPD_RESULT_KEY_VERSION_OLD, priv->version_old);
fwupd_pad_kv_str(str, FWUPD_RESULT_KEY_VENDOR, priv->vendor);
fwupd_pad_kv_int(str, FWUPD_RESULT_KEY_VENDOR_ID, priv->vendor_id);
/* metadata */
keys = g_hash_table_get_keys(priv->metadata);
for (GList *l = keys; l != NULL; l = l->next) {
const gchar *key = l->data;
const gchar *value = g_hash_table_lookup(priv->metadata, key);
fwupd_pad_kv_str(str, key, value);
}
return g_string_free(str, FALSE);
}
static void
fwupd_report_init(FwupdReport *self)
{
FwupdReportPrivate *priv = GET_PRIVATE(self);
priv->metadata = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free);
}
static void
fwupd_report_finalize(GObject *object)
{
FwupdReport *self = FWUPD_REPORT(object);
FwupdReportPrivate *priv = GET_PRIVATE(self);
g_free(priv->vendor);
g_free(priv->device_name);
g_free(priv->distro_id);
g_free(priv->distro_version);
g_free(priv->distro_variant);
g_free(priv->version_old);
g_hash_table_unref(priv->metadata);
G_OBJECT_CLASS(fwupd_report_parent_class)->finalize(object);
}
static void
fwupd_report_class_init(FwupdReportClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS(klass);
object_class->finalize = fwupd_report_finalize;
}
static void
fwupd_report_set_from_variant_iter(FwupdReport *self, GVariantIter *iter)
{
GVariant *value;
const gchar *key;
while (g_variant_iter_next(iter, "{&sv}", &key, &value)) {
fwupd_report_from_key_value(self, key, value);
g_variant_unref(value);
}
}
/**
* fwupd_report_from_variant:
* @value: (not nullable): the serialized data
*
* Creates a new report using serialized data.
*
* Returns: (transfer full): a new #FwupdReport, or %NULL if @value was invalid
*
* Since: 1.8.8
**/
FwupdReport *
fwupd_report_from_variant(GVariant *value)
{
FwupdReport *self = NULL;
const gchar *type_string;
g_autoptr(GVariantIter) iter = NULL;
/* format from GetDetails */
type_string = g_variant_get_type_string(value);
if (g_strcmp0(type_string, "(a{sv})") == 0) {
self = fwupd_report_new();
g_variant_get(value, "(a{sv})", &iter);
fwupd_report_set_from_variant_iter(self, iter);
} else if (g_strcmp0(type_string, "a{sv}") == 0) {
self = fwupd_report_new();
g_variant_get(value, "a{sv}", &iter);
fwupd_report_set_from_variant_iter(self, iter);
} else {
g_warning("type %s not known", type_string);
}
return self;
}
/**
* fwupd_report_new:
*
* Creates a new report.
*
* Returns: a new #FwupdReport
*
* Since: 1.8.8
**/
FwupdReport *
fwupd_report_new(void)
{
FwupdReport *self;
self = g_object_new(FWUPD_TYPE_REPORT, NULL);
return FWUPD_REPORT(self);
}