| /* |
| * Copyright 2015 Richard Hughes <richard@hughsie.com> |
| * |
| * SPDX-License-Identifier: LGPL-2.1-or-later |
| */ |
| |
| #include "config.h" |
| |
| #include <fwupdplugin.h> |
| |
| #include <fcntl.h> |
| #include <glib/gstdio.h> |
| #include <string.h> |
| |
| #include "fwupd-remote-private.h" |
| |
| #include "../plugins/test/fu-test-plugin.h" |
| #include "fu-bios-settings-private.h" |
| #include "fu-cabinet.h" |
| #include "fu-client-list.h" |
| #include "fu-config-private.h" |
| #include "fu-console.h" |
| #include "fu-context-private.h" |
| #include "fu-device-list.h" |
| #include "fu-device-private.h" |
| #include "fu-drm-device-private.h" |
| #include "fu-efivars-private.h" |
| #include "fu-engine-config.h" |
| #include "fu-engine-helper.h" |
| #include "fu-engine-requirements.h" |
| #include "fu-engine.h" |
| #include "fu-history.h" |
| #include "fu-idle.h" |
| #include "fu-plugin-list.h" |
| #include "fu-plugin-private.h" |
| #include "fu-release-common.h" |
| #include "fu-remote-list.h" |
| #include "fu-remote.h" |
| #include "fu-security-attrs-private.h" |
| #include "fu-usb-backend.h" |
| |
| #ifdef HAVE_GIO_UNIX |
| #include "fu-unix-seekable-input-stream.h" |
| #endif |
| |
| #pragma GCC diagnostic ignored "-Wanalyzer-null-argument" |
| |
| typedef struct { |
| FuPlugin *plugin; |
| FuContext *ctx; |
| } FuTest; |
| |
| /* nocheck:static */ |
| static GMainLoop *_test_loop = NULL; |
| static guint _test_loop_timeout_id = 0; |
| |
| static gboolean |
| fu_test_hang_check_cb(gpointer user_data) |
| { |
| g_main_loop_quit(_test_loop); |
| _test_loop_timeout_id = 0; |
| return G_SOURCE_REMOVE; |
| } |
| |
| static void |
| fu_test_loop_run_with_timeout(guint timeout_ms) |
| { |
| g_assert_cmpint(_test_loop_timeout_id, ==, 0); |
| g_assert_null(_test_loop); |
| _test_loop = g_main_loop_new(NULL, FALSE); |
| _test_loop_timeout_id = g_timeout_add(timeout_ms, fu_test_hang_check_cb, NULL); |
| g_main_loop_run(_test_loop); |
| } |
| |
| static void |
| fu_test_loop_quit(void) |
| { |
| if (_test_loop_timeout_id > 0) { |
| g_source_remove(_test_loop_timeout_id); |
| _test_loop_timeout_id = 0; |
| } |
| if (_test_loop != NULL) { |
| g_main_loop_quit(_test_loop); |
| g_main_loop_unref(_test_loop); |
| _test_loop = NULL; |
| } |
| } |
| |
| static void |
| fu_self_test_mkroot(void) |
| { |
| if (g_file_test("/tmp/fwupd-self-test", G_FILE_TEST_EXISTS)) { |
| g_autoptr(GError) error = NULL; |
| if (!fu_path_rmtree("/tmp/fwupd-self-test", &error)) |
| g_warning("failed to mkroot: %s", error->message); |
| } |
| g_assert_cmpint(g_mkdir_with_parents("/tmp/fwupd-self-test/var/lib/fwupd", 0755), ==, 0); |
| } |
| |
| static void |
| fu_test_copy_file(const gchar *source, const gchar *target) |
| { |
| gboolean ret; |
| g_autofree gchar *data = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| g_debug("copying %s to %s", source, target); |
| ret = g_file_get_contents(source, &data, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = g_file_set_contents(target, data, -1, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_test_free(FuTest *self) |
| { |
| if (self->ctx != NULL) |
| g_object_unref(self->ctx); |
| g_free(self); |
| } |
| |
| #pragma clang diagnostic push |
| #pragma clang diagnostic ignored "-Wunused-function" |
| G_DEFINE_AUTOPTR_CLEANUP_FUNC(FuTest, fu_test_free) |
| #pragma clang diagnostic pop |
| |
| static void |
| fu_client_list_func(void) |
| { |
| g_autoptr(FuClient) client_find = NULL; |
| g_autoptr(FuClient) client = NULL; |
| g_autoptr(FuClient) client_orig = NULL; |
| g_autoptr(FuClientList) client_list = fu_client_list_new(NULL); |
| g_autoptr(GPtrArray) clients_empty = NULL; |
| g_autoptr(GPtrArray) clients_full = NULL; |
| |
| /* ensure empty */ |
| clients_empty = fu_client_list_get_all(client_list); |
| g_assert_cmpint(clients_empty->len, ==, 0); |
| |
| /* register a client, then find it */ |
| client_orig = fu_client_list_register(client_list, ":hello"); |
| g_assert_nonnull(client_orig); |
| client_find = fu_client_list_get_by_sender(client_list, ":hello"); |
| g_assert_nonnull(client_find); |
| g_assert_true(client_orig == client_find); |
| clients_full = fu_client_list_get_all(client_list); |
| g_assert_cmpint(clients_full->len, ==, 1); |
| |
| /* register a duplicate, check properties */ |
| client = fu_client_list_register(client_list, ":hello"); |
| g_assert_nonnull(client); |
| g_assert_true(client_orig == client); |
| g_assert_cmpstr(fu_client_get_sender(client), ==, ":hello"); |
| g_assert_cmpint(fu_client_get_feature_flags(client), ==, FWUPD_FEATURE_FLAG_NONE); |
| g_assert_cmpstr(fu_client_lookup_hint(client, "key"), ==, NULL); |
| g_assert_true(fu_client_has_flag(client, FU_CLIENT_FLAG_ACTIVE)); |
| fu_client_insert_hint(client, "key", "value"); |
| fu_client_set_feature_flags(client, FWUPD_FEATURE_FLAG_UPDATE_ACTION); |
| g_assert_cmpstr(fu_client_lookup_hint(client, "key"), ==, "value"); |
| g_assert_cmpint(fu_client_get_feature_flags(client), ==, FWUPD_FEATURE_FLAG_UPDATE_ACTION); |
| |
| /* emulate disconnect */ |
| fu_client_remove_flag(client, FU_CLIENT_FLAG_ACTIVE); |
| g_assert_false(fu_client_has_flag(client, FU_CLIENT_FLAG_ACTIVE)); |
| } |
| |
| static void |
| fu_idle_func(void) |
| { |
| guint token; |
| g_autoptr(FuIdle) idle = fu_idle_new(); |
| |
| fu_idle_reset(idle); |
| g_assert_false(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_TIMEOUT)); |
| g_assert_false(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_SIGNALS)); |
| |
| token = fu_idle_inhibit(idle, FU_IDLE_INHIBIT_TIMEOUT | FU_IDLE_INHIBIT_SIGNALS, NULL); |
| g_assert_true(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_TIMEOUT)); |
| g_assert_true(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_SIGNALS)); |
| |
| /* wrong token */ |
| fu_idle_uninhibit(idle, token + 1); |
| g_assert_true(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_SIGNALS)); |
| |
| /* correct token */ |
| fu_idle_uninhibit(idle, token); |
| g_assert_false(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_TIMEOUT)); |
| g_assert_false(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_SIGNALS)); |
| |
| /* locker section */ |
| { |
| g_autoptr(FuIdleLocker) idle_locker1 = |
| fu_idle_locker_new(idle, FU_IDLE_INHIBIT_TIMEOUT, NULL); |
| g_autoptr(FuIdleLocker) idle_locker2 = |
| fu_idle_locker_new(idle, FU_IDLE_INHIBIT_SIGNALS, NULL); |
| g_assert_nonnull(idle_locker1); |
| g_assert_nonnull(idle_locker2); |
| g_assert_true(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_TIMEOUT)); |
| g_assert_true(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_SIGNALS)); |
| } |
| g_assert_false(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_TIMEOUT)); |
| g_assert_false(fu_idle_has_inhibit(idle, FU_IDLE_INHIBIT_SIGNALS)); |
| } |
| |
| static void |
| fu_engine_generate_md_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| const gchar *tmp; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GBytes) data = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| |
| /* put cab file somewhere we can parse it */ |
| filename = g_test_build_filename(G_TEST_BUILT, |
| "..", |
| "libfwupdplugin", |
| "tests", |
| "colorhug", |
| "colorhug-als-3.0.2.cab", |
| NULL); |
| data = fu_bytes_get_contents(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(data); |
| ret = fu_bytes_set_contents("/tmp/fwupd-self-test/var/cache/fwupd/foo.cab", data, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* load engine and check the device was found */ |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_REMOTES | FU_ENGINE_LOAD_FLAG_NO_CACHE, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.3"); |
| component = fu_engine_get_component_by_guids(engine, device); |
| g_assert_nonnull(component); |
| |
| /* check remote ID set */ |
| tmp = xb_node_query_text(component, "../custom/value[@key='fwupd::RemoteId']", NULL); |
| g_assert_cmpstr(tmp, ==, "directory"); |
| |
| /* verify checksums */ |
| tmp = xb_node_query_text(component, "releases/release/checksum[@target='container']", NULL); |
| g_assert_cmpstr(tmp, ==, "71aefb2a9b412833d8c519d5816ef4c5668e5e76"); |
| tmp = xb_node_query_text(component, "releases/release/checksum[@target='content']", NULL); |
| g_assert_cmpstr(tmp, ==, NULL); |
| } |
| |
| static void |
| fu_engine_requirements_missing_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <id compare=\"ge\" version=\"1.2.3\">not.going.to.exist</id>" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy version */ |
| fu_engine_add_runtime_version(engine, "org.test.dummy", "1.2.3"); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_soft_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <suggests>" |
| " <id compare=\"ge\" version=\"1.2.3\">not.going.to.exist</id>" |
| " </suggests>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy version */ |
| fu_engine_add_runtime_version(engine, "org.test.dummy", "1.2.3"); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, |
| release, |
| FWUPD_INSTALL_FLAG_IGNORE_REQUIREMENTS, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_client_fail_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <client>detach-action</client>" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_client_invalid_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <client>hello-dave</client>" |
| " <id compare=\"ge\" version=\"1.4.5\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_client_pass_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <client>detach-action</client>" |
| " <id compare=\"ge\" version=\"1.4.5\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy version */ |
| fu_engine_request_set_feature_flags(request, FWUPD_FEATURE_FLAG_DETACH_ACTION); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_vercmp_glob_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <client>id-requirement-glob</client>" |
| " <id compare=\"ge\" " |
| "version=\"1.8.*=1.8.5|1.9.*=1.9.7|2.0.13\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* hardcode to specific branch */ |
| fu_context_add_runtime_version(self->ctx, "org.freedesktop.fwupd", "1.9.8"); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* reset back to reality */ |
| fu_context_add_runtime_version(self->ctx, "org.freedesktop.fwupd", VERSION); |
| } |
| |
| static void |
| fu_engine_requirements_vercmp_glob_fallback_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <id compare=\"ge\" " |
| "version=\"1.8.*=1.8.5|1.9.*=1.9.7|2.0.13\">org.freedesktop.fwupd</id>\n" |
| " <client>id-requirement-glob</client>" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_not_hardware_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <not_hardware>ffffffff-ffff-ffff-ffff-ffffffffffff</not_hardware>" |
| " <id compare=\"ge\" version=\"1.9.10\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy version */ |
| fu_engine_request_set_feature_flags(request, FWUPD_FEATURE_FLAG_DETACH_ACTION); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_phased_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(FwupdRemote) remote = fwupd_remote_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <phased_update>10</phased_update>" |
| " <id compare=\"ge\" version=\"2.0.17\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* do not include into seed */ |
| g_assert_cmpstr(fu_engine_get_host_machine_id(engine), ==, NULL); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_request(release, request); |
| fu_release_set_remote(release, remote); |
| fwupd_remote_set_mtime(remote, 12340); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check this still passes as we're ignoring */ |
| fwupd_remote_set_mtime(remote, 12345); |
| ret = fu_engine_requirements_check(engine, |
| release, |
| FWUPD_INSTALL_FLAG_IGNORE_REQUIREMENTS, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check this now fails */ |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_false(ret); |
| g_clear_error(&error); |
| |
| /* user disabled this */ |
| fwupd_remote_add_flag(remote, FWUPD_REMOTE_FLAG_NO_PHASED_UPDATES); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_phased_old_fwupd_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(FwupdRemote) remote = fwupd_remote_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <phased_update>10</phased_update>" |
| " <id compare=\"ge\" version=\"2.0.16\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* do not include into seed */ |
| g_assert_cmpstr(fu_engine_get_host_machine_id(engine), ==, NULL); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_request(release, request); |
| fu_release_set_remote(release, remote); |
| fwupd_remote_set_mtime(remote, 12340); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check this fails because the fwupd requirement is too low */ |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_version_require_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<component>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.4\">" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy device */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_set_version_bootloader(device, "4.5.6"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_VERSION_CHECK_REQUIRED); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_true( |
| g_str_has_prefix(error->message, "device requires firmware with a version check")); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_version_lowest_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<component>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.2\">" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy device */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_set_version_lowest(device, "1.2.3"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_release_check_version(release, component, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_true( |
| g_str_has_prefix(error->message, "Specified firmware is older than the minimum")); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_unsupported_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <UNKNOWN compare=\"ge\" version=\"2.6.0\"/>" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy version */ |
| fu_engine_add_runtime_version(engine, "org.test.dummy", "1.2.3"); |
| |
| /* make the component require one thing that we don't support */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_child_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) child = fu_device_new(NULL); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<component>" |
| " <requires>" |
| " <firmware compare=\"eq\" version=\"0.0.1\">not-child</firmware>" |
| " <id compare=\"ge\" version=\"1.2.11\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.4\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy device */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_set_version_bootloader(device, "4.5.6"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| |
| fu_device_set_version_format(child, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(child, "0.0.999"); |
| fu_device_set_physical_id(child, "dummy"); |
| fu_device_add_child(device, child); |
| |
| /* make the component require three things */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_child_fail_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) child = fu_device_new(NULL); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<component>" |
| " <requires>" |
| " <firmware compare=\"glob\" version=\"0.0.*\">not-child</firmware>" |
| " <id compare=\"ge\" version=\"1.2.11\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.4\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy device */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_set_version_bootloader(device, "4.5.6"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| |
| fu_device_set_version_format(child, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(child, "0.0.1"); |
| fu_device_set_physical_id(child, "dummy"); |
| fu_device_add_child(device, child); |
| |
| /* make the component require three things */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_nonnull( |
| g_strstr_len(error->message, -1, "Not compatible with child device version")); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = "<component>" |
| " <requires>" |
| " <id compare=\"ge\" version=\"1.2.3\">org.test.dummy</id>" |
| " <hardware>6ff95c9c-ae41-5f59-9d90-3ec1ea66091e</hardware>" |
| " <id compare=\"ge\" version=\"1.0.1\">org.freedesktop.fwupd</id>\n" |
| " <id compare=\"ge\" version=\"1.9.10\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up some dummy versions */ |
| fu_engine_add_runtime_version(engine, "org.test.dummy", "1.2.3"); |
| fu_engine_add_runtime_version(engine, "com.hughski.colorhug", "7.8.9"); |
| |
| /* make the component require one thing */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_device_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<component>" |
| " <requires>" |
| " <firmware compare=\"ge\" version=\"1.2.3\"/>" |
| " <firmware compare=\"eq\" version=\"4.5.6\">bootloader</firmware>" |
| " <firmware compare=\"regex\" version=\"USB:0xFFFF|DMI:Lenovo\">vendor-id</firmware>" |
| #ifdef __linux__ |
| " <id compare=\"ge\" version=\"4.0.0\">org.kernel</id>" |
| #endif |
| " <id compare=\"ge\" version=\"1.2.11\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.4\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy device */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_set_version_bootloader(device, "4.5.6"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_build_vendor_id_u16(device, "PCI", 0x0000); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_VERSION_CHECK_REQUIRED); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| |
| /* make the component require three things */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check this fails, as the wrong requirement is specified */ |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_ENFORCE_REQUIRES); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_nonnull(g_strstr_len(error->message, -1, "child, parent or sibling requirement")); |
| g_assert_false(ret); |
| |
| #ifndef SUPPORTED_BUILD |
| /* we can force this */ |
| g_clear_error(&error); |
| ret = fu_engine_requirements_check(engine, |
| release, |
| FWUPD_INSTALL_FLAG_IGNORE_REQUIREMENTS, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| #endif |
| } |
| |
| static void |
| fu_engine_requirements_device_plain_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<component>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"51H0AALB\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " <checksum type=\"sha1\" filename=\"bios.cab\" target=\"container\"/>" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy device */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_PLAIN); |
| fu_device_set_version(device, "5101AALB"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| |
| /* make the component require three things */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpstr(fu_release_get_filename(release), ==, "bios.cab"); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_version_format_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<component>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.4\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " </release>" |
| " </releases>" |
| " <custom>" |
| " <value key=\"LVFS::VersionFormat\">triplet</value>" |
| " </custom>" |
| "</component>"; |
| |
| /* set up a dummy device */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_QUAD); |
| fu_device_set_version(device, "1.2.3.4"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| |
| /* make the component require three things */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_release_check_version(release, component, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_nonnull( |
| g_strstr_len(error->message, -1, "Firmware version formats were different")); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_only_upgrade_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<component>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy device */ |
| fu_device_set_version(device, "1.2.4"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_ONLY_VERSION_UPGRADE); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| |
| /* make the component require three things */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_release_check_version(release, component, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_nonnull(g_strstr_len(error->message, -1, "Device only supports version upgrades")); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_requirements_only_upgrade_reinstall_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<component>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.3\"/>" |
| " </releases>" |
| "</component>"; |
| |
| /* set up a dummy device */ |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_ONLY_VERSION_UPGRADE); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| |
| /* make the component require three things */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_release_check_version(release, component, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_nonnull(g_strstr_len(error->message, -1, "Device only supports version upgrades")); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_plugin_device_gtype(FuTest *self, GType gtype) |
| { |
| gboolean ret; |
| g_autofree gchar *str = NULL; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuProgress) progress_tmp = fu_progress_new(G_STRLOC); |
| g_autoptr(FuSecurityAttrs) attrs = fu_security_attrs_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GHashTable) metadata_post = NULL; |
| g_autoptr(GHashTable) metadata_pre = NULL; |
| const gchar *nolocker[] = { |
| "FuPciPspDevice", |
| "FuSynapticsRmiPs2Device", |
| "FuUefiSbatDevice", |
| NULL, |
| }; |
| |
| g_debug("loading %s", g_type_name(gtype)); |
| device = g_object_new(gtype, "context", self->ctx, "physical-id", "/sys", NULL); |
| g_assert_nonnull(device); |
| fu_device_set_plugin(device, "test"); |
| |
| /* version convert */ |
| if (fu_device_get_version_format(device) != FWUPD_VERSION_FORMAT_UNKNOWN) |
| fu_device_set_version_raw(device, 0); |
| |
| /* progress steps */ |
| fu_device_set_progress(device, progress_tmp); |
| |
| /* report metadata */ |
| metadata_pre = fu_device_report_metadata_pre(device); |
| if (metadata_pre != NULL) |
| g_debug("got %u metadata items", g_hash_table_size(metadata_pre)); |
| metadata_post = fu_device_report_metadata_post(device); |
| if (metadata_post != NULL) |
| g_debug("got %u metadata items", g_hash_table_size(metadata_post)); |
| |
| /* security attrs */ |
| fu_device_add_security_attrs(device, attrs); |
| |
| /* quirk kvs */ |
| ret = fu_device_set_quirk_kv(device, |
| "NoGoingTo", |
| "Exist", |
| FU_CONTEXT_QUIRK_SOURCE_FALLBACK, |
| &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_false(ret); |
| |
| /* to string */ |
| str = fu_device_to_string(device); |
| g_assert_nonnull(str); |
| |
| /* ->probe() and ->setup */ |
| if (!g_strv_contains(nolocker, g_type_name(gtype))) { |
| g_autoptr(FuDeviceLocker) locker = fu_device_locker_new(device, NULL); |
| if (locker != NULL) |
| g_debug("did ->probe() and ->setup()!"); |
| } |
| } |
| |
| static void |
| fu_engine_plugin_firmware_gtype(FuTest *self, GType gtype) |
| { |
| gboolean ret; |
| g_autoptr(FuFirmware) firmware = NULL; |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GBytes) fw = g_bytes_new_static((const guint8 *)"x", 1); |
| g_autoptr(GError) error = NULL; |
| const gchar *noxml[] = { |
| "FuArchiveFirmware", |
| "FuGenesysUsbhubFirmware", |
| "FuIntelThunderboltFirmware", |
| "FuIntelThunderboltNvm", |
| "FuUefiUpdateInfo", |
| NULL, |
| }; |
| |
| g_debug("loading %s", g_type_name(gtype)); |
| firmware = g_object_new(gtype, NULL); |
| g_assert_nonnull(firmware); |
| |
| /* ensure we have data set even if parsing fails */ |
| fu_firmware_set_bytes(firmware, fw); |
| |
| /* version convert */ |
| if (fu_firmware_get_version_format(firmware) != FWUPD_VERSION_FORMAT_UNKNOWN) |
| fu_firmware_set_version_raw(firmware, 0); |
| |
| /* parse nonsense */ |
| if (gtype != FU_TYPE_FIRMWARE && |
| !fu_firmware_has_flag(FU_FIRMWARE(firmware), FU_FIRMWARE_FLAG_NO_AUTO_DETECTION)) { |
| ret = fu_firmware_parse_bytes(firmware, |
| fw, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_NO_SEARCH | |
| FU_FIRMWARE_PARSE_FLAG_CACHE_STREAM, |
| NULL); |
| g_assert_false(ret); |
| } |
| |
| /* write */ |
| blob = fu_firmware_write(firmware, NULL); |
| if (blob != NULL && g_bytes_get_size(blob) > 0) |
| g_debug("saved 0x%x bytes", (guint)g_bytes_get_size(blob)); |
| |
| /* export -> build */ |
| if (!g_strv_contains(noxml, g_type_name(gtype))) { |
| g_autofree gchar *xml = fu_firmware_export_to_xml( |
| firmware, |
| FU_FIRMWARE_EXPORT_FLAG_INCLUDE_DEBUG | FU_FIRMWARE_EXPORT_FLAG_ASCII_DATA, |
| NULL); |
| if (xml != NULL) { |
| ret = fu_firmware_build_from_xml(firmware, xml, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| } |
| } |
| |
| static void |
| fu_engine_test_plugin_mutable_enumeration(gconstpointer user_data) |
| { |
| const gchar *fake_localconf_fn = "/tmp/fwupd-self-test/var/etc/fwupd/fwupd.conf"; |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuEngine) engine = NULL; |
| g_autoptr(FuPlugin) plugin = fu_plugin_new(NULL); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| gboolean ret; |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| g_unsetenv("CONFIGURATION_DIRECTORY"); |
| (void)g_setenv("FWUPD_SYSCONFDIR", "/tmp/fwupd-self-test", TRUE); |
| |
| ret = fu_path_mkdir_parent(fake_localconf_fn, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = g_file_set_contents(fake_localconf_fn, |
| "# use `man 5 fwupd.conf` for documentation\n" |
| "[fwupd]\n" |
| "RequireImmutableEnumeration=true\n", |
| -1, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| engine = fu_engine_new(self->ctx); |
| g_assert_nonnull(engine); |
| |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* engine requires, plugin doesn't have */ |
| ret = fu_engine_plugin_allows_enumeration(engine, plugin); |
| g_assert_true(ret); |
| |
| /* engine requires, plugin does have */ |
| fu_plugin_add_flag(plugin, FWUPD_PLUGIN_FLAG_MUTABLE_ENUMERATION); |
| ret = fu_engine_plugin_allows_enumeration(engine, plugin); |
| g_assert_false(ret); |
| |
| /* clear config and reload engine */ |
| fu_self_test_mkroot(); |
| g_clear_object(&engine); |
| |
| engine = fu_engine_new(self->ctx); |
| g_assert_nonnull(engine); |
| |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* engine requires, plugin does have */ |
| ret = fu_engine_plugin_allows_enumeration(engine, plugin); |
| g_assert_true(ret); |
| |
| /* drop flag, engine shouldn't care */ |
| fu_plugin_remove_flag(plugin, FWUPD_PLUGIN_FLAG_MUTABLE_ENUMERATION); |
| ret = fu_engine_plugin_allows_enumeration(engine, plugin); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_plugin_gtypes_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| GPtrArray *plugins; |
| gboolean ret; |
| g_autoptr(FuDrmDevice) drm_device = g_object_new(FU_TYPE_DRM_DEVICE, NULL); |
| g_autoptr(FuEdid) edid = fu_edid_new(); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(FuSecurityAttrs) attrs = fu_security_attrs_new(); |
| g_autoptr(GArray) firmware_gtypes = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| const gchar *plugin_names_safe[] = { |
| "linux_lockdown", |
| "linux_sleep", |
| "linux_swap", |
| "linux_tainted", |
| }; |
| const gchar *external_plugins[] = { |
| "flashrom", |
| "modem-manager", |
| }; |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* load these from the build directory, not the install directory */ |
| if (g_getenv("G_TEST_BUILDDIR") != NULL) { |
| g_autoptr(GPtrArray) external_plugin_dirs = g_ptr_array_new_with_free_func(g_free); |
| g_autofree gchar *external_plugindir = NULL; |
| for (guint i = 0; i < G_N_ELEMENTS(external_plugins); i++) { |
| g_ptr_array_add(external_plugin_dirs, |
| g_test_build_filename(G_TEST_BUILT, |
| "..", |
| "plugins", |
| external_plugins[i], |
| NULL)); |
| } |
| external_plugindir = fu_strjoin(",", external_plugin_dirs); |
| (void)g_setenv("FWUPD_LIBDIR_PKG", external_plugindir, TRUE); |
| } |
| |
| /* load all plugins */ |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_BUILTIN_PLUGINS | |
| FU_ENGINE_LOAD_FLAG_EXTERNAL_PLUGINS, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| plugins = fu_engine_get_plugins(engine); |
| g_assert_nonnull(plugins); |
| g_assert_cmpint(plugins->len, >, 5); |
| |
| /* start up some "safe" plugins */ |
| for (guint i = 0; i < G_N_ELEMENTS(plugin_names_safe); i++) { |
| FuPlugin *plugin = fu_engine_get_plugin_by_name(engine, plugin_names_safe[i], NULL); |
| if (plugin != NULL) { |
| ret = fu_plugin_runner_startup(plugin, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| } |
| |
| /* add security attrs where possible */ |
| for (guint i = 0; i < plugins->len; i++) { |
| FuPlugin *plugin = g_ptr_array_index(plugins, i); |
| fu_plugin_runner_add_security_attrs(plugin, attrs); |
| } |
| |
| /* run the post-reboot action */ |
| for (guint i = 0; i < plugins->len; i++) { |
| FuPlugin *plugin = g_ptr_array_index(plugins, i); |
| g_autoptr(FuDevice) device_nop = fu_device_new(self->ctx); |
| g_autoptr(GError) error_local = NULL; |
| if (!fu_plugin_runner_reboot_cleanup(plugin, device_nop, &error_local)) |
| g_debug("ignoring: %s", error_local->message); |
| } |
| |
| /* run the device unlock action */ |
| for (guint i = 0; i < plugins->len; i++) { |
| FuPlugin *plugin = g_ptr_array_index(plugins, i); |
| g_autoptr(FuDevice) device_nop = fu_device_new(self->ctx); |
| g_autoptr(GError) error_local = NULL; |
| if (!fu_plugin_runner_unlock(plugin, device_nop, &error_local)) |
| g_debug("ignoring: %s", error_local->message); |
| } |
| |
| /* run the backend-device-added action */ |
| for (guint i = 0; i < plugins->len; i++) { |
| FuPlugin *plugin = g_ptr_array_index(plugins, i); |
| GArray *device_gtypes = fu_plugin_get_device_gtypes(plugin); |
| if (device_gtypes == NULL || device_gtypes->len == 0) { |
| g_autoptr(FuDevice) device_nop = fu_device_new(self->ctx); |
| g_autoptr(GError) error_local = NULL; |
| if (!fu_plugin_runner_backend_device_added(plugin, |
| device_nop, |
| progress, |
| &error_local)) |
| g_debug("ignoring: %s", error_local->message); |
| } |
| } |
| |
| /* register a DRM device as some plugins use these for quirks */ |
| fu_edid_set_pnp_id(edid, "PNP"); |
| fu_edid_set_eisa_id(edid, "IBM"); |
| fu_edid_set_product_name(edid, "Display"); |
| fu_edid_set_serial_number(edid, "123456"); |
| fu_edid_set_product_code(edid, 0x1234); |
| fu_drm_device_set_edid(drm_device, edid); |
| for (guint i = 0; i < plugins->len; i++) { |
| FuPlugin *plugin = g_ptr_array_index(plugins, i); |
| fu_plugin_runner_device_register(plugin, FU_DEVICE(drm_device)); |
| } |
| |
| /* create each custom device with a context only */ |
| for (guint i = 0; i < plugins->len; i++) { |
| FuPlugin *plugin = g_ptr_array_index(plugins, i); |
| GArray *device_gtypes = fu_plugin_get_device_gtypes(plugin); |
| for (guint j = 0; device_gtypes != NULL && j < device_gtypes->len; j++) { |
| GType gtype = g_array_index(device_gtypes, GType, j); |
| fu_engine_plugin_device_gtype(self, gtype); |
| } |
| } |
| |
| /* create each firmware */ |
| firmware_gtypes = fu_context_get_firmware_gtypes(self->ctx); |
| for (guint j = 0; j < firmware_gtypes->len; j++) { |
| GType gtype = g_array_index(firmware_gtypes, GType, j); |
| fu_engine_plugin_firmware_gtype(self, gtype); |
| } |
| } |
| |
| static void |
| fu_engine_requirements_sibling_device_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) unrelated_device3 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) parent = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release1 = fu_release_new(); |
| g_autoptr(FuRelease) release2 = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| const gchar *xml = |
| "<component>" |
| " <requires>" |
| " <firmware depth=\"0\">1ff60ab2-3905-06a1-b476-0371f00c9e9b</firmware>" |
| " <id compare=\"ge\" version=\"1.6.1\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.4\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up a dummy device */ |
| fu_device_set_id(device1, "id1"); |
| fu_device_set_version_format(device1, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device1, "1.2.3"); |
| fu_device_build_vendor_id_u16(device1, "USB", 0xFFFF); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device1, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_protocol(device1, "com.acme"); |
| fu_engine_add_device(engine, device1); |
| |
| /* setup the parent */ |
| fu_device_set_id(parent, "parent"); |
| fu_device_set_version_format(parent, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(parent, "1.0.0"); |
| fu_device_add_flag(parent, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(parent, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(parent, "42f3d696-0b6f-4d69-908f-357f98ef115e"); |
| fu_device_add_protocol(parent, "com.acme"); |
| fu_device_add_child(parent, device1); |
| fu_engine_add_device(engine, parent); |
| |
| /* set up a different device */ |
| fu_device_set_id(unrelated_device3, "id3"); |
| fu_device_build_vendor_id(unrelated_device3, "USB", "FFFF"); |
| fu_device_add_protocol(unrelated_device3, "com.acme"); |
| fu_device_set_name(unrelated_device3, "Foo bar device"); |
| fu_device_set_version_format(unrelated_device3, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(unrelated_device3, "1.5.3"); |
| fu_device_add_flag(unrelated_device3, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(unrelated_device3, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(unrelated_device3, "3e455c08-352e-4a16-84d3-f04287289fa2"); |
| fu_engine_add_device(engine, unrelated_device3); |
| |
| /* import firmware metainfo */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_device(release1, device1); |
| fu_release_set_request(release1, request); |
| ret = fu_release_load(release1, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release1, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_false(ret); |
| g_clear_error(&error); |
| |
| /* set up a sibling device */ |
| fu_device_set_id(device2, "id2"); |
| fu_device_build_vendor_id_u16(device2, "USB", 0xFFFF); |
| fu_device_add_protocol(device2, "com.acme"); |
| fu_device_set_name(device2, "Secondary firmware"); |
| fu_device_set_version_format(device2, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device2, "4.5.6"); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device2, "1ff60ab2-3905-06a1-b476-0371f00c9e9b"); |
| fu_device_add_child(parent, device2); |
| fu_engine_add_device(engine, device2); |
| |
| /* check this passes */ |
| fu_release_set_device(release2, device1); |
| fu_release_set_request(release2, request); |
| ret = fu_release_load(release2, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release2, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check this still works, as a child requirement is specified */ |
| fu_device_add_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_ENFORCE_REQUIRES); |
| ret = fu_engine_requirements_check(engine, release2, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_other_device_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| const gchar *xml = |
| "<component>" |
| " <requires>" |
| " <firmware compare=\"gt\" " |
| "version=\"4.0.0\">1ff60ab2-3905-06a1-b476-0371f00c9e9b</firmware>" |
| " <id compare=\"ge\" version=\"1.2.11\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.4\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up a dummy device */ |
| fu_device_set_version_format(device1, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device1, "1.2.3"); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device1, "12345678-1234-1234-1234-123456789012"); |
| |
| /* set up a different device */ |
| fu_device_set_id(device2, "id2"); |
| fu_device_build_vendor_id_u16(device2, "USB", 0xFFFF); |
| fu_device_add_protocol(device2, "com.acme"); |
| fu_device_set_name(device2, "Secondary firmware"); |
| fu_device_set_version_format(device2, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device2, "4.5.6"); |
| fu_device_add_instance_id(device2, "1ff60ab2-3905-06a1-b476-0371f00c9e9b"); |
| fu_engine_add_device(engine, device2); |
| |
| /* import firmware metainfo */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_device(release, device1); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_protocol_check_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(FuRelease) release1 = fu_release_new(); |
| g_autoptr(FuRelease) release2 = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| gboolean ret; |
| |
| const gchar *xml = |
| "<component>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"4.5.7\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " </release>" |
| " </releases>" |
| " <custom>" |
| " <value key=\"LVFS::UpdateProtocol\">org.bar</value>" |
| " </custom>" |
| |
| "</component>"; |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| fu_device_set_id(device1, "NVME"); |
| fu_device_add_protocol(device1, "com.acme"); |
| fu_device_set_name(device1, "NVME device"); |
| fu_device_build_vendor_id(device1, "DMI", "ACME"); |
| fu_device_set_version_format(device1, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device1, "1.2.3"); |
| fu_device_add_instance_id(device1, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device1); |
| |
| fu_device_set_id(device2, "UEFI"); |
| fu_device_add_protocol(device2, "org.bar"); |
| fu_device_set_name(device2, "UEFI device"); |
| fu_device_build_vendor_id(device2, "DMI", "ACME"); |
| fu_device_set_version_format(device2, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device2, "1.2.3"); |
| fu_device_add_instance_id(device2, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device2); |
| |
| /* make sure both devices added */ |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 2); |
| |
| /* import firmware metainfo */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this fails */ |
| fu_release_set_device(release1, device1); |
| fu_release_set_request(release1, request); |
| ret = fu_release_load(release1, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_false(ret); |
| g_clear_error(&error); |
| |
| /* check this passes */ |
| fu_release_set_device(release2, device2); |
| fu_release_set_request(release2, request); |
| ret = fu_release_load(release2, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_parent_device_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| const gchar *xml = |
| "<component>" |
| " <requires>" |
| " <firmware depth=\"1\" compare=\"eq\" version=\"1.2.3\"/>" |
| " <firmware depth=\"1\">12345678-1234-1234-1234-123456789012</firmware>" |
| " <id compare=\"ge\" version=\"1.3.4\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <provides>" |
| " <firmware type=\"flashed\">1ff60ab2-3905-06a1-b476-0371f00c9e9b</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"4.5.7\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up child device */ |
| fu_device_set_id(device2, "child"); |
| fu_device_set_name(device2, "child"); |
| fu_device_set_version_format(device2, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device2, "4.5.6"); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device2, "1ff60ab2-3905-06a1-b476-0371f00c9e9b"); |
| |
| /* set up a parent device */ |
| fu_device_set_id(device1, "parent"); |
| fu_device_build_vendor_id_u16(device1, "USB", 0xFFFF); |
| fu_device_add_protocol(device1, "com.acme"); |
| fu_device_set_name(device1, "parent"); |
| fu_device_set_version_format(device1, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device1, "1.2.3"); |
| fu_device_add_instance_id(device1, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_child(device1, device2); |
| fu_engine_add_device(engine, device1); |
| |
| /* import firmware metainfo */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_device(release, device2); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_requirements_child_device_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| const gchar *xml = |
| "<component>" |
| " <requires>" |
| " <firmware depth=\"-1\">1ff60ab2-3905-06a1-b476-0371f00c9e9b</firmware>" |
| " <id compare=\"ge\" version=\"1.9.7\">org.freedesktop.fwupd</id>\n" |
| " </requires>" |
| " <provides>" |
| " <firmware type=\"flashed\">12345678-1234-1234-1234-123456789012</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"4.5.7\">" |
| " <checksum type=\"sha1\" filename=\"bios.bin\" target=\"content\"/>" |
| " </release>" |
| " </releases>" |
| "</component>"; |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up a parent device */ |
| fu_device_set_id(device1, "parent"); |
| fu_device_build_vendor_id_u16(device1, "USB", 0xFFFF); |
| fu_device_add_protocol(device1, "com.acme"); |
| fu_device_set_name(device1, "parent"); |
| fu_device_set_version_format(device1, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device1, "1.2.3"); |
| fu_device_add_instance_id(device1, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| |
| /* set up child device */ |
| fu_device_set_id(device2, "child"); |
| fu_device_set_name(device2, "child"); |
| fu_device_set_version_format(device2, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device2, "4.5.6"); |
| fu_device_add_instance_id(device2, "1ff60ab2-3905-06a1-b476-0371f00c9e9b"); |
| fu_device_add_child(device1, device2); |
| fu_engine_add_device(engine, device1); |
| |
| /* import firmware metainfo */ |
| silo = xb_silo_new_from_xml(xml, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| component = xb_silo_query_first(silo, "component", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check this passes */ |
| fu_release_set_device(release, device1); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_device_parent_guid_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device3 = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* add child */ |
| fu_device_set_id(device1, "child"); |
| fu_device_build_vendor_id_u16(device2, "USB", 0xFFFF); |
| fu_device_add_protocol(device2, "com.acme"); |
| fu_device_add_instance_id(device1, "child-GUID-1"); |
| fu_device_add_parent_guid(device1, "parent-GUID"); |
| fu_engine_add_device(engine, device1); |
| |
| /* parent */ |
| fu_device_set_id(device2, "parent"); |
| fu_device_build_vendor_id_u16(device2, "USB", 0xFFFF); |
| fu_device_add_protocol(device2, "com.acme"); |
| fu_device_add_instance_id(device2, "parent-GUID"); |
| fu_device_set_vendor(device2, "oem"); |
| |
| /* add another child */ |
| fu_device_set_id(device3, "child2"); |
| fu_device_add_instance_id(device3, "child-GUID-2"); |
| fu_device_add_parent_guid(device3, "parent-GUID"); |
| fu_device_add_child(device2, device3); |
| |
| /* add two together */ |
| fu_engine_add_device(engine, device2); |
| |
| /* this is normally done by fu_plugin_device_add() */ |
| fu_engine_add_device(engine, device3); |
| |
| /* verify both children were adopted */ |
| g_assert_true(fu_device_get_parent(device3) == device2); |
| g_assert_true(fu_device_get_parent(device1) == device2); |
| g_assert_cmpstr(fu_device_get_vendor(device3), ==, "oem"); |
| |
| /* verify order */ |
| g_assert_cmpint(fu_device_get_order(device1), ==, -1); |
| g_assert_cmpint(fu_device_get_order(device2), ==, 0); |
| g_assert_cmpint(fu_device_get_order(device3), ==, -1); |
| } |
| |
| static void |
| fu_engine_device_parent_id_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device3 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device4 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device5 = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* add child */ |
| fu_device_set_id(device1, "child1"); |
| fu_device_set_name(device1, "Child1"); |
| fu_device_set_physical_id(device1, "child-ID1"); |
| fu_device_build_vendor_id_u16(device1, "USB", 0xFFFF); |
| fu_device_add_protocol(device1, "com.acme"); |
| fu_device_add_instance_id(device1, "child-GUID-1"); |
| fu_device_add_parent_physical_id(device1, "parent-ID-notfound"); |
| fu_device_add_parent_physical_id(device1, "parent-ID"); |
| fu_engine_add_device(engine, device1); |
| |
| /* parent */ |
| fu_device_set_id(device2, "parent"); |
| fu_device_set_name(device2, "Parent"); |
| fu_device_set_backend_id(device2, "/sys/devices/foo/bar/baz"); |
| fu_device_set_physical_id(device2, "parent-ID"); |
| fu_device_build_vendor_id_u16(device2, "USB", 0xFFFF); |
| fu_device_add_protocol(device2, "com.acme"); |
| fu_device_add_instance_id(device2, "parent-GUID"); |
| fu_device_set_vendor(device2, "oem"); |
| fu_device_add_private_flag(device2, FU_DEVICE_PRIVATE_FLAG_AUTO_PARENT_CHILDREN); |
| |
| /* add another child */ |
| fu_device_set_id(device3, "child2"); |
| fu_device_set_name(device3, "Child2"); |
| fu_device_set_physical_id(device3, "child-ID2"); |
| fu_device_add_instance_id(device3, "child-GUID-2"); |
| fu_device_add_parent_physical_id(device3, "parent-ID"); |
| fu_device_add_child(device2, device3); |
| |
| /* add two together */ |
| fu_engine_add_device(engine, device2); |
| |
| /* add non-child */ |
| fu_device_set_id(device4, "child4"); |
| fu_device_set_name(device4, "Child4"); |
| fu_device_set_physical_id(device4, "child-ID4"); |
| fu_device_build_vendor_id(device4, "USB", "FFFF"); |
| fu_device_add_protocol(device4, "com.acme"); |
| fu_device_add_instance_id(device4, "child-GUID-4"); |
| fu_device_add_parent_physical_id(device4, "parent-ID"); |
| fu_engine_add_device(engine, device4); |
| |
| /* this is normally done by fu_plugin_device_add() */ |
| fu_engine_add_device(engine, device4); |
| |
| /* add child with the parent backend ID */ |
| fu_device_set_id(device5, "child5"); |
| fu_device_set_name(device5, "Child5"); |
| fu_device_set_physical_id(device5, "child-ID5"); |
| fu_device_build_vendor_id(device5, "USB", "FFFF"); |
| fu_device_add_protocol(device5, "com.acme"); |
| fu_device_add_instance_id(device5, "child-GUID-5"); |
| fu_device_add_parent_backend_id(device5, "/sys/devices/foo/bar/baz"); |
| fu_engine_add_device(engine, device5); |
| |
| /* this is normally done by fu_plugin_device_add() */ |
| fu_engine_add_device(engine, device5); |
| |
| /* verify both children were adopted */ |
| g_assert_true(fu_device_get_parent(device3) == device2); |
| g_assert_true(fu_device_get_parent(device4) == device2); |
| g_assert_true(fu_device_get_parent(device5) == device2); |
| g_assert_true(fu_device_get_parent(device1) == device2); |
| g_assert_cmpstr(fu_device_get_vendor(device3), ==, "oem"); |
| } |
| |
| static void |
| fu_engine_partial_hash_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new(NULL); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GError) error_none = NULL; |
| g_autoptr(GError) error_both = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up dummy plugin */ |
| fu_plugin_set_name(plugin, "test"); |
| fu_engine_add_plugin(engine, plugin); |
| |
| /* add two dummy devices */ |
| fu_device_set_id(device1, "device1"); |
| fu_device_build_vendor_id_u16(device1, "USB", 0xFFFF); |
| fu_device_add_protocol(device1, "com.acme"); |
| fu_device_set_plugin(device1, "test"); |
| fu_device_add_instance_id(device1, "12345678-1234-1234-1234-123456789012"); |
| fu_device_set_id(device1, "99249eb1bd9ef0b6e192b271a8cb6a3090cfec7a"); |
| fu_engine_add_device(engine, device1); |
| fu_device_set_id(device2, "device21"); |
| fu_device_build_vendor_id_u16(device2, "USB", 0xFFFF); |
| fu_device_add_protocol(device2, "com.acme"); |
| fu_device_set_plugin(device2, "test"); |
| fu_device_set_equivalent_id(device2, "b92f5b7560b84ca005a79f5a15de3c003ce494cf"); |
| fu_device_add_instance_id(device2, "87654321-1234-1234-1234-123456789012"); |
| fu_device_set_id(device2, "99244162a6daa0b033d649c8d464529cec41d3de"); |
| fu_engine_add_device(engine, device2); |
| |
| /* match nothing */ |
| ret = fu_engine_unlock(engine, "deadbeef", &error_none); |
| g_assert_error(error_none, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_false(ret); |
| |
| /* match both */ |
| ret = fu_engine_unlock(engine, "9924", &error_both); |
| g_assert_error(error_both, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_false(ret); |
| |
| /* match one exactly */ |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_LOCKED); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_LOCKED); |
| ret = fu_engine_unlock(engine, "99244162a6daa0b033d649c8d464529cec41d3de", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* match one partially */ |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_LOCKED); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_LOCKED); |
| ret = fu_engine_unlock(engine, "99249", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* match equivalent ID */ |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_LOCKED); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_LOCKED); |
| ret = fu_engine_unlock(engine, "b92f", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_device_unlock_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FwupdRelease *rel; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GFile) file = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add the hardcoded 'fwupd' metadata */ |
| filename = g_test_build_filename(G_TEST_DIST, "tests", "metadata.xml", NULL); |
| file = g_file_new_for_path(filename); |
| ret = xb_builder_source_load_file(source, file, XB_BUILDER_SOURCE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "UEFI-dummy-dev0"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_LOCKED); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_PLAIN); |
| fu_engine_add_device(engine, device); |
| |
| /* ensure the metainfo was matched */ |
| rel = fwupd_device_get_release_default(FWUPD_DEVICE(device)); |
| g_assert_nonnull(rel); |
| g_assert_false(fwupd_release_has_flag(rel, FWUPD_RELEASE_FLAG_TRUSTED_REPORT)); |
| } |
| |
| static void |
| fu_engine_device_equivalent_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device_best = NULL; |
| g_autoptr(FuDevice) device_worst = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a wireless (worse) device */ |
| fu_device_set_id(device1, "99249eb1bd9ef0b6e192b271a8cb6a3090cfec7a"); |
| fu_device_set_name(device1, "device1"); |
| fu_device_build_vendor_id_u16(device1, "USB", 0xFFFF); |
| fu_device_add_protocol(device1, "com.acme"); |
| fu_device_add_instance_id(device1, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device1); |
| |
| /* add a wired (better) device */ |
| fu_device_set_id(device2, "1a8d0d9a96ad3e67ba76cf3033623625dc6d6882"); |
| fu_device_set_name(device2, "device2"); |
| fu_device_set_equivalent_id(device2, "99249eb1bd9ef0b6e192b271a8cb6a3090cfec7a"); |
| fu_device_set_priority(device2, 999); |
| fu_device_build_vendor_id_u16(device2, "USB", 0xFFFF); |
| fu_device_add_protocol(device2, "com.acme"); |
| fu_device_add_instance_id(device2, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device2); |
| |
| /* make sure the daemon chooses the best device */ |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 2); |
| device_best = fu_engine_get_device(engine, "9924", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device_best); |
| g_assert_cmpstr(fu_device_get_id(device_best), |
| ==, |
| "1a8d0d9a96ad3e67ba76cf3033623625dc6d6882"); |
| g_assert_true(fu_device_has_flag(device_best, FWUPD_DEVICE_FLAG_UPDATABLE)); |
| g_assert_false(fu_device_has_problem(device_best, FWUPD_DEVICE_PROBLEM_LOWER_PRIORITY)); |
| |
| /* get the worst device and make sure it's not updatable */ |
| for (guint i = 0; i < devices->len; i++) { |
| FuDevice *device_tmp = g_ptr_array_index(devices, i); |
| if (device_tmp != device_best) { |
| device_worst = g_object_ref(device_tmp); |
| break; |
| } |
| } |
| g_assert_nonnull(device_worst); |
| g_assert_false(fu_device_has_flag(device_worst, FWUPD_DEVICE_FLAG_UPDATABLE)); |
| g_assert_true(fu_device_has_problem(device_worst, FWUPD_DEVICE_PROBLEM_LOWER_PRIORITY)); |
| } |
| |
| static void |
| fu_engine_device_md_set_flags_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n" |
| "<components version=\"0.9\">\n" |
| " <component type=\"firmware\">\n" |
| " <id>org.fwupd.8330a096d9f1af8567c7374cb8403e1ce9cf3163.device</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">2d47f29b-83a2-4f31-a2e8-63474f4d4c2e</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1\" />\n" |
| " </releases>\n" |
| " <custom>\n" |
| " <value key=\"LVFS::DeviceFlags\">save-into-backup-remote</value>\n" |
| " </custom>\n" |
| " </component>\n" |
| "</components>\n"; |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add the XML metadata */ |
| ret = xb_builder_source_load_xml(source, xml, XB_BUILDER_SOURCE_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "UEFI-dummy-dev0"); |
| fu_device_set_version(device, "0"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_FLAGS); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_PLAIN); |
| fu_engine_add_device(engine, device); |
| |
| /* check the flag got set */ |
| g_assert_true( |
| fu_device_has_private_flag(device, FU_DEVICE_PRIVATE_FLAG_SAVE_INTO_BACKUP_REMOTE)); |
| } |
| |
| static void |
| fu_engine_device_md_checksum_set_version_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n" |
| "<components version=\"0.9\">\n" |
| " <component type=\"firmware\">\n" |
| " <id>org.fwupd.8330a096d9f1af8567c7374cb8403e1ce9cf3163.device</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">2d47f29b-83a2-4f31-a2e8-63474f4d4c2e</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"124\">\n" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum type=\"sha256\" " |
| "target=\"device\">cdb7c90d3ab8833d5324f5d8516d41fa990b9ca721fe643fffaef9057d9f9e48</" |
| "checksum>\n" |
| " </release>\n" |
| " </releases>\n" |
| " <custom>\n" |
| " <value key=\"LVFS::UpdateProtocol\">com.acme</value>\n" |
| " <value key=\"LVFS::VersionFormat\">plain</value>" |
| " </custom>\n" |
| " </component>\n" |
| "</components>\n"; |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add the XML metadata */ |
| ret = xb_builder_source_load_xml(source, xml, XB_BUILDER_SOURCE_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "UEFI-dummy-dev0"); |
| fu_device_set_version(device, "123"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_ONLY_CHECKSUM); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_VERSION); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_VERFMT); |
| fu_device_add_checksum(device, |
| "cdb7c90d3ab8833d5324f5d8516d41fa990b9ca721fe643fffaef9057d9f9e48"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_NUMBER); |
| fu_engine_add_device(engine, device); |
| |
| /* check the version got set */ |
| g_assert_cmpstr(fu_device_get_version(device), ==, "124"); |
| g_assert_cmpint(fu_device_get_version_format(device), ==, FWUPD_VERSION_FORMAT_PLAIN); |
| } |
| |
| static void |
| fu_engine_device_md_checksum_set_version_wrong_proto_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| const gchar *xml = |
| "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n" |
| "<components version=\"0.9\">\n" |
| " <component type=\"firmware\">\n" |
| " <id>org.fwupd.8330a096d9f1af8567c7374cb8403e1ce9cf3163.device</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">2d47f29b-83a2-4f31-a2e8-63474f4d4c2e</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"124\">\n" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum type=\"sha256\" " |
| "target=\"device\">cdb7c90d3ab8833d5324f5d8516d41fa990b9ca721fe643fffaef9057d9f9e48</" |
| "checksum>\n" |
| " </release>\n" |
| " </releases>\n" |
| " <custom>\n" |
| " <value key=\"LVFS::UpdateProtocol\">com.acme</value>\n" |
| " <value key=\"LVFS::VersionFormat\">plain</value>" |
| " </custom>\n" |
| " </component>\n" |
| "</components>\n"; |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add the XML metadata */ |
| ret = xb_builder_source_load_xml(source, xml, XB_BUILDER_SOURCE_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "UEFI-dummy-dev0"); |
| fu_device_set_version(device, "123"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "SOMETHING_ELSE_ENTIRELY"); |
| fu_device_add_instance_id(device, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_ONLY_CHECKSUM); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_VERSION); |
| fu_device_add_checksum(device, |
| "cdb7c90d3ab8833d5324f5d8516d41fa990b9ca721fe643fffaef9057d9f9e48"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_NUMBER); |
| fu_engine_add_device(engine, device); |
| |
| /* check the version did not get set, because the protocol was different */ |
| g_assert_cmpstr(fu_device_get_version(device), ==, "123"); |
| g_assert_cmpint(fu_device_get_version_format(device), ==, FWUPD_VERSION_FORMAT_NUMBER); |
| } |
| |
| static void |
| fu_engine_require_hwid_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuCabinet) cabinet = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* get generated file as a blob */ |
| filename = |
| g_test_build_filename(G_TEST_BUILT, "tests", "missing-hwid", "hwid-1.2.3.cab", NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| cabinet = fu_engine_build_cabinet_from_stream(engine, stream, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(cabinet); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device); |
| |
| /* get component */ |
| component = fu_cabinet_get_component(cabinet, "com.hughski.test.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* check requirements */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| ret = fu_release_load(release, NULL, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_requirements_check(engine, release, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_INVALID_FILE); |
| g_assert_nonnull(error); |
| g_assert_cmpstr(error->message, |
| ==, |
| "no HWIDs matched 9342d47a-1bab-5709-9869-c840b2eac501"); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_get_details_added_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FuDevice *device_tmp; |
| FwupdRelease *release; |
| gboolean ret; |
| g_autofree gchar *checksum_sha256 = NULL; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "test_device"); |
| fu_device_set_name(device, "test device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device); |
| |
| /* get details */ |
| filename = |
| g_test_build_filename(G_TEST_BUILT, "tests", "missing-hwid", "hwid-1.2.3.cab", NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| checksum_sha256 = fu_input_stream_compute_checksum(stream, G_CHECKSUM_SHA256, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(checksum_sha256); |
| devices = fu_engine_get_details(engine, request, stream, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| device_tmp = g_ptr_array_index(devices, 0); |
| g_assert_cmpstr(fu_device_get_name(device_tmp), ==, "test device"); |
| release = fu_device_get_release_default(device_tmp); |
| g_assert_nonnull(release); |
| g_assert_cmpstr(fwupd_release_get_version(release), ==, "1.2.3"); |
| g_assert_true(fwupd_release_has_checksum(release, checksum_sha256)); |
| } |
| |
| static void |
| fu_engine_get_details_missing_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FuDevice *device_tmp; |
| FwupdRelease *release; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GInputStream) stream = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* get details */ |
| filename = |
| g_test_build_filename(G_TEST_BUILT, "tests", "missing-hwid", "hwid-1.2.3.cab", NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| devices = fu_engine_get_details(engine, request, stream, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| device_tmp = g_ptr_array_index(devices, 0); |
| g_assert_cmpstr(fu_device_get_name(device_tmp), ==, NULL); |
| release = fu_device_get_release_default(device_tmp); |
| g_assert_nonnull(release); |
| g_assert_cmpstr(fwupd_release_get_version(release), ==, "1.2.3"); |
| } |
| |
| static void |
| fu_engine_downgrade_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FwupdRelease *rel; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(GPtrArray) devices_pre = NULL; |
| g_autoptr(GPtrArray) releases_dg = NULL; |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(GPtrArray) releases_up = NULL; |
| g_autoptr(GPtrArray) releases_up2 = NULL; |
| g_autoptr(GPtrArray) remotes = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* write a broken file */ |
| ret = g_file_set_contents("/tmp/fwupd-self-test/broken.xml.gz", |
| "this is not a valid", |
| -1, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* write the main file */ |
| ret = g_file_set_contents( |
| "/tmp/fwupd-self-test/stable.xml", |
| "<components>" |
| " <component type=\"firmware\">" |
| " <id>test</id>" |
| " <name>Test Device</name>" |
| " <provides>" |
| " <firmware type=\"flashed\">aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.3\" date=\"2017-09-15\">" |
| " <size type=\"installed\">123</size>" |
| " <size type=\"download\">456</size>" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum filename=\"foo.cab\" target=\"container\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdead1111</checksum>" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " </release>" |
| " <release version=\"1.2.2\" date=\"2017-09-01\">" |
| " <size type=\"installed\">123</size>" |
| " <size type=\"download\">456</size>" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum filename=\"foo.cab\" target=\"container\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdead2222</checksum>" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " </release>" |
| " </releases>" |
| " </component>" |
| "</components>", |
| -1, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* write the extra file */ |
| ret = g_file_set_contents( |
| "/tmp/fwupd-self-test/testing.xml", |
| "<components>" |
| " <component type=\"firmware\">" |
| " <id>test</id>" |
| " <name>Test Device</name>" |
| " <provides>" |
| " <firmware type=\"flashed\">aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.5\" date=\"2017-09-16\">" |
| " <size type=\"installed\">123</size>" |
| " <size type=\"download\">456</size>" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum filename=\"foo.cab\" target=\"container\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdead3333</checksum>" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " </release>" |
| " <release version=\"1.2.4\" date=\"2017-09-15\">" |
| " <size type=\"installed\">123</size>" |
| " <size type=\"download\">456</size>" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum filename=\"foo.cab\" target=\"container\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdead4444</checksum>" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " </release>" |
| " </releases>" |
| " </component>" |
| "</components>", |
| -1, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_REMOTES | FU_ENGINE_LOAD_FLAG_NO_CACHE, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_test_assert_expected_messages(); |
| |
| /* return all the remotes, even the broken one */ |
| remotes = fu_engine_get_remotes(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(remotes); |
| g_assert_cmpint(remotes->len, ==, 7); |
| |
| /* ensure there are no devices already */ |
| devices_pre = fu_engine_get_devices(engine, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOTHING_TO_DO); |
| g_assert_null(devices_pre); |
| g_clear_error(&error); |
| |
| /* add a device so we can get upgrades and downgrades */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_name(device, "Test Device"); |
| fu_device_add_instance_id(device, "aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device); |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| g_assert_true(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_SUPPORTED)); |
| g_assert_true(fu_device_has_private_flag(device, FU_DEVICE_PRIVATE_FLAG_REGISTERED)); |
| |
| /* get the releases for one device */ |
| releases = fu_engine_get_releases(engine, request, fu_device_get_id(device), &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases); |
| g_assert_cmpint(releases->len, ==, 4); |
| |
| /* no upgrades, as no firmware is approved */ |
| releases_up = fu_engine_get_upgrades(engine, request, fu_device_get_id(device), &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOTHING_TO_DO); |
| g_assert_null(releases_up); |
| g_clear_error(&error); |
| |
| /* retry with approved firmware set */ |
| fu_engine_add_approved_firmware(engine, "deadbeefdeadbeefdeadbeefdead1111"); |
| fu_engine_add_approved_firmware(engine, "deadbeefdeadbeefdeadbeefdead2222"); |
| fu_engine_add_approved_firmware(engine, "deadbeefdeadbeefdeadbeefdead3333"); |
| fu_engine_add_approved_firmware(engine, "deadbeefdeadbeefdeadbeefdead4444"); |
| fu_engine_add_approved_firmware(engine, "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"); |
| |
| /* upgrades */ |
| releases_up = fu_engine_get_upgrades(engine, request, fu_device_get_id(device), &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases_up); |
| g_assert_cmpint(releases_up->len, ==, 2); |
| |
| /* ensure the list is sorted */ |
| rel = FWUPD_RELEASE(g_ptr_array_index(releases_up, 0)); |
| g_assert_cmpstr(fwupd_release_get_version(rel), ==, "1.2.5"); |
| rel = FWUPD_RELEASE(g_ptr_array_index(releases_up, 1)); |
| g_assert_cmpstr(fwupd_release_get_version(rel), ==, "1.2.4"); |
| |
| /* downgrades */ |
| releases_dg = fu_engine_get_downgrades(engine, request, fu_device_get_id(device), &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases_dg); |
| g_assert_cmpint(releases_dg->len, ==, 1); |
| rel = FWUPD_RELEASE(g_ptr_array_index(releases_dg, 0)); |
| g_assert_cmpstr(fwupd_release_get_version(rel), ==, "1.2.2"); |
| |
| /* enforce that updates have to be explicit */ |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_ONLY_EXPLICIT_UPDATES); |
| releases_up2 = fu_engine_get_upgrades(engine, request, fu_device_get_id(device), &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOTHING_TO_DO); |
| g_assert_null(releases_up2); |
| } |
| |
| static void |
| fu_engine_md_verfmt_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FwupdRemote *remote; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* write the main file */ |
| ret = g_file_set_contents( |
| "/tmp/fwupd-self-test/stable.xml", |
| "<components>" |
| " <component type=\"firmware\">" |
| " <id>test</id>" |
| " <name>Test Device</name>" |
| " <icon>computer</icon>" |
| " <developer_name>ACME</developer_name>" |
| " <provides>" |
| " <firmware type=\"flashed\">aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee</firmware>" |
| " </provides>" |
| " <categories>" |
| " <category>X-GraphicsTablet</category>" |
| " </categories>" |
| " <releases>" |
| " <release version=\"1.2.3\" date=\"2017-09-15\">" |
| " <size type=\"installed\">123</size>" |
| " <size type=\"download\">456</size>" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum filename=\"foo.cab\" target=\"container\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " <artifacts>" |
| " <artifact type=\"binary\">" |
| " <size type=\"installed\">1024</size>" |
| " <size type=\"download\">2048</size>" |
| " </artifact>" |
| " </artifacts>" |
| " </release>" |
| " </releases>" |
| " <custom>" |
| " <value key=\"LVFS::VersionFormat\">triplet</value>" |
| " <value key=\"LVFS::DeviceIntegrity\">signed</value>" |
| " <value key=\"LVFS::DeviceFlags\">host-cpu,needs-shutdown</value>" |
| " </custom>" |
| " </component>" |
| "</components>", |
| -1, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_REMOTES | FU_ENGINE_LOAD_FLAG_NO_CACHE, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_test_assert_expected_messages(); |
| |
| /* pretend this has a signature */ |
| remote = fu_engine_get_remote_by_id(engine, "stable", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(remote); |
| |
| /* add a device with no defined version format */ |
| fu_device_set_version(device, "16908291"); |
| fu_device_set_version_raw(device, 0x01020003); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_NAME_CATEGORY); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_ICON); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_VENDOR); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_SIGNED); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_VERFMT); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_FLAGS); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_REQUIRED_FREE); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_engine_add_device(engine, device); |
| |
| /* ensure the version format was set from the metadata */ |
| g_assert_cmpint(fu_device_get_version_format(device), ==, FWUPD_VERSION_FORMAT_TRIPLET); |
| g_assert_cmpstr(fu_device_get_version(device), ==, "1.2.3"); |
| g_assert_cmpstr(fu_device_get_name(device), ==, "Graphics Tablet"); |
| g_assert_cmpstr(fu_device_get_vendor(device), ==, "ACME"); |
| g_assert_true(fu_device_has_icon(device, "computer")); |
| g_assert_true(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_SIGNED_PAYLOAD)); |
| g_assert_true(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_NEEDS_SHUTDOWN)); |
| g_assert_true(fu_device_has_private_flag(device, FU_DEVICE_PRIVATE_FLAG_HOST_CPU)); |
| g_assert_cmpint(fu_device_get_required_free(device), ==, 1024); |
| |
| /* ensure the device was added */ |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| g_assert_true(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_SUPPORTED)); |
| g_assert_true(fu_device_has_private_flag(device, FU_DEVICE_PRIVATE_FLAG_REGISTERED)); |
| |
| /* ensure the releases are set */ |
| releases = fu_engine_get_releases(engine, request, fu_device_get_id(device), &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases); |
| g_assert_cmpint(releases->len, ==, 1); |
| } |
| |
| static void |
| fu_engine_install_duration_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FwupdRelease *rel; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* write the main file */ |
| ret = g_file_set_contents( |
| "/tmp/fwupd-self-test/stable.xml", |
| "<components>" |
| " <component type=\"firmware\">" |
| " <id>test</id>" |
| " <provides>" |
| " <firmware type=\"flashed\">aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.3\" date=\"2017-09-15\" install_duration=\"120\">" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum filename=\"foo.cab\" target=\"container\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " </release>" |
| " </releases>" |
| " </component>" |
| "</components>", |
| -1, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_REMOTES | FU_ENGINE_LOAD_FLAG_NO_CACHE, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a device so we can get the install duration */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee"); |
| fu_device_set_install_duration(device, 999); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device); |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| g_assert_true(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_SUPPORTED)); |
| |
| /* check the release install duration */ |
| releases = fu_engine_get_releases(engine, request, fu_device_get_id(device), &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases); |
| g_assert_cmpint(releases->len, ==, 1); |
| rel = FWUPD_RELEASE(g_ptr_array_index(releases, 0)); |
| g_assert_cmpint(fwupd_release_get_install_duration(rel), ==, 120); |
| } |
| |
| static void |
| fu_engine_release_dedupe_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* write the main file */ |
| ret = g_file_set_contents( |
| "/tmp/fwupd-self-test/stable.xml", |
| "<components>" |
| " <component type=\"firmware\">" |
| " <id>test</id>" |
| " <provides>" |
| " <firmware type=\"flashed\">aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee</firmware>" |
| " </provides>" |
| " <releases>" |
| " <release version=\"1.2.3\" date=\"2017-09-15\" install_duration=\"120\">" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum filename=\"foo.cab\" target=\"container\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " </release>" |
| " <release version=\"1.2.3\" date=\"2017-09-15\" install_duration=\"120\">" |
| " <location>https://test.org/foo.cab</location>" |
| " <checksum filename=\"foo.cab\" target=\"container\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"md5\">deadbeefdeadbeefdeadbeefdeadbeef</checksum>" |
| " </release>" |
| " </releases>" |
| " </component>" |
| "</components>", |
| -1, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_REMOTES | FU_ENGINE_LOAD_FLAG_NO_CACHE, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a device so we can get the install duration */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee"); |
| fu_device_set_install_duration(device, 999); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device); |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| g_assert_true(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_SUPPORTED)); |
| |
| /* check the release install duration */ |
| releases = fu_engine_get_releases(engine, request, fu_device_get_id(device), &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases); |
| g_assert_cmpint(releases->len, ==, 1); |
| } |
| |
| static void |
| fu_engine_history_modify_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuHistory) history = fu_history_new(self->ctx); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| |
| /* add a new entry */ |
| fu_device_set_id(device, "foobarbaz"); |
| fu_history_remove_device(history, device, NULL); |
| ret = fu_history_add_device(history, device, release, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* try to modify something that does exist */ |
| ret = fu_history_modify_device(history, device, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* does not exist */ |
| fu_device_set_id(device, "DOES-NOT-EXIST"); |
| ret = fu_history_modify_device(history, device, &error); |
| g_assert_false(ret); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| } |
| |
| static void |
| fu_engine_history_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autofree gchar *checksum = NULL; |
| g_autofree gchar *device_str_expected = NULL; |
| g_autofree gchar *device_str = NULL; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuCabinet) cabinet = NULL; |
| g_autoptr(FuDevice) device2 = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuHistory) history = NULL; |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(FwupdDevice) device3 = NULL; |
| g_autoptr(FwupdDevice) device4 = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up dummy plugin */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_plugin_set_config_value(plugin, "AnotherWriteRequired", "true", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_engine_add_plugin(engine, plugin); |
| |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a device so we can get upgrade it */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_name(device, "Test Device"); |
| fu_device_set_plugin(device, "test"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_checksum(device, "0123456789abcdef0123456789abcdef01234567"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_set_created_usec(device, 1515338000ull * G_USEC_PER_SEC); |
| fu_engine_add_device(engine, device); |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| g_assert_true(fu_device_has_private_flag(device, FU_DEVICE_PRIVATE_FLAG_REGISTERED)); |
| |
| filename = |
| g_test_build_filename(G_TEST_BUILT, "tests", "missing-hwid", "noreqs-1.2.3.cab", NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| cabinet = fu_engine_build_cabinet_from_stream(engine, stream, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(cabinet); |
| |
| /* get component */ |
| component = fu_cabinet_get_component(cabinet, "com.hughski.test.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* set the counter */ |
| fu_device_set_metadata_integer(device, "nr-update", 0); |
| |
| /* install it */ |
| fu_release_set_device(release, device); |
| ret = fu_release_load(release, cabinet, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_install_release(engine, |
| release, |
| progress, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check the write was done more than once */ |
| g_assert_cmpint(fu_device_get_metadata_integer(device, "nr-update"), ==, 2); |
| |
| /* check the history database */ |
| history = fu_history_new(self->ctx); |
| device2 = fu_history_get_device_by_id(history, fu_device_get_id(device), &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device2); |
| g_assert_cmpint(fu_device_get_update_state(device2), ==, FWUPD_UPDATE_STATE_SUCCESS); |
| g_assert_cmpstr(fu_device_get_update_error(device2), ==, NULL); |
| fu_device_set_modified_usec(device2, 1514338000ull * G_USEC_PER_SEC); |
| g_hash_table_remove_all(fwupd_release_get_metadata(fu_device_get_release_default(device2))); |
| device_str = fu_device_to_string(device2); |
| checksum = fu_input_stream_compute_checksum(stream, G_CHECKSUM_SHA1, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(checksum); |
| device_str_expected = |
| g_strdup_printf("FuDevice:\n" |
| " DeviceId: 894e8c17a29428b09d10cd90d1db74ea76fbcfe8\n" |
| " Name: Test Device\n" |
| " Guid: 12345678-1234-1234-1234-123456789012\n" |
| " Plugin: test\n" |
| " Flags: updatable|historical|unsigned-payload\n" |
| " Version: 1.2.2\n" |
| " VersionFormat: triplet\n" |
| " Created: 2018-01-07 15:13:20\n" |
| " Modified: 2017-12-27 01:26:40\n" |
| " UpdateState: success\n" |
| " FuRelease:\n" |
| " AppstreamId: com.hughski.test.firmware\n" |
| " Version: 1.2.3\n" |
| " Checksum: SHA1(%s)\n" |
| " Flags: trusted-payload|trusted-metadata\n" |
| " InstanceId[vi]: 12345678-1234-1234-1234-123456789012\n" |
| " AcquiesceDelay: 50\n", |
| checksum); |
| g_debug("%s", device_str); |
| ret = g_strcmp0(device_str, device_str_expected) == 0; |
| g_assert_true(ret); |
| |
| /* GetResults() */ |
| device3 = fu_engine_get_results(engine, FWUPD_DEVICE_ID_ANY, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device3); |
| g_assert_cmpstr(fu_device_get_id(device3), ==, "894e8c17a29428b09d10cd90d1db74ea76fbcfe8"); |
| g_assert_cmpint(fu_device_get_update_state(device3), ==, FWUPD_UPDATE_STATE_SUCCESS); |
| g_assert_cmpstr(fu_device_get_update_error(device3), ==, NULL); |
| |
| /* ClearResults() */ |
| ret = fu_engine_clear_results(engine, FWUPD_DEVICE_ID_ANY, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* GetResults() */ |
| device4 = fu_engine_get_results(engine, FWUPD_DEVICE_ID_ANY, &error); |
| g_assert_null(device4); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOTHING_TO_DO); |
| } |
| |
| static void |
| fu_engine_history_verfmt_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = g_object_new(FU_TYPE_DPAUX_DEVICE, "context", self->ctx, NULL); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up dummy plugin */ |
| fu_engine_add_plugin(engine, plugin); |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* absorb version format from the database */ |
| fu_device_set_version_raw(device, 65563); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_NUMBER); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_plugin(device, "test"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_checksum(device, "0123456789abcdef0123456789abcdef01234567"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_MD_SET_VERFMT); |
| fu_device_set_created_usec(device, 1515338000ull * G_USEC_PER_SEC); |
| fu_engine_add_device(engine, device); |
| g_assert_cmpint(fu_device_get_version_format(device), ==, FWUPD_VERSION_FORMAT_TRIPLET); |
| g_assert_cmpstr(fu_device_get_version(device), ==, "0.1.27"); |
| } |
| |
| static void |
| fu_engine_install_loop_restart_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream_fw = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up dummy plugin */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_plugin_set_config_value(plugin, "InstallLoopRestart", "true", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_engine_add_plugin(engine, plugin); |
| |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a device so we can install it */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_plugin(device, "test"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_engine_add_device(engine, device); |
| |
| /* set up counters */ |
| fu_device_set_metadata_integer(device, "nr-update", 0); |
| fu_device_set_metadata_integer(device, "nr-attach", 0); |
| |
| stream_fw = g_memory_input_stream_new_from_data((const guint8 *)"1.2.3", 5, NULL); |
| fu_release_set_stream(release, stream_fw); |
| ret = fu_engine_install_blob(engine, |
| device, |
| release, |
| progress, |
| FWUPD_INSTALL_FLAG_NO_HISTORY | |
| FU_FIRMWARE_PARSE_FLAG_CACHE_STREAM, |
| FWUPD_FEATURE_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check we did two write loops */ |
| g_assert_cmpint(fu_device_get_metadata_integer(device, "nr-update"), ==, 2); |
| |
| /* check we only attached once */ |
| g_assert_cmpint(fu_device_get_metadata_integer(device, "nr-attach"), ==, 1); |
| } |
| |
| static void |
| fu_engine_multiple_rels_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuCabinet) cabinet = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(GPtrArray) rels = NULL; |
| g_autoptr(XbQuery) query = NULL; |
| |
| #ifndef HAVE_LIBARCHIVE |
| g_test_skip("no libarchive support"); |
| return; |
| #endif |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up dummy plugin */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_engine_add_plugin(engine, plugin); |
| |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a device so we can get upgrade it */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_name(device, "Test Device"); |
| fu_device_set_plugin(device, "test"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_checksum(device, "0123456789abcdef0123456789abcdef01234567"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_INSTALL_ALL_RELEASES); |
| fu_device_set_created_usec(device, 1515338000ull * G_USEC_PER_SEC); |
| fu_engine_add_device(engine, device); |
| |
| filename = g_test_build_filename(G_TEST_BUILT, |
| "tests", |
| "multiple-rels", |
| "multiple-rels-1.2.4.cab", |
| NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| cabinet = fu_engine_build_cabinet_from_stream(engine, stream, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(cabinet); |
| |
| /* get component */ |
| component = fu_cabinet_get_component(cabinet, "com.hughski.test.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* set up counters */ |
| fu_device_set_metadata_integer(device, "nr-update", 0); |
| fu_device_set_metadata_integer(device, "nr-attach", 0); |
| |
| /* get all */ |
| query = xb_query_new_full(xb_node_get_silo(component), |
| "releases/release", |
| XB_QUERY_FLAG_FORCE_NODE_CACHE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(query); |
| rels = xb_node_query_full(component, query, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(rels); |
| |
| releases = g_ptr_array_new_with_free_func((GDestroyNotify)g_object_unref); |
| for (guint i = 0; i < rels->len; i++) { |
| XbNode *rel = g_ptr_array_index(rels, i); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| fu_release_set_device(release, device); |
| ret = fu_release_load(release, |
| cabinet, |
| component, |
| rel, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_ptr_array_add(releases, g_object_ref(release)); |
| } |
| |
| /* install them */ |
| fu_progress_reset(progress); |
| ret = fu_engine_install_releases(engine, |
| request, |
| releases, |
| cabinet, |
| progress, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check we did 1.2.2 -> 1.2.3 -> 1.2.4 */ |
| g_assert_cmpint(fu_device_get_metadata_integer(device, "nr-update"), ==, 2); |
| g_assert_cmpint(fu_device_get_metadata_integer(device, "nr-attach"), ==, 2); |
| g_assert_cmpstr(fu_device_get_version(device), ==, "1.2.4"); |
| |
| /* reset the config back to defaults */ |
| ret = fu_engine_reset_config(engine, "fwupd", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_engine_history_inherit(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autofree gchar *localstatedir = NULL; |
| g_autofree gchar *history_db = NULL; |
| g_autoptr(FuCabinet) cabinet = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* delete history */ |
| localstatedir = fu_path_from_kind(FU_PATH_KIND_LOCALSTATEDIR_PKG); |
| history_db = g_build_filename(localstatedir, "pending.db", NULL); |
| (void)g_unlink(history_db); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up dummy plugin */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_plugin_set_config_value(plugin, "NeedsActivation", "true", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_engine_add_plugin(engine, plugin); |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a device so we can get upgrade it */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_name(device, "Test Device"); |
| fu_device_set_plugin(device, "test"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_set_created_usec(device, 1515338000ull * G_USEC_PER_SEC); |
| fu_engine_add_device(engine, device); |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| g_assert_true(fu_device_has_private_flag(device, FU_DEVICE_PRIVATE_FLAG_REGISTERED)); |
| |
| filename = |
| g_test_build_filename(G_TEST_BUILT, "tests", "missing-hwid", "noreqs-1.2.3.cab", NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| cabinet = fu_engine_build_cabinet_from_stream(engine, stream, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(cabinet); |
| |
| /* get component */ |
| component = fu_cabinet_get_component(cabinet, "com.hughski.test.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* install it */ |
| fu_release_set_device(release, device); |
| ret = fu_release_load(release, cabinet, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_install_release(engine, |
| release, |
| progress, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check the device requires an activation */ |
| g_assert_true(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_NEEDS_ACTIVATION)); |
| g_assert_cmpstr(fu_device_get_version(device), ==, "1.2.2"); |
| |
| /* activate the device */ |
| fu_progress_reset(progress); |
| ret = fu_engine_activate(engine, fu_device_get_id(device), progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check the device no longer requires an activation */ |
| g_assert_false(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_NEEDS_ACTIVATION)); |
| g_assert_cmpstr(fu_device_get_version(device), ==, "1.2.3"); |
| |
| /* emulate getting the flag for a fresh boot on old firmware */ |
| fu_progress_reset(progress); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| ret = fu_engine_install_release(engine, |
| release, |
| progress, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_object_unref(engine); |
| g_object_unref(device); |
| engine = fu_engine_new(self->ctx); |
| fu_engine_set_silo(engine, silo_empty); |
| fu_engine_add_plugin(engine, plugin); |
| device = fu_device_new(self->ctx); |
| fu_device_add_private_flag(device, FU_DEVICE_PRIVATE_FLAG_INHERIT_ACTIVATION); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_name(device, "Test Device"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_engine_add_device(engine, device); |
| g_assert_true(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_NEEDS_ACTIVATION)); |
| |
| /* emulate not getting the flag */ |
| g_object_unref(engine); |
| g_object_unref(device); |
| engine = fu_engine_new(self->ctx); |
| fu_engine_set_silo(engine, silo_empty); |
| fu_engine_add_plugin(engine, plugin); |
| device = fu_device_new(self->ctx); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_name(device, "Test Device"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_engine_add_device(engine, device); |
| g_assert_false(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_NEEDS_ACTIVATION)); |
| } |
| |
| static void |
| fu_engine_install_needs_reboot(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuCabinet) cabinet = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| g_autofree gchar *rundir = fu_path_from_kind(FU_PATH_KIND_RUNDIR); |
| g_autofree gchar *reboot_file = NULL; |
| |
| /* create rundir */ |
| ret = fu_path_mkdir(rundir, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up dummy plugin */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_plugin_set_config_value(plugin, "NeedsReboot", "true", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_engine_add_plugin(engine, plugin); |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a device so we can get upgrade it */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_name(device, "Test Device"); |
| fu_device_set_plugin(device, "test"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_set_created_usec(device, 1515338000ull * G_USEC_PER_SEC); |
| fu_engine_add_device(engine, device); |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| g_assert_true(fu_device_has_private_flag(device, FU_DEVICE_PRIVATE_FLAG_REGISTERED)); |
| |
| filename = |
| g_test_build_filename(G_TEST_BUILT, "tests", "missing-hwid", "noreqs-1.2.3.cab", NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| cabinet = fu_engine_build_cabinet_from_stream(engine, stream, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(cabinet); |
| |
| /* get component */ |
| component = fu_cabinet_get_component(cabinet, "com.hughski.test.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* install it */ |
| fu_release_set_device(release, device); |
| ret = fu_release_load(release, cabinet, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_install_release(engine, |
| release, |
| progress, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check the device requires reboot */ |
| g_assert_true(fu_device_has_flag(device, FWUPD_DEVICE_FLAG_NEEDS_REBOOT)); |
| g_assert_cmpstr(fu_device_get_version(device), ==, "1.2.2"); |
| |
| reboot_file = g_build_filename(rundir, "reboot-required", NULL); |
| g_assert_true(g_file_test(reboot_file, G_FILE_TEST_EXISTS)); |
| } |
| |
| typedef struct { |
| guint request_cnt; |
| FwupdStatus last_status; |
| } FuTestRequestHelper; |
| |
| static void |
| fu_test_engine_status_changed_cb(FuProgress *progress, FwupdStatus status, gpointer user_data) |
| { |
| FuTestRequestHelper *helper = (FuTestRequestHelper *)user_data; |
| g_debug("status now %s", fwupd_status_to_string(status)); |
| helper->last_status = status; |
| } |
| |
| static void |
| fu_test_engine_request_cb(FuEngine *engine, FwupdRequest *request, gpointer user_data) |
| { |
| FuTestRequestHelper *helper = (FuTestRequestHelper *)user_data; |
| g_assert_cmpint(fwupd_request_get_kind(request), ==, FWUPD_REQUEST_KIND_IMMEDIATE); |
| g_assert_cmpstr(fwupd_request_get_id(request), ==, FWUPD_REQUEST_ID_REMOVE_REPLUG); |
| g_assert_true(fwupd_request_has_flag(request, FWUPD_REQUEST_FLAG_ALLOW_GENERIC_MESSAGE)); |
| g_assert_nonnull(fwupd_request_get_message(request)); |
| g_assert_cmpint(helper->last_status, ==, FWUPD_STATUS_WAITING_FOR_USER); |
| helper->request_cnt++; |
| } |
| |
| static void |
| fu_engine_install_request(gconstpointer user_data) |
| { |
| FuTestRequestHelper helper = {.request_cnt = 0, .last_status = FWUPD_STATUS_UNKNOWN}; |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuCabinet) cabinet = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up dummy plugin */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_plugin_set_config_value(plugin, "RequestSupported", "true", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_engine_add_plugin(engine, plugin); |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a device so we can get upgrade it */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_name(device, "Test Device"); |
| fu_device_set_plugin(device, "test"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_request_flag(device, FWUPD_REQUEST_FLAG_ALLOW_GENERIC_MESSAGE); |
| fu_device_set_created_usec(device, 1515338000ull * G_USEC_PER_SEC); |
| fu_engine_add_device(engine, device); |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| g_assert_true(fu_device_has_private_flag(device, FU_DEVICE_PRIVATE_FLAG_REGISTERED)); |
| |
| filename = |
| g_test_build_filename(G_TEST_BUILT, "tests", "missing-hwid", "noreqs-1.2.3.cab", NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| cabinet = fu_engine_build_cabinet_from_stream(engine, stream, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(cabinet); |
| |
| /* get component */ |
| component = fu_cabinet_get_component(cabinet, "com.hughski.test.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| |
| /* install it */ |
| fu_release_set_device(release, device); |
| ret = fu_release_load(release, cabinet, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpstr(fu_release_get_firmware_basename(release), ==, "firmware.bin"); |
| g_assert_cmpstr(fu_release_get_version(release), ==, "1.2.3"); |
| |
| g_signal_connect(FU_ENGINE(engine), |
| "device-request", |
| G_CALLBACK(fu_test_engine_request_cb), |
| &helper); |
| g_signal_connect(FU_PROGRESS(progress), |
| "status-changed", |
| G_CALLBACK(fu_test_engine_status_changed_cb), |
| &helper); |
| |
| ret = fu_engine_install_release(engine, |
| release, |
| progress, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(helper.request_cnt, ==, 1); |
| g_assert_cmpint(helper.last_status, ==, FWUPD_STATUS_DEVICE_BUSY); |
| } |
| |
| static void |
| fu_engine_history_error_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autofree gchar *checksum = NULL; |
| g_autofree gchar *device_str_expected = NULL; |
| g_autofree gchar *device_str = NULL; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuCabinet) cabinet = NULL; |
| g_autoptr(FuDevice) device2 = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuHistory) history = NULL; |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error2 = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* set up dummy plugin */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_plugin_set_config_value(plugin, "WriteSupported", "false", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_engine_add_plugin(engine, plugin); |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a device so we can get upgrade it */ |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_set_id(device, "test_device"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_set_name(device, "Test Device"); |
| fu_device_set_plugin(device, "test"); |
| fu_device_add_instance_id(device, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_set_created_usec(device, 1515338000ull * G_USEC_PER_SEC); |
| fu_engine_add_device(engine, device); |
| devices = fu_engine_get_devices(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(devices); |
| g_assert_cmpint(devices->len, ==, 1); |
| g_assert_true(fu_device_has_private_flag(device, FU_DEVICE_PRIVATE_FLAG_REGISTERED)); |
| |
| /* install the wrong thing */ |
| filename = |
| g_test_build_filename(G_TEST_BUILT, "tests", "missing-hwid", "noreqs-1.2.3.cab", NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| cabinet = fu_engine_build_cabinet_from_stream(engine, stream, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(cabinet); |
| component = fu_cabinet_get_component(cabinet, "com.hughski.test.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| fu_release_set_device(release, device); |
| ret = fu_release_load(release, cabinet, component, NULL, FWUPD_INSTALL_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_engine_install_release(engine, |
| release, |
| progress, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_cmpstr(error->message, |
| ==, |
| "failed to write-firmware: device was not in supported mode"); |
| g_assert_false(ret); |
| |
| /* check the history database */ |
| history = fu_history_new(self->ctx); |
| device2 = fu_history_get_device_by_id(history, fu_device_get_id(device), &error2); |
| g_assert_no_error(error2); |
| g_assert_nonnull(device2); |
| g_assert_cmpint(fu_device_get_update_state(device2), ==, FWUPD_UPDATE_STATE_FAILED); |
| g_assert_cmpstr(fu_device_get_update_error(device2), ==, error->message); |
| g_clear_error(&error); |
| fu_device_set_modified_usec(device2, 1514338000ull * G_USEC_PER_SEC); |
| g_hash_table_remove_all(fwupd_release_get_metadata(fu_device_get_release_default(device2))); |
| device_str = fu_device_to_string(device2); |
| checksum = fu_input_stream_compute_checksum(stream, G_CHECKSUM_SHA1, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(checksum); |
| device_str_expected = g_strdup_printf( |
| "FuDevice:\n" |
| " DeviceId: 894e8c17a29428b09d10cd90d1db74ea76fbcfe8\n" |
| " Name: Test Device\n" |
| " Guid: 12345678-1234-1234-1234-123456789012\n" |
| " Plugin: test\n" |
| " Flags: updatable|historical|unsigned-payload\n" |
| " Version: 1.2.2\n" |
| " VersionFormat: triplet\n" |
| " Created: 2018-01-07 15:13:20\n" |
| " Modified: 2017-12-27 01:26:40\n" |
| " UpdateState: failed\n" |
| " UpdateError: failed to write-firmware: device was not in supported mode\n" |
| " FuRelease:\n" |
| " AppstreamId: com.hughski.test.firmware\n" |
| " Version: 1.2.3\n" |
| " Checksum: SHA1(%s)\n" |
| " Flags: trusted-payload|trusted-metadata\n" |
| " InstanceId[vi]: 12345678-1234-1234-1234-123456789012\n" |
| " AcquiesceDelay: 50\n", |
| checksum); |
| g_debug("%s", device_str); |
| ret = g_strcmp0(device_str, device_str_expected) == 0; |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_test_device_list_count_cb(FuDeviceList *device_list, FuDevice *device, gpointer user_data) |
| { |
| guint *cnt = (guint *)user_data; |
| (*cnt)++; |
| } |
| |
| static void |
| fu_device_list_no_auto_remove_children_func(gconstpointer user_data) |
| { |
| g_autoptr(FuDevice) child = fu_device_new(NULL); |
| g_autoptr(FuDevice) parent = fu_device_new(NULL); |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| g_autoptr(GPtrArray) active1 = NULL; |
| g_autoptr(GPtrArray) active2 = NULL; |
| g_autoptr(GPtrArray) active3 = NULL; |
| |
| /* normal behavior, remove child with parent */ |
| fu_device_set_id(parent, "parent"); |
| fu_device_set_id(child, "child"); |
| fu_device_add_child(parent, child); |
| fu_device_list_add(device_list, parent); |
| fu_device_list_add(device_list, child); |
| fu_device_list_remove(device_list, parent); |
| active1 = fu_device_list_get_active(device_list); |
| g_assert_cmpint(active1->len, ==, 0); |
| |
| /* new-style behavior, do not remove child */ |
| fu_device_add_private_flag(parent, FU_DEVICE_PRIVATE_FLAG_NO_AUTO_REMOVE_CHILDREN); |
| fu_device_list_add(device_list, parent); |
| fu_device_list_add(device_list, child); |
| fu_device_list_remove(device_list, parent); |
| active2 = fu_device_list_get_active(device_list); |
| g_assert_cmpint(active2->len, ==, 1); |
| fu_device_list_remove(device_list, child); |
| active3 = fu_device_list_get_active(device_list); |
| g_assert_cmpint(active3->len, ==, 0); |
| } |
| |
| static void |
| fu_device_list_delay_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| guint added_cnt = 0; |
| guint changed_cnt = 0; |
| guint removed_cnt = 0; |
| |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "added", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &added_cnt); |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "removed", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &removed_cnt); |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "changed", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &changed_cnt); |
| |
| /* add one device */ |
| fu_device_set_id(device1, "device1"); |
| fu_device_add_instance_id(device1, "foobar"); |
| fu_device_add_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_DELAYED_REMOVAL); |
| fu_device_set_remove_delay(device1, 100); |
| fu_device_list_add(device_list, device1); |
| g_assert_cmpint(added_cnt, ==, 1); |
| g_assert_cmpint(removed_cnt, ==, 0); |
| g_assert_cmpint(changed_cnt, ==, 0); |
| |
| /* add the same device again */ |
| fu_device_list_add(device_list, device1); |
| g_assert_cmpint(added_cnt, ==, 1); |
| g_assert_cmpint(removed_cnt, ==, 0); |
| g_assert_cmpint(changed_cnt, ==, 1); |
| |
| /* add a device with the same ID */ |
| fu_device_set_id(device2, "device1"); |
| fu_device_add_private_flag(device2, FU_DEVICE_PRIVATE_FLAG_DELAYED_REMOVAL); |
| fu_device_list_add(device_list, device2); |
| fu_device_set_remove_delay(device2, 100); |
| g_assert_cmpint(added_cnt, ==, 1); |
| g_assert_cmpint(removed_cnt, ==, 0); |
| g_assert_cmpint(changed_cnt, ==, 2); |
| |
| /* spin a bit */ |
| fu_test_loop_run_with_timeout(10); |
| fu_test_loop_quit(); |
| |
| /* verify only a changed event was generated */ |
| added_cnt = removed_cnt = changed_cnt = 0; |
| fu_device_list_remove(device_list, device1); |
| fu_device_list_add(device_list, device1); |
| g_assert_cmpint(added_cnt, ==, 0); |
| g_assert_cmpint(removed_cnt, ==, 0); |
| g_assert_cmpint(changed_cnt, ==, 1); |
| } |
| |
| typedef struct { |
| FuDevice *device_new; |
| FuDevice *device_old; |
| FuDeviceList *device_list; |
| } FuDeviceListReplugHelper; |
| |
| static gboolean |
| fu_device_list_remove_cb(gpointer user_data) |
| { |
| FuDeviceListReplugHelper *helper = (FuDeviceListReplugHelper *)user_data; |
| fu_device_list_remove(helper->device_list, helper->device_old); |
| return FALSE; |
| } |
| |
| static gboolean |
| fu_device_list_add_cb(gpointer user_data) |
| { |
| FuDeviceListReplugHelper *helper = (FuDeviceListReplugHelper *)user_data; |
| fu_device_list_add(helper->device_list, helper->device_new); |
| return FALSE; |
| } |
| |
| static void |
| fu_device_list_replug_auto_func(gconstpointer user_data) |
| { |
| gboolean ret; |
| g_autoptr(FuDevice) device1 = fu_device_new(NULL); |
| g_autoptr(FuDevice) device2 = fu_device_new(NULL); |
| g_autoptr(FuDevice) parent = fu_device_new(NULL); |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| g_autoptr(GError) error = NULL; |
| FuDeviceListReplugHelper helper; |
| |
| /* parent */ |
| fu_device_set_id(parent, "parent"); |
| |
| /* fake child devices */ |
| fu_device_set_id(device1, "device1"); |
| fu_device_add_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_REPLUG_MATCH_GUID); |
| fu_device_set_physical_id(device1, "ID"); |
| fu_device_set_plugin(device1, "self-test"); |
| fu_device_set_remove_delay(device1, FU_DEVICE_REMOVE_DELAY_RE_ENUMERATE); |
| fu_device_add_child(parent, device1); |
| fu_device_set_id(device2, "device2"); |
| fu_device_add_private_flag(device2, FU_DEVICE_PRIVATE_FLAG_REPLUG_MATCH_GUID); |
| fu_device_set_physical_id(device2, "ID"); /* matches */ |
| fu_device_set_plugin(device2, "self-test"); |
| fu_device_set_remove_delay(device2, FU_DEVICE_REMOVE_DELAY_RE_ENUMERATE); |
| |
| /* not yet added */ |
| ret = fu_device_list_wait_for_replug(device_list, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add device */ |
| fu_device_list_add(device_list, device1); |
| |
| /* not waiting */ |
| ret = fu_device_list_wait_for_replug(device_list, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* waiting */ |
| helper.device_old = device1; |
| helper.device_new = device2; |
| helper.device_list = device_list; |
| g_timeout_add(100, fu_device_list_remove_cb, &helper); |
| g_timeout_add(200, fu_device_list_add_cb, &helper); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| ret = fu_device_list_wait_for_replug(device_list, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_false(fu_device_has_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| |
| /* check device2 now has parent too */ |
| g_assert_true(fu_device_get_parent(device2) == parent); |
| |
| /* waiting, failed */ |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| ret = fu_device_list_wait_for_replug(device_list, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_false(ret); |
| g_assert_false(fu_device_has_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| } |
| |
| static void |
| fu_device_list_replug_user_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| g_autoptr(GError) error = NULL; |
| FuDeviceListReplugHelper helper; |
| |
| /* fake devices */ |
| fu_device_set_id(device1, "device1"); |
| fu_device_set_name(device1, "device1"); |
| fu_device_add_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_REPLUG_MATCH_GUID); |
| fu_device_add_instance_id(device1, "foo"); |
| fu_device_add_instance_id(device1, "bar"); |
| fu_device_set_plugin(device1, "self-test"); |
| fu_device_set_remove_delay(device1, FU_DEVICE_REMOVE_DELAY_USER_REPLUG); |
| fu_device_set_id(device2, "device2"); |
| fu_device_set_name(device2, "device2"); |
| fu_device_add_private_flag(device2, FU_DEVICE_PRIVATE_FLAG_REPLUG_MATCH_GUID); |
| fu_device_add_instance_id(device2, "baz"); |
| fu_device_add_instance_id_full(device2, |
| "bar", |
| FU_DEVICE_INSTANCE_FLAG_COUNTERPART); /* matches */ |
| fu_device_set_plugin(device2, "self-test"); |
| fu_device_set_remove_delay(device2, FU_DEVICE_REMOVE_DELAY_USER_REPLUG); |
| |
| /* not yet added */ |
| ret = fu_device_list_wait_for_replug(device_list, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add device */ |
| fu_device_list_add(device_list, device1); |
| |
| /* add duplicate */ |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| fu_device_list_add(device_list, device1); |
| g_assert_false(fu_device_has_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| |
| /* not waiting */ |
| ret = fu_device_list_wait_for_replug(device_list, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* waiting */ |
| helper.device_old = device1; |
| helper.device_new = device2; |
| helper.device_list = device_list; |
| g_timeout_add(100, fu_device_list_remove_cb, &helper); |
| g_timeout_add(200, fu_device_list_add_cb, &helper); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| ret = fu_device_list_wait_for_replug(device_list, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_false(fu_device_has_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| |
| /* should not be possible, but here we are */ |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| fu_device_list_add(device_list, device1); |
| g_assert_false(fu_device_has_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| g_assert_false(fu_device_has_flag(device2, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| |
| /* add back the old device */ |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| fu_device_list_remove(device_list, device2); |
| fu_device_list_add(device_list, device1); |
| g_assert_false(fu_device_has_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| g_assert_false(fu_device_has_flag(device2, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| } |
| |
| static void |
| fu_device_list_compatible_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device_old = NULL; |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| g_autoptr(GPtrArray) devices_all = NULL; |
| g_autoptr(GPtrArray) devices_active = NULL; |
| FuDevice *device; |
| guint added_cnt = 0; |
| guint changed_cnt = 0; |
| guint removed_cnt = 0; |
| |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "added", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &added_cnt); |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "removed", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &removed_cnt); |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "changed", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &changed_cnt); |
| |
| /* add one device in runtime mode */ |
| fu_device_set_id(device1, "device1"); |
| fu_device_set_plugin(device1, "plugin-for-runtime"); |
| fu_device_build_vendor_id(device1, "USB", "0x20A0"); |
| fu_device_set_version_format(device1, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device1, "1.2.3"); |
| fu_device_add_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_REPLUG_MATCH_GUID); |
| fu_device_add_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_DELAYED_REMOVAL); |
| fu_device_add_instance_id(device1, "foobar"); |
| fu_device_add_instance_id_full(device1, "bootloader", FU_DEVICE_INSTANCE_FLAG_COUNTERPART); |
| fu_device_set_remove_delay(device1, 100); |
| fu_device_list_add(device_list, device1); |
| g_assert_cmpint(added_cnt, ==, 1); |
| g_assert_cmpint(removed_cnt, ==, 0); |
| g_assert_cmpint(changed_cnt, ==, 0); |
| |
| /* add another device in bootloader mode */ |
| fu_device_set_id(device2, "device2"); |
| fu_device_set_plugin(device2, "plugin-for-bootloader"); |
| fu_device_add_instance_id(device2, "bootloader"); |
| fu_device_add_private_flag(device2, FU_DEVICE_PRIVATE_FLAG_REPLUG_MATCH_GUID); |
| |
| /* verify only a changed event was generated */ |
| added_cnt = removed_cnt = changed_cnt = 0; |
| fu_device_list_remove(device_list, device1); |
| fu_device_list_add(device_list, device2); |
| g_assert_cmpint(added_cnt, ==, 0); |
| g_assert_cmpint(removed_cnt, ==, 0); |
| g_assert_cmpint(changed_cnt, ==, 1); |
| |
| /* device2 should inherit the vendor ID and version from device1 */ |
| g_assert_true(fu_device_has_vendor_id(device2, "USB:0x20A0")); |
| g_assert_cmpstr(fu_device_get_version(device2), ==, "1.2.3"); |
| |
| /* one device is active */ |
| devices_active = fu_device_list_get_active(device_list); |
| g_assert_cmpint(devices_active->len, ==, 1); |
| device = g_ptr_array_index(devices_active, 0); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "1a8d0d9a96ad3e67ba76cf3033623625dc6d6882"); |
| |
| /* the list knows about both devices, list in order of active->old */ |
| devices_all = fu_device_list_get_all(device_list); |
| g_assert_cmpint(devices_all->len, ==, 2); |
| device = g_ptr_array_index(devices_all, 0); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "1a8d0d9a96ad3e67ba76cf3033623625dc6d6882"); |
| device = g_ptr_array_index(devices_all, 1); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "99249eb1bd9ef0b6e192b271a8cb6a3090cfec7a"); |
| |
| /* verify we can get the old device from the new device */ |
| device_old = fu_device_list_get_old(device_list, device2); |
| g_assert_true(device_old == device1); |
| } |
| |
| static void |
| fu_device_list_remove_chain_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| g_autoptr(FuDevice) device_child = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device_parent = fu_device_new(self->ctx); |
| |
| guint added_cnt = 0; |
| guint changed_cnt = 0; |
| guint removed_cnt = 0; |
| |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "added", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &added_cnt); |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "removed", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &removed_cnt); |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "changed", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &changed_cnt); |
| |
| /* add child */ |
| fu_device_set_id(device_child, "child"); |
| fu_device_add_instance_id(device_child, "child-GUID-1"); |
| fu_device_list_add(device_list, device_child); |
| g_assert_cmpint(added_cnt, ==, 1); |
| g_assert_cmpint(removed_cnt, ==, 0); |
| g_assert_cmpint(changed_cnt, ==, 0); |
| |
| /* add parent */ |
| fu_device_set_id(device_parent, "parent"); |
| fu_device_add_instance_id(device_parent, "parent-GUID-1"); |
| fu_device_add_child(device_parent, device_child); |
| fu_device_list_add(device_list, device_parent); |
| g_assert_cmpint(added_cnt, ==, 2); |
| g_assert_cmpint(removed_cnt, ==, 0); |
| g_assert_cmpint(changed_cnt, ==, 0); |
| |
| /* make sure that removing the parent causes both to go; but the child to go first */ |
| fu_device_list_remove(device_list, device_parent); |
| g_assert_cmpint(added_cnt, ==, 2); |
| g_assert_cmpint(removed_cnt, ==, 2); |
| g_assert_cmpint(changed_cnt, ==, 0); |
| } |
| |
| static void |
| fu_device_list_explicit_order_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device_child = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device_root = fu_device_new(self->ctx); |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| |
| /* add both */ |
| fu_device_set_id(device_root, "device"); |
| fu_device_add_instance_id(device_root, "foobar"); |
| fu_device_set_id(device_child, "device-child"); |
| fu_device_add_instance_id(device_child, "baz"); |
| fu_device_add_child(device_root, device_child); |
| fu_device_list_add(device_list, device_root); |
| |
| fu_device_add_private_flag(device_root, FU_DEVICE_PRIVATE_FLAG_EXPLICIT_ORDER); |
| fu_device_list_depsolve_order(device_list, device_root); |
| g_assert_cmpint(fu_device_get_order(device_root), ==, G_MAXINT); |
| g_assert_cmpint(fu_device_get_order(device_child), ==, G_MAXINT); |
| } |
| |
| static void |
| fu_device_list_explicit_order_post_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device_child = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device_root = fu_device_new(self->ctx); |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| |
| /* add both */ |
| fu_device_set_id(device_root, "device"); |
| fu_device_add_instance_id(device_root, "foobar"); |
| fu_device_set_id(device_child, "device-child"); |
| fu_device_add_instance_id(device_child, "baz"); |
| fu_device_add_child(device_root, device_child); |
| fu_device_list_add(device_list, device_root); |
| fu_device_list_add(device_list, device_child); |
| |
| fu_device_list_depsolve_order(device_list, device_root); |
| g_assert_cmpint(fu_device_get_order(device_root), ==, 0); |
| g_assert_cmpint(fu_device_get_order(device_child), ==, -1); |
| |
| fu_device_add_private_flag(device_root, FU_DEVICE_PRIVATE_FLAG_EXPLICIT_ORDER); |
| g_assert_cmpint(fu_device_get_order(device_root), ==, G_MAXINT); |
| g_assert_cmpint(fu_device_get_order(device_child), ==, G_MAXINT); |
| } |
| |
| static void |
| fu_device_list_install_parent_first_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device_child = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device_root = fu_device_new(self->ctx); |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| |
| /* add both */ |
| fu_device_set_id(device_root, "device"); |
| fu_device_add_instance_id(device_root, "foobar"); |
| fu_device_add_private_flag(device_root, FU_DEVICE_PRIVATE_FLAG_INSTALL_PARENT_FIRST); |
| fu_device_set_id(device_child, "device-child"); |
| fu_device_add_instance_id(device_child, "baz"); |
| fu_device_add_child(device_root, device_child); |
| fu_device_list_add(device_list, device_root); |
| fu_device_list_add(device_list, device_child); |
| |
| fu_device_list_depsolve_order(device_list, device_root); |
| g_assert_cmpint(fu_device_get_order(device_root), <, fu_device_get_order(device_child)); |
| } |
| |
| static void |
| fu_device_list_install_parent_first_child_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device_child = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device_root = fu_device_new(self->ctx); |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| |
| /* add both */ |
| fu_device_set_id(device_root, "device"); |
| fu_device_add_instance_id(device_root, "foobar"); |
| fu_device_set_id(device_child, "device-child"); |
| fu_device_add_instance_id(device_child, "baz"); |
| fu_device_add_private_flag(device_child, FU_DEVICE_PRIVATE_FLAG_INSTALL_PARENT_FIRST); |
| fu_device_add_child(device_root, device_child); |
| fu_device_list_add(device_list, device_root); |
| fu_device_list_add(device_list, device_child); |
| |
| fu_device_list_depsolve_order(device_list, device_root); |
| g_assert_cmpint(fu_device_get_order(device_root), <, fu_device_get_order(device_child)); |
| } |
| |
| static void |
| fu_device_list_better_than_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device_best = NULL; |
| g_autoptr(FuDevice) device_replug = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuPlugin) plugin1 = fu_plugin_new(self->ctx); |
| g_autoptr(FuPlugin) plugin2 = fu_plugin_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| |
| /* add a bad plugin */ |
| fu_plugin_set_name(plugin2, "plugin2"); |
| fu_engine_add_plugin(engine, plugin2); |
| |
| /* add a good plugin */ |
| fu_plugin_set_name(plugin1, "plugin1"); |
| fu_plugin_add_rule(plugin1, FU_PLUGIN_RULE_BETTER_THAN, "plugin2"); |
| fu_engine_add_plugin(engine, plugin1); |
| |
| /* load the daemon */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add a higher priority device */ |
| fu_device_set_id(device1, "87ea5dfc8b8e384d848979496e706390b497e547"); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device1, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_protocol(device1, "com.acme"); |
| fu_device_set_remove_delay(device1, FU_DEVICE_REMOVE_DELAY_RE_ENUMERATE); |
| fu_plugin_device_add(plugin1, device1); |
| |
| /* should be ignored */ |
| fu_device_set_id(device2, "87ea5dfc8b8e384d848979496e706390b497e547"); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_instance_id(device2, "12345678-1234-1234-1234-123456789012"); |
| fu_device_add_protocol(device2, "com.acme"); |
| fu_plugin_device_add(plugin2, device2); |
| |
| /* ensure we still have device1 */ |
| device_best = |
| fu_engine_get_device(engine, "87ea5dfc8b8e384d848979496e706390b497e547", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device_best); |
| g_assert_true(device_best == device1); |
| |
| /* should be replaced */ |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| fu_plugin_device_add(plugin2, device2); |
| |
| /* ensure we now have device2 */ |
| device_replug = |
| fu_engine_get_device(engine, "87ea5dfc8b8e384d848979496e706390b497e547", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device_replug); |
| g_assert_true(device_replug == device2); |
| } |
| |
| static void |
| fu_device_list_counterpart_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| |
| /* add and then remove runtime */ |
| fu_device_set_id(device1, "device-runtime"); |
| fu_device_add_instance_id(device1, "runtime"); /* 420dde7c-3102-5d8f-86bc-aaabd7920150 */ |
| fu_device_add_instance_id_full(device1, "bootloader", FU_DEVICE_INSTANCE_FLAG_COUNTERPART); |
| fu_device_set_remove_delay(device1, 100); |
| fu_device_list_add(device_list, device1); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG); |
| fu_device_list_remove(device_list, device1); |
| g_assert_true(fu_device_has_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| |
| /* add bootloader */ |
| fu_device_set_id(device2, "device-bootloader"); |
| fu_device_add_instance_id(device2, "bootloader"); /* 015370aa-26f2-5daa-9661-a75bf4c1a913 */ |
| fu_device_add_private_flag(device2, FU_DEVICE_PRIVATE_FLAG_REPLUG_MATCH_GUID); |
| fu_device_add_private_flag(device2, FU_DEVICE_PRIVATE_FLAG_ADD_COUNTERPART_GUIDS); |
| fu_device_list_add(device_list, device2); |
| |
| /* should have matched the runtime */ |
| g_assert_false(fu_device_has_flag(device1, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG)); |
| |
| /* should not have *visible* GUID of runtime */ |
| g_assert_false(fu_device_has_guid(device2, "runtime")); |
| g_assert_false( |
| fu_device_has_instance_id(device2, "runtime", FU_DEVICE_INSTANCE_FLAG_VISIBLE)); |
| } |
| |
| static void |
| fu_device_list_equivalent_id_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device3 = NULL; |
| g_autoptr(FuDevice) device4 = NULL; |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| g_autoptr(GError) error = NULL; |
| |
| fu_device_set_id(device1, "8e9cb71aeca70d2faedb5b8aaa263f6175086b2e"); |
| fu_device_list_add(device_list, device1); |
| |
| fu_device_set_id(device2, "1a8d0d9a96ad3e67ba76cf3033623625dc6d6882"); |
| fu_device_set_equivalent_id(device2, "8e9cb71aeca70d2faedb5b8aaa263f6175086b2e"); |
| fu_device_set_priority(device2, 999); |
| fu_device_list_add(device_list, device2); |
| |
| device3 = fu_device_list_get_by_id(device_list, "8e9c", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device3); |
| g_assert_cmpstr(fu_device_get_id(device3), ==, "1a8d0d9a96ad3e67ba76cf3033623625dc6d6882"); |
| |
| /* two devices with the 'same' priority */ |
| fu_device_set_priority(device2, 0); |
| device4 = fu_device_list_get_by_id(device_list, "8e9c", &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_assert_null(device4); |
| } |
| |
| static void |
| fu_device_list_unconnected_no_delay_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| |
| fu_device_set_id(device1, "device1"); |
| fu_device_add_flag(device1, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_instance_id(device1, "foobar"); |
| fu_device_list_add(device_list, device1); |
| g_assert_false(fu_device_has_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_UNCONNECTED)); |
| |
| /* remove */ |
| fu_device_list_remove(device_list, device1); |
| g_assert_true(fu_device_has_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_UNCONNECTED)); |
| |
| /* add back exact same device, then remove */ |
| fu_device_list_add(device_list, device1); |
| g_assert_false(fu_device_has_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_UNCONNECTED)); |
| fu_device_list_remove(device_list, device1); |
| g_assert_true(fu_device_has_private_flag(device1, FU_DEVICE_PRIVATE_FLAG_UNCONNECTED)); |
| |
| /* add back device with same ID, then remove */ |
| fu_device_set_id(device2, "device1"); |
| fu_device_add_flag(device2, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_instance_id(device2, "foobar"); |
| fu_device_list_add(device_list, device2); |
| g_assert_false(fu_device_has_private_flag(device2, FU_DEVICE_PRIVATE_FLAG_UNCONNECTED)); |
| fu_device_list_remove(device_list, device2); |
| g_assert_true(fu_device_has_private_flag(device2, FU_DEVICE_PRIVATE_FLAG_UNCONNECTED)); |
| } |
| |
| static void |
| fu_device_list_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuDeviceList) device_list = fu_device_list_new(); |
| g_autoptr(FuDevice) device1 = fu_device_new(self->ctx); |
| g_autoptr(FuDevice) device2 = fu_device_new(self->ctx); |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(GPtrArray) devices2 = NULL; |
| g_autoptr(GError) error = NULL; |
| FuDevice *device; |
| guint added_cnt = 0; |
| guint changed_cnt = 0; |
| guint removed_cnt = 0; |
| |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "added", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &added_cnt); |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "removed", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &removed_cnt); |
| g_signal_connect(FU_DEVICE_LIST(device_list), |
| "changed", |
| G_CALLBACK(fu_test_device_list_count_cb), |
| &changed_cnt); |
| |
| /* add both */ |
| fu_device_set_id(device1, "device1"); |
| fu_device_add_instance_id(device1, "foobar"); |
| fu_device_list_add(device_list, device1); |
| fu_device_set_id(device2, "device2"); |
| fu_device_add_instance_id(device2, "baz"); |
| fu_device_list_add(device_list, device2); |
| g_assert_cmpint(added_cnt, ==, 2); |
| g_assert_cmpint(removed_cnt, ==, 0); |
| g_assert_cmpint(changed_cnt, ==, 0); |
| |
| /* get all */ |
| devices = fu_device_list_get_all(device_list); |
| g_assert_cmpint(devices->len, ==, 2); |
| device = g_ptr_array_index(devices, 0); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "99249eb1bd9ef0b6e192b271a8cb6a3090cfec7a"); |
| |
| /* find by ID */ |
| device = fu_device_list_get_by_id(device_list, |
| "99249eb1bd9ef0b6e192b271a8cb6a3090cfec7a", |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "99249eb1bd9ef0b6e192b271a8cb6a3090cfec7a"); |
| g_clear_object(&device); |
| |
| /* find by GUID */ |
| device = |
| fu_device_list_get_by_guid(device_list, "579a3b1c-d1db-5bdc-b6b9-e2c1b28d5b8a", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "1a8d0d9a96ad3e67ba76cf3033623625dc6d6882"); |
| g_clear_object(&device); |
| |
| /* find by missing GUID */ |
| device = fu_device_list_get_by_guid(device_list, "notfound", &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_null(device); |
| |
| /* remove device */ |
| added_cnt = removed_cnt = changed_cnt = 0; |
| fu_device_list_remove(device_list, device1); |
| g_assert_cmpint(added_cnt, ==, 0); |
| g_assert_cmpint(removed_cnt, ==, 1); |
| g_assert_cmpint(changed_cnt, ==, 0); |
| devices2 = fu_device_list_get_all(device_list); |
| g_assert_cmpint(devices2->len, ==, 1); |
| device = g_ptr_array_index(devices2, 0); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "1a8d0d9a96ad3e67ba76cf3033623625dc6d6882"); |
| } |
| |
| static void |
| fu_plugin_list_func(gconstpointer user_data) |
| { |
| GPtrArray *plugins; |
| FuPlugin *plugin; |
| g_autoptr(FuPluginList) plugin_list = fu_plugin_list_new(); |
| g_autoptr(FuPlugin) plugin1 = fu_plugin_new(NULL); |
| g_autoptr(FuPlugin) plugin2 = fu_plugin_new(NULL); |
| g_autoptr(GError) error = NULL; |
| |
| fu_plugin_set_name(plugin1, "plugin1"); |
| fu_plugin_set_name(plugin2, "plugin2"); |
| |
| /* get all the plugins */ |
| fu_plugin_list_add(plugin_list, plugin1); |
| fu_plugin_list_add(plugin_list, plugin2); |
| plugins = fu_plugin_list_get_all(plugin_list); |
| g_assert_cmpint(plugins->len, ==, 2); |
| |
| /* get a single plugin */ |
| plugin = fu_plugin_list_find_by_name(plugin_list, "plugin1", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(plugin); |
| g_assert_cmpstr(fu_plugin_get_name(plugin), ==, "plugin1"); |
| |
| /* does not exist */ |
| plugin = fu_plugin_list_find_by_name(plugin_list, "nope", &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_null(plugin); |
| } |
| |
| static void |
| fu_plugin_list_depsolve_func(gconstpointer user_data) |
| { |
| GPtrArray *plugins; |
| FuPlugin *plugin; |
| gboolean ret; |
| g_autoptr(FuPluginList) plugin_list = fu_plugin_list_new(); |
| g_autoptr(FuPlugin) plugin1 = fu_plugin_new(NULL); |
| g_autoptr(FuPlugin) plugin2 = fu_plugin_new(NULL); |
| g_autoptr(GError) error = NULL; |
| |
| fu_plugin_set_name(plugin1, "plugin1"); |
| fu_plugin_set_name(plugin2, "plugin2"); |
| |
| /* add rule then depsolve */ |
| fu_plugin_list_add(plugin_list, plugin1); |
| fu_plugin_list_add(plugin_list, plugin2); |
| fu_plugin_add_rule(plugin1, FU_PLUGIN_RULE_RUN_AFTER, "plugin2"); |
| ret = fu_plugin_list_depsolve(plugin_list, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| plugins = fu_plugin_list_get_all(plugin_list); |
| g_assert_cmpint(plugins->len, ==, 2); |
| plugin = g_ptr_array_index(plugins, 0); |
| g_assert_cmpstr(fu_plugin_get_name(plugin), ==, "plugin2"); |
| g_assert_cmpint(fu_plugin_get_order(plugin), ==, 0); |
| g_assert_false(fu_plugin_has_flag(plugin, FWUPD_PLUGIN_FLAG_DISABLED)); |
| |
| /* add another rule, then re-depsolve */ |
| fu_plugin_add_rule(plugin1, FU_PLUGIN_RULE_CONFLICTS, "plugin2"); |
| ret = fu_plugin_list_depsolve(plugin_list, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| plugin = fu_plugin_list_find_by_name(plugin_list, "plugin1", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(plugin); |
| g_assert_false(fu_plugin_has_flag(plugin, FWUPD_PLUGIN_FLAG_DISABLED)); |
| plugin = fu_plugin_list_find_by_name(plugin_list, "plugin2", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(plugin); |
| g_assert_true(fu_plugin_has_flag(plugin, FWUPD_PLUGIN_FLAG_DISABLED)); |
| } |
| |
| static void |
| fu_history_migrate_v1_func(gconstpointer user_data) |
| { |
| gboolean ret; |
| g_autoptr(FuContext) ctx = fu_context_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GFile) file_dst = NULL; |
| g_autoptr(GFile) file_src = NULL; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuHistory) history = NULL; |
| g_autofree gchar *filename = NULL; |
| |
| /* load old version */ |
| filename = g_test_build_filename(G_TEST_DIST, "tests", "history_v1.db", NULL); |
| file_src = g_file_new_for_path(filename); |
| file_dst = g_file_new_for_path("/tmp/fwupd-self-test/var/lib/fwupd/pending.db"); |
| ret = g_file_copy(file_src, file_dst, G_FILE_COPY_OVERWRITE, NULL, NULL, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* create, migrating as required */ |
| history = fu_history_new(ctx); |
| g_assert_nonnull(history); |
| |
| /* get device */ |
| device = fu_history_get_device_by_id(history, |
| "2ba16d10df45823dd4494ff10a0bfccfef512c9d", |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "2ba16d10df45823dd4494ff10a0bfccfef512c9d"); |
| } |
| |
| static void |
| fu_history_migrate_v2_func(gconstpointer user_data) |
| { |
| gboolean ret; |
| g_autoptr(FuContext) ctx = fu_context_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GFile) file_dst = NULL; |
| g_autoptr(GFile) file_src = NULL; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuHistory) history = NULL; |
| g_autofree gchar *filename = NULL; |
| |
| /* load old version */ |
| filename = g_test_build_filename(G_TEST_DIST, "tests", "history_v2.db", NULL); |
| file_src = g_file_new_for_path(filename); |
| file_dst = g_file_new_for_path("/tmp/fwupd-self-test/var/lib/fwupd/pending.db"); |
| ret = g_file_copy(file_src, file_dst, G_FILE_COPY_OVERWRITE, NULL, NULL, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* create, migrating as required */ |
| history = fu_history_new(ctx); |
| g_assert_nonnull(history); |
| |
| /* get device */ |
| device = fu_history_get_device_by_id(history, |
| "2ba16d10df45823dd4494ff10a0bfccfef512c9d", |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "2ba16d10df45823dd4494ff10a0bfccfef512c9d"); |
| } |
| |
| static void |
| fu_test_plugin_device_added_cb(FuPlugin *plugin, FuDevice *device, gpointer user_data) |
| { |
| FuDevice **dev = (FuDevice **)user_data; |
| *dev = g_object_ref(device); |
| fu_test_loop_quit(); |
| } |
| |
| static void |
| fu_test_plugin_device_register_cb(FuPlugin *plugin, FuDevice *device, gpointer user_data) |
| { |
| /* fake being a daemon */ |
| fu_plugin_runner_device_register(plugin, device); |
| } |
| |
| static void |
| fu_backend_usb_hotplug_cb(FuBackend *backend, FuDevice *device, gpointer user_data) |
| { |
| guint *cnt = (guint *)user_data; |
| (*cnt)++; |
| } |
| |
| static void |
| fu_backend_usb_load_file(FuBackend *backend, const gchar *fn) |
| { |
| gboolean ret; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(JsonParser) parser = json_parser_new(); |
| |
| ret = json_parser_load_from_file(parser, fn, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fwupd_codec_from_json(FWUPD_CODEC(backend), json_parser_get_root(parser), &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| /* |
| * To generate the fwupd DS20 descriptor in the usb-devices.json file save fw-ds20.builder.xml: |
| * |
| * <firmware gtype="FuUsbDeviceFwDs20"> |
| * <idx>42</idx> <!-- bVendorCode --> |
| * <size>32</size> <!-- wLength --> |
| * </firmware> |
| * |
| * Then run: |
| * |
| * fwupdtool firmware-build fw-ds20.builder.xml fw-ds20.bin |
| * base64 fw-ds20.bin |
| * |
| * To generate the fake control transfer response, save fw-ds20.quirk: |
| * |
| * [USB\VID_273F&PID_1004] |
| * Plugin = dfu |
| * Icon = computer |
| * |
| * Then run: |
| * |
| * contrib/generate-ds20.py fw-ds20.quirk --bufsz 32 |
| */ |
| static void |
| fu_backend_usb_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| guint cnt_added = 0; |
| guint cnt_removed = 0; |
| FuDevice *device_tmp; |
| g_autofree gchar *usb_emulate_fn = NULL; |
| g_autofree gchar *usb_emulate_fn2 = NULL; |
| g_autofree gchar *usb_emulate_fn3 = NULL; |
| g_autofree gchar *devicestr = NULL; |
| g_autoptr(FuBackend) backend = fu_usb_backend_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(GPtrArray) possible_plugins = NULL; |
| |
| /* check there were events */ |
| g_object_set(backend, "device-gtype", FU_TYPE_USB_DEVICE, NULL); |
| g_signal_connect(backend, |
| "device-added", |
| G_CALLBACK(fu_backend_usb_hotplug_cb), |
| &cnt_added); |
| g_signal_connect(backend, |
| "device-removed", |
| G_CALLBACK(fu_backend_usb_hotplug_cb), |
| &cnt_removed); |
| |
| /* load the JSON into the backend */ |
| g_assert_cmpstr(fu_backend_get_name(backend), ==, "usb"); |
| g_assert_true(fu_backend_get_enabled(backend)); |
| ret = fu_backend_setup(backend, FU_BACKEND_SETUP_FLAG_NONE, progress, &error); |
| g_assert_cmpint(cnt_added, ==, 0); |
| g_assert_cmpint(cnt_removed, ==, 0); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(cnt_added, ==, 0); |
| g_assert_cmpint(cnt_removed, ==, 0); |
| ret = fu_backend_coldplug(backend, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(cnt_added, ==, 0); |
| g_assert_cmpint(cnt_removed, ==, 0); |
| usb_emulate_fn = g_test_build_filename(G_TEST_DIST, "tests", "usb-devices.json", NULL); |
| g_assert_nonnull(usb_emulate_fn); |
| fu_backend_usb_load_file(backend, usb_emulate_fn); |
| g_assert_cmpint(cnt_added, ==, 1); |
| g_assert_cmpint(cnt_removed, ==, 0); |
| devices = fu_backend_get_devices(backend); |
| g_assert_cmpint(devices->len, ==, 1); |
| device_tmp = g_ptr_array_index(devices, 0); |
| fu_device_set_context(device_tmp, self->ctx); |
| ret = fu_device_probe(device_tmp, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_true(fu_device_has_flag(device_tmp, FWUPD_DEVICE_FLAG_EMULATED)); |
| |
| /* for debugging */ |
| devicestr = fu_device_to_string(device_tmp); |
| g_debug("%s", devicestr); |
| |
| /* check the fwupd DS20 descriptor was parsed */ |
| g_assert_true(fu_device_has_icon(device_tmp, "computer")); |
| possible_plugins = fu_device_get_possible_plugins(device_tmp); |
| g_assert_cmpint(possible_plugins->len, ==, 1); |
| g_assert_cmpstr(g_ptr_array_index(possible_plugins, 0), ==, "dfu"); |
| |
| /* load another device with the same VID:PID, and check that we did not get a replug */ |
| usb_emulate_fn2 = |
| g_test_build_filename(G_TEST_DIST, "tests", "usb-devices-replace.json", NULL); |
| g_assert_nonnull(usb_emulate_fn2); |
| fu_backend_usb_load_file(backend, usb_emulate_fn2); |
| g_assert_cmpint(cnt_added, ==, 1); |
| g_assert_cmpint(cnt_removed, ==, 0); |
| |
| /* load another device with a different VID:PID, and check that we *did* get a replug */ |
| usb_emulate_fn3 = |
| g_test_build_filename(G_TEST_DIST, "tests", "usb-devices-bootloader.json", NULL); |
| g_assert_nonnull(usb_emulate_fn3); |
| fu_backend_usb_load_file(backend, usb_emulate_fn3); |
| g_assert_cmpint(cnt_added, ==, 2); |
| g_assert_cmpint(cnt_removed, ==, 1); |
| } |
| |
| static void |
| fu_backend_usb_invalid_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| FuDevice *device_tmp; |
| g_autofree gchar *usb_emulate_fn = NULL; |
| g_autoptr(FuBackend) backend = fu_usb_backend_new(self->ctx); |
| g_autoptr(FuDeviceLocker) locker = NULL; |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(JsonParser) parser = json_parser_new(); |
| |
| #ifndef SUPPORTED_BUILD |
| g_test_expect_message("FuUsbDevice", |
| G_LOG_LEVEL_WARNING, |
| "*invalid platform version 0x0000000a, expected >= 0x00010805*"); |
| g_test_expect_message("FuUsbDevice", |
| G_LOG_LEVEL_WARNING, |
| "failed to parse * BOS descriptor: *did not find magic*"); |
| #endif |
| |
| /* load the JSON into the backend */ |
| g_object_set(backend, "device-gtype", FU_TYPE_USB_DEVICE, NULL); |
| usb_emulate_fn = |
| g_test_build_filename(G_TEST_DIST, "tests", "usb-devices-invalid.json", NULL); |
| ret = json_parser_load_from_file(parser, usb_emulate_fn, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_backend_setup(backend, FU_BACKEND_SETUP_FLAG_NONE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fwupd_codec_from_json(FWUPD_CODEC(backend), json_parser_get_root(parser), &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_backend_coldplug(backend, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| devices = fu_backend_get_devices(backend); |
| g_assert_cmpint(devices->len, ==, 1); |
| device_tmp = g_ptr_array_index(devices, 0); |
| fu_device_set_context(device_tmp, self->ctx); |
| |
| locker = fu_device_locker_new(device_tmp, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(locker); |
| |
| /* check the device was processed correctly by FuUsbDevice */ |
| g_assert_cmpstr(fu_device_get_name(device_tmp), ==, "ColorHug2"); |
| g_assert_true(fu_device_has_instance_id(device_tmp, |
| "USB\\VID_273F&PID_1004", |
| FU_DEVICE_INSTANCE_FLAG_VISIBLE)); |
| g_assert_true(fu_device_has_vendor_id(device_tmp, "USB:0x273F")); |
| |
| /* check the fwupd DS20 descriptor was *not* parsed */ |
| g_assert_false(fu_device_has_icon(device_tmp, "computer")); |
| } |
| |
| static void |
| fu_plugin_module_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| GError *error = NULL; |
| gboolean ret; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* create a fake device */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_plugin_set_config_value(plugin, "RegistrationSupported", "true", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_plugin_runner_startup(plugin, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_signal_connect(FU_PLUGIN(plugin), |
| "device-added", |
| G_CALLBACK(fu_test_plugin_device_added_cb), |
| &device); |
| g_signal_connect(FU_PLUGIN(plugin), |
| "device-register", |
| G_CALLBACK(fu_test_plugin_device_register_cb), |
| &device); |
| ret = fu_plugin_runner_coldplug(plugin, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check we did the right thing */ |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "08d460be0f1f9f128413f816022a6439e0078018"); |
| g_assert_cmpstr(fu_device_get_version_lowest(device), ==, "1.2.0"); |
| g_assert_cmpstr(fu_device_get_version(device), ==, "1.2.2"); |
| g_assert_cmpstr(fu_device_get_version_bootloader(device), ==, "0.1.2"); |
| g_assert_cmpstr(fu_device_get_guid_default(device), |
| ==, |
| "b585990a-003e-5270-89d5-3705a17f9a43"); |
| g_assert_cmpstr(fu_device_get_name(device), ==, "Integrated Webcamâ„¢"); |
| g_signal_handlers_disconnect_by_data(plugin, &device); |
| } |
| |
| static void |
| fu_history_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| GPtrArray *checksums; |
| gboolean ret; |
| FuDevice *device; |
| FuRelease *release; |
| g_autoptr(FuDevice) device_found = NULL; |
| g_autoptr(FuHistory) history = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GPtrArray) approved_firmware = NULL; |
| g_autofree gchar *dirname = NULL; |
| g_autofree gchar *filename = NULL; |
| |
| /* create */ |
| history = fu_history_new(self->ctx); |
| g_assert_nonnull(history); |
| |
| /* delete the database */ |
| dirname = fu_path_from_kind(FU_PATH_KIND_LOCALSTATEDIR_PKG); |
| if (!g_file_test(dirname, G_FILE_TEST_IS_DIR)) |
| return; |
| filename = g_build_filename(dirname, "pending.db", NULL); |
| (void)g_unlink(filename); |
| |
| /* add a device */ |
| device = fu_device_new(self->ctx); |
| fu_device_set_id(device, "self-test"); |
| fu_device_set_name(device, "ColorHug"), |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_device_set_version(device, "3.0.1"), |
| fu_device_set_update_state(device, FWUPD_UPDATE_STATE_FAILED); |
| fu_device_set_update_error(device, "word"); |
| fu_device_add_instance_id(device, "827edddd-9bb6-5632-889f-2c01255503da"); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_INTERNAL); |
| fu_device_set_created_usec(device, 1514338000ull * G_USEC_PER_SEC); |
| fu_device_set_modified_usec(device, 1514338999ull * G_USEC_PER_SEC); |
| release = fu_release_new(); |
| fu_release_set_filename(release, "/var/lib/dave.cap"), |
| fu_release_add_checksum(release, "abcdef"); |
| fu_release_set_version(release, "3.0.2"); |
| fu_release_add_metadata_item(release, "FwupdVersion", VERSION); |
| ret = fu_history_add_device(history, device, release, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_object_unref(release); |
| |
| /* ensure database was created */ |
| g_assert_true(g_file_test(filename, G_FILE_TEST_EXISTS)); |
| |
| g_object_unref(device); |
| |
| /* get device */ |
| device = fu_history_get_device_by_id(history, |
| "2ba16d10df45823dd4494ff10a0bfccfef512c9d", |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_device_get_id(device), ==, "2ba16d10df45823dd4494ff10a0bfccfef512c9d"); |
| g_assert_cmpstr(fu_device_get_name(device), ==, "ColorHug"); |
| g_assert_cmpstr(fu_device_get_version(device), ==, "3.0.1"); |
| g_assert_cmpint(fu_device_get_update_state(device), ==, FWUPD_UPDATE_STATE_FAILED); |
| g_assert_cmpstr(fu_device_get_update_error(device), ==, "word"); |
| g_assert_cmpstr(fu_device_get_guid_default(device), |
| ==, |
| "827edddd-9bb6-5632-889f-2c01255503da"); |
| g_assert_cmpint(fu_device_get_flags(device), |
| ==, |
| FWUPD_DEVICE_FLAG_INTERNAL | FWUPD_DEVICE_FLAG_HISTORICAL); |
| g_assert_cmpint(fu_device_get_created_usec(device), ==, 1514338000ull * G_USEC_PER_SEC); |
| g_assert_cmpint(fu_device_get_modified_usec(device), ==, 1514338999ull * G_USEC_PER_SEC); |
| release = FU_RELEASE(fu_device_get_release_default(device)); |
| g_assert_nonnull(release); |
| g_assert_cmpstr(fu_release_get_version(release), ==, "3.0.2"); |
| g_assert_cmpstr(fu_release_get_filename(release), ==, "/var/lib/dave.cap"); |
| g_assert_cmpstr(fu_release_get_metadata_item(release, "FwupdVersion"), ==, VERSION); |
| checksums = fu_release_get_checksums(release); |
| g_assert_nonnull(checksums); |
| g_assert_cmpint(checksums->len, ==, 1); |
| g_assert_cmpstr(fwupd_checksum_get_by_kind(checksums, G_CHECKSUM_SHA1), ==, "abcdef"); |
| ret = fu_history_add_device(history, device, release, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* get device that does not exist */ |
| device_found = fu_history_get_device_by_id(history, "XXXXXXXXXXXXX", &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_null(device_found); |
| g_clear_error(&error); |
| |
| /* get device that does exist */ |
| device_found = fu_history_get_device_by_id(history, |
| "2ba16d10df45823dd4494ff10a0bfccfef512c9d", |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device_found); |
| g_object_unref(device_found); |
| |
| /* remove device */ |
| ret = fu_history_remove_device(history, device, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_object_unref(device); |
| |
| /* get device that does not exist */ |
| device_found = fu_history_get_device_by_id(history, |
| "2ba16d10df45823dd4494ff10a0bfccfef512c9d", |
| &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_null(device_found); |
| g_clear_error(&error); |
| |
| /* approved firmware */ |
| ret = fu_history_clear_approved_firmware(history, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_history_add_approved_firmware(history, "foo", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_history_add_approved_firmware(history, "bar", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| approved_firmware = fu_history_get_approved_firmware(history, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(approved_firmware); |
| g_assert_cmpint(approved_firmware->len, ==, 2); |
| g_assert_cmpstr(g_ptr_array_index(approved_firmware, 0), ==, "foo"); |
| g_assert_cmpstr(g_ptr_array_index(approved_firmware, 1), ==, "bar"); |
| |
| /* emulation-tag */ |
| ret = fu_history_add_emulation_tag(history, "id", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_history_has_emulation_tag(history, "id", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_history_has_emulation_tag(history, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_history_remove_emulation_tag(history, "id", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_history_remove_emulation_tag(history, "id", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_history_has_emulation_tag(history, "id", &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_false(ret); |
| } |
| |
| static GBytes * |
| fu_test_build_cab(gboolean compressed, ...) |
| { |
| gboolean ret; |
| va_list args; |
| g_autoptr(FuCabFirmware) cabinet = fu_cab_firmware_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GBytes) cabinet_blob = NULL; |
| |
| fu_cab_firmware_set_compressed(cabinet, compressed); |
| |
| /* add each file */ |
| va_start(args, compressed); |
| do { |
| const gchar *fn; |
| const gchar *text; |
| g_autoptr(FuCabImage) img = fu_cab_image_new(); |
| g_autoptr(GBytes) blob = NULL; |
| |
| /* get filename */ |
| fn = va_arg(args, const gchar *); |
| if (fn == NULL) |
| break; |
| |
| /* get contents */ |
| text = va_arg(args, const gchar *); |
| if (text == NULL) |
| break; |
| g_debug("creating %s with %s", fn, text); |
| |
| /* add a GCabFile to the cabinet */ |
| blob = g_bytes_new_static(text, strlen(text)); |
| fu_firmware_set_id(FU_FIRMWARE(img), fn); |
| fu_firmware_set_bytes(FU_FIRMWARE(img), blob); |
| ret = fu_firmware_add_image(FU_FIRMWARE(cabinet), FU_FIRMWARE(img), &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } while (TRUE); |
| va_end(args); |
| |
| /* write the archive to a blob */ |
| cabinet_blob = fu_firmware_write(FU_FIRMWARE(cabinet), &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(cabinet_blob); |
| return g_steal_pointer(&cabinet_blob); |
| } |
| |
| static void |
| fu_test_plugin_composite_device_added_cb(FuPlugin *plugin, FuDevice *device, gpointer user_data) |
| { |
| GPtrArray *devices = (GPtrArray *)user_data; |
| g_ptr_array_add(devices, g_object_ref(device)); |
| } |
| |
| static gint |
| fu_plugin_composite_release_sort_cb(gconstpointer a, gconstpointer b) |
| { |
| FuRelease *release1 = *((FuRelease **)a); |
| FuRelease *release2 = *((FuRelease **)b); |
| FuDevice *device1 = fu_release_get_device(release1); |
| FuDevice *device2 = fu_release_get_device(release2); |
| if (fu_device_get_order(device1) < fu_device_get_order(device2)) |
| return 1; |
| if (fu_device_get_order(device1) > fu_device_get_order(device2)) |
| return -1; |
| return 0; |
| } |
| |
| static void |
| fu_plugin_composite_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FuDevice *dev_tmp; |
| GError *error = NULL; |
| gboolean ret; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GPtrArray) components = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(GPtrArray) releases = |
| g_ptr_array_new_with_free_func((GDestroyNotify)g_object_unref); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(XbSilo) silo_empty = xb_silo_new(); |
| |
| /* no metadata in daemon */ |
| fu_engine_set_silo(engine, silo_empty); |
| |
| /* create CAB file */ |
| blob = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\"/>\n" |
| " </releases>\n" |
| "</component>", |
| "acme.module1.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware.module1</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">7fddead7-12b5-4fb9-9fa0-6d30305df755</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"2\"/>\n" |
| " </releases>\n" |
| " <custom>\n" |
| " <value key=\"LVFS::VersionFormat\">plain</value>\n" |
| " </custom>\n" |
| "</component>", |
| "acme.module2.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware.module2</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b8fe6b45-8702-4bcd-8120-ef236caac76f</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"11\"/>\n" |
| " </releases>\n" |
| " <custom>\n" |
| " <value key=\"LVFS::VersionFormat\">plain</value>\n" |
| " </custom>\n" |
| "</component>", |
| "firmware.bin", |
| "world", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_BLOB, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| components = fu_cabinet_get_components(cabinet, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(components); |
| g_assert_cmpint(components->len, ==, 3); |
| |
| /* set up dummy plugin */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_plugin_set_config_value(plugin, "CompositeChild", "true", &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_engine_add_plugin(engine, plugin); |
| |
| ret = fu_plugin_runner_startup(plugin, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| devices = g_ptr_array_new_with_free_func((GDestroyNotify)g_object_unref); |
| g_signal_connect(FU_PLUGIN(plugin), |
| "device-added", |
| G_CALLBACK(fu_test_plugin_composite_device_added_cb), |
| devices); |
| |
| ret = fu_plugin_runner_coldplug(plugin, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check we found all composite devices */ |
| g_assert_cmpint(devices->len, ==, 3); |
| for (guint i = 0; i < devices->len; i++) { |
| FuDevice *device = g_ptr_array_index(devices, i); |
| fu_engine_add_device(engine, device); |
| if (g_strcmp0(fu_device_get_id(device), |
| "08d460be0f1f9f128413f816022a6439e0078018") == 0) { |
| g_assert_cmpstr(fu_device_get_version(device), ==, "1.2.2"); |
| } else if (g_strcmp0(fu_device_get_id(device), |
| "c0a0a4aa6480ac28eea1ce164fbb466ca934e1ff") == 0) { |
| g_assert_cmpstr(fu_device_get_version(device), ==, "1"); |
| g_assert_nonnull(fu_device_get_parent(device)); |
| } else if (g_strcmp0(fu_device_get_id(device), |
| "bf455e9f371d2608d1cb67660fd2b335d3f6ef73") == 0) { |
| g_assert_cmpstr(fu_device_get_version(device), ==, "10"); |
| g_assert_nonnull(fu_device_get_parent(device)); |
| } |
| } |
| |
| /* produce install tasks */ |
| for (guint i = 0; i < components->len; i++) { |
| XbNode *component = g_ptr_array_index(components, i); |
| |
| /* do any devices pass the requirements */ |
| for (guint j = 0; j < devices->len; j++) { |
| FuDevice *device = g_ptr_array_index(devices, j); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(GError) error_local = NULL; |
| |
| /* is this component valid for the device */ |
| fu_release_set_device(release, device); |
| fu_release_set_request(release, request); |
| if (!fu_release_load(release, |
| cabinet, |
| component, |
| NULL, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error_local)) { |
| g_debug("requirement on %s:%s failed: %s", |
| fu_device_get_id(device), |
| xb_node_query_text(component, "id", NULL), |
| error_local->message); |
| continue; |
| } |
| |
| g_ptr_array_add(releases, g_steal_pointer(&release)); |
| } |
| } |
| g_assert_cmpint(releases->len, ==, 3); |
| |
| /* sort these by version, forcing fu_engine_install_releases() to sort by device order */ |
| g_ptr_array_sort(releases, fu_plugin_composite_release_sort_cb); |
| dev_tmp = fu_release_get_device(FU_RELEASE(g_ptr_array_index(releases, 0))); |
| g_assert_cmpstr(fu_device_get_logical_id(dev_tmp), ==, "child1"); |
| dev_tmp = fu_release_get_device(FU_RELEASE(g_ptr_array_index(releases, 1))); |
| g_assert_cmpstr(fu_device_get_logical_id(dev_tmp), ==, "child2"); |
| dev_tmp = fu_release_get_device(FU_RELEASE(g_ptr_array_index(releases, 2))); |
| g_assert_cmpstr(fu_device_get_logical_id(dev_tmp), ==, NULL); |
| |
| /* install the cab */ |
| ret = fu_engine_install_releases(engine, |
| request, |
| releases, |
| cabinet, |
| progress, |
| FWUPD_INSTALL_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* verify we installed the parent first */ |
| dev_tmp = fu_release_get_device(FU_RELEASE(g_ptr_array_index(releases, 0))); |
| g_assert_cmpstr(fu_device_get_logical_id(dev_tmp), ==, NULL); |
| dev_tmp = fu_release_get_device(FU_RELEASE(g_ptr_array_index(releases, 1))); |
| g_assert_cmpstr(fu_device_get_logical_id(dev_tmp), ==, "child2"); |
| dev_tmp = fu_release_get_device(FU_RELEASE(g_ptr_array_index(releases, 2))); |
| g_assert_cmpstr(fu_device_get_logical_id(dev_tmp), ==, "child1"); |
| |
| /* verify everything upgraded */ |
| for (guint i = 0; i < devices->len; i++) { |
| FuDevice *device = g_ptr_array_index(devices, i); |
| const gchar *metadata; |
| if (g_strcmp0(fu_device_get_id(device), |
| "08d460be0f1f9f128413f816022a6439e0078018") == 0) { |
| g_assert_cmpstr(fu_device_get_version(device), ==, "1.2.3"); |
| } else if (g_strcmp0(fu_device_get_id(device), |
| "c0a0a4aa6480ac28eea1ce164fbb466ca934e1ff") == 0) { |
| g_assert_cmpstr(fu_device_get_version(device), ==, "2"); |
| } else if (g_strcmp0(fu_device_get_id(device), |
| "bf455e9f371d2608d1cb67660fd2b335d3f6ef73") == 0) { |
| g_assert_cmpstr(fu_device_get_version(device), ==, "11"); |
| } |
| |
| /* verify prepare and cleanup ran on all devices */ |
| metadata = fu_device_get_metadata(device, "frimbulator"); |
| g_assert_cmpstr(metadata, ==, "1"); |
| metadata = fu_device_get_metadata(device, "frombulator"); |
| g_assert_cmpstr(metadata, ==, "1"); |
| } |
| } |
| |
| static void |
| fu_plugin_composite_multistep_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| g_autoptr(FuPlugin) plugin = fu_plugin_new_from_gtype(fu_test_plugin_get_type(), self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GPtrArray) devices = NULL; |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add the fake metadata */ |
| ret = xb_builder_source_load_xml( |
| source, |
| "<?xml version=\"1.0\"?>\n" |
| "<components>\n" |
| "<component type=\"firmware\">\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <custom>\n" |
| " <value key=\"LVFS::VersionFormat\">triplet</value>\n" |
| " <value key=\"LVFS::UpdateProtocol\">com.acme.test</value>\n" |
| " </custom>\n" |
| " <releases>\n" |
| " <release id=\"1\" version=\"1.2.3\">\n" |
| " <checksum type=\"sha1\" target=\"content\">aaa</checksum>\n" |
| " <artifacts>\n" |
| " <artifact type=\"binary\">\n" |
| " <location>file://filename.cab</location>\n" |
| " <checksum type=\"sha1\">ccc</checksum>\n" |
| " </artifact>\n" |
| " </artifacts>\n" |
| " </release>\n" |
| " <release id=\"1\" version=\"1.2.4\">\n" |
| " <checksum type=\"sha1\" target=\"content\">bbb</checksum>\n" |
| " <artifacts>\n" |
| " <artifact type=\"binary\">\n" |
| " <location>file://filename.cab</location>\n" |
| " <checksum type=\"sha1\">ccc</checksum>\n" |
| " </artifact>\n" |
| " </artifacts>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>\n" |
| "</components>", |
| XB_BUILDER_SOURCE_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* set up dummy plugin */ |
| ret = fu_plugin_reset_config_values(plugin, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fu_engine_add_plugin(engine, plugin); |
| ret = fu_plugin_runner_startup(plugin, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| devices = g_ptr_array_new_with_free_func((GDestroyNotify)g_object_unref); |
| g_signal_connect(FU_PLUGIN(plugin), |
| "device-added", |
| G_CALLBACK(fu_test_plugin_composite_device_added_cb), |
| devices); |
| ret = fu_plugin_runner_coldplug(plugin, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add all the found devices */ |
| g_assert_cmpint(devices->len, ==, 1); |
| for (guint i = 0; i < devices->len; i++) { |
| FuDevice *device_tmp = g_ptr_array_index(devices, i); |
| fu_engine_add_device(engine, device_tmp); |
| } |
| |
| /* check we did not dedupe the composite cab */ |
| device = fu_engine_get_device(engine, "08d460be0f1f9f128413f816022a6439e0078018", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| releases = fu_engine_get_releases(engine, |
| request, |
| "08d460be0f1f9f128413f816022a6439e0078018", |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases); |
| g_assert_cmpint(releases->len, ==, 2); |
| } |
| |
| static void |
| fu_plugin_engine_get_results_appstream_id_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| FwupdRelease *release_default; |
| g_autoptr(FuDevice) device_tmp = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuHistory) history = fu_history_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(FuRelease) release = fu_release_new(); |
| g_autoptr(FwupdDevice) device = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add the fake metadata */ |
| ret = xb_builder_source_load_xml( |
| source, |
| "<?xml version=\"1.0\"?>\n" |
| "<components>\n" |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.WRONGDEVICE.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">00000000-0000-0000-0000-000000000000</firmware>\n" |
| " </provides>\n" |
| " <custom>\n" |
| " <value key=\"LVFS::VersionFormat\">triplet</value>\n" |
| " <value key=\"LVFS::UpdateProtocol\">com.acme.test</value>\n" |
| " </custom>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\">\n" |
| " <checksum type=\"sha1\" target=\"content\">aaa</checksum>\n" |
| " <artifacts>\n" |
| " <artifact type=\"binary\">\n" |
| " <checksum type=\"sha1\">7c211433f02071597741e6ff5a8ea34789abbf43</checksum>\n" |
| " </artifact>\n" |
| " </artifacts>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>\n" |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <custom>\n" |
| " <value key=\"LVFS::VersionFormat\">triplet</value>\n" |
| " <value key=\"LVFS::UpdateProtocol\">com.acme.test</value>\n" |
| " </custom>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\">\n" |
| " <checksum type=\"sha1\" target=\"content\">aaa</checksum>\n" |
| " <artifacts>\n" |
| " <artifact type=\"binary\">\n" |
| " <checksum type=\"sha1\">7c211433f02071597741e6ff5a8ea34789abbf43</checksum>\n" |
| " </artifact>\n" |
| " </artifacts>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>\n" |
| "</components>", |
| XB_BUILDER_SOURCE_FLAG_NONE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device_tmp, "08d460be0f1f9f128413f816022a6439e0078018"); |
| fu_engine_add_device(engine, device_tmp); |
| fu_release_set_appstream_id(release, "com.acme.example.firmware"); |
| fu_release_add_checksum(release, "7c211433f02071597741e6ff5a8ea34789abbf43"); |
| fu_device_add_release(device_tmp, FWUPD_RELEASE(release)); |
| fu_device_set_update_state(device_tmp, FWUPD_UPDATE_STATE_SUCCESS); |
| ret = fu_history_add_device(history, device_tmp, release, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check we got the correct component */ |
| device = fu_engine_get_results(engine, "08d460be0f1f9f128413f816022a6439e0078018", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| release_default = fu_device_get_release_default(device); |
| g_assert_nonnull(release_default); |
| g_assert_cmpstr(fwupd_release_get_appstream_id(release_default), |
| ==, |
| "com.acme.example.firmware"); |
| } |
| |
| static void |
| fu_security_attr_func(gconstpointer user_data) |
| { |
| gboolean ret; |
| g_autofree gchar *json1 = NULL; |
| g_autofree gchar *json2 = NULL; |
| g_autoptr(FuSecurityAttrs) attrs1 = fu_security_attrs_new(); |
| g_autoptr(FuSecurityAttrs) attrs2 = fu_security_attrs_new(); |
| g_autoptr(FwupdSecurityAttr) attr1 = fwupd_security_attr_new("org.fwupd.hsi.foo"); |
| g_autoptr(FwupdSecurityAttr) attr2 = fwupd_security_attr_new("org.fwupd.hsi.bar"); |
| g_autoptr(GError) error = NULL; |
| |
| fwupd_security_attr_set_plugin(attr1, "foo"); |
| fwupd_security_attr_set_created(attr1, 0); |
| fwupd_security_attr_set_plugin(attr2, "bar"); |
| fwupd_security_attr_set_created(attr2, 0); |
| fu_security_attrs_append(attrs1, attr1); |
| fu_security_attrs_append(attrs1, attr2); |
| |
| json1 = fwupd_codec_to_json_string(FWUPD_CODEC(attrs1), FWUPD_CODEC_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(json1); |
| ret = g_strcmp0(json1, |
| "{\n" |
| " \"SecurityAttributes\" : [\n" |
| " {\n" |
| " \"AppstreamId\" : \"org.fwupd.hsi.foo\",\n" |
| " \"HsiLevel\" : 0,\n" |
| " \"Plugin\" : \"foo\",\n" |
| " \"Uri\" : " |
| "\"https://fwupd.github.io/libfwupdplugin/hsi.html#org.fwupd.hsi.foo\"\n" |
| " },\n" |
| " {\n" |
| " \"AppstreamId\" : \"org.fwupd.hsi.bar\",\n" |
| " \"HsiLevel\" : 0,\n" |
| " \"Plugin\" : \"bar\",\n" |
| " \"Uri\" : " |
| "\"https://fwupd.github.io/libfwupdplugin/hsi.html#org.fwupd.hsi.bar\"\n" |
| " }\n" |
| " ]\n" |
| "}") == 0; |
| g_assert_true(ret); |
| |
| ret = fwupd_codec_from_json_string(FWUPD_CODEC(attrs2), json1, &error); |
| if (g_error_matches(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED)) { |
| g_test_skip(error->message); |
| return; |
| } |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| json2 = fwupd_codec_to_json_string(FWUPD_CODEC(attrs2), FWUPD_CODEC_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(json2); |
| ret = g_strcmp0(json2, json1) == 0; |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_common_cabinet_func(void) |
| { |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(FuFirmware) img1 = NULL; |
| g_autoptr(FuFirmware) img2 = NULL; |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GBytes) jcat_blob1 = g_bytes_new_static("hello", 6); |
| g_autoptr(GBytes) jcat_blob2 = g_bytes_new_static("hellX", 6); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| |
| filename = g_test_build_filename(G_TEST_BUILT, |
| "tests", |
| "multiple-rels", |
| "multiple-rels-1.2.4.cab", |
| NULL); |
| stream = fu_input_stream_from_path(filename, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(stream); |
| ret = fu_firmware_parse_stream(FU_FIRMWARE(cabinet), |
| stream, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_STREAM, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* add */ |
| ret = fu_cabinet_add_file(cabinet, "firmware.jcat", jcat_blob1, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* replace */ |
| ret = fu_cabinet_add_file(cabinet, "firmware.jcat", jcat_blob2, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* get data */ |
| img1 = fu_firmware_get_image_by_id(FU_FIRMWARE(cabinet), "firmware.jcat", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(img1); |
| blob = fu_firmware_get_bytes(FU_FIRMWARE(img1), &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(blob); |
| g_assert_cmpstr(g_bytes_get_data(blob, NULL), ==, "hellX"); |
| |
| /* get data that does not exist */ |
| img2 = fu_firmware_get_image_by_id(FU_FIRMWARE(cabinet), "foo.jcat", &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND); |
| g_assert_null(img2); |
| } |
| |
| static void |
| fu_memcpy_func(gconstpointer user_data) |
| { |
| const guint8 src[] = {'a', 'b', 'c', 'd', 'e'}; |
| gboolean ret; |
| guint8 dst[4] = {0}; |
| g_autoptr(GError) error = NULL; |
| |
| /* copy entire buffer */ |
| ret = fu_memcpy_safe(dst, sizeof(dst), 0x0, src, sizeof(src), 0x0, 4, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(memcmp(src, dst, 4), ==, 0); |
| |
| /* copy first char */ |
| ret = fu_memcpy_safe(dst, sizeof(dst), 0x0, src, sizeof(src), 0x0, 1, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(dst[0], ==, 'a'); |
| |
| /* copy last char */ |
| ret = fu_memcpy_safe(dst, sizeof(dst), 0x0, src, sizeof(src), 0x4, 1, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(dst[0], ==, 'e'); |
| |
| /* copy nothing */ |
| ret = fu_memcpy_safe(dst, sizeof(dst), 0x0, src, sizeof(src), 0x0, 0, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* write past the end of dst */ |
| ret = fu_memcpy_safe(dst, sizeof(dst), 0x0, src, sizeof(src), 0x0, 5, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_WRITE); |
| g_assert_false(ret); |
| g_clear_error(&error); |
| |
| /* write past the end of dst with offset */ |
| ret = fu_memcpy_safe(dst, sizeof(dst), 0x1, src, sizeof(src), 0x0, 4, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_WRITE); |
| g_assert_false(ret); |
| g_clear_error(&error); |
| |
| /* read past the end of dst */ |
| ret = fu_memcpy_safe(dst, sizeof(dst), 0x0, src, sizeof(src), 0x0, 6, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_READ); |
| g_assert_false(ret); |
| g_clear_error(&error); |
| |
| /* read past the end of src with offset */ |
| ret = fu_memcpy_safe(dst, sizeof(dst), 0x0, src, sizeof(src), 0x4, 4, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_READ); |
| g_assert_false(ret); |
| g_clear_error(&error); |
| } |
| |
| static void |
| fu_console_func(gconstpointer user_data) |
| { |
| g_autoptr(FuConsole) console = fu_console_new(); |
| |
| fu_console_set_status_length(console, 20); |
| fu_console_set_percentage_length(console, 50); |
| |
| g_print("\n"); |
| for (guint i = 0; i < 100; i++) { |
| fu_console_set_progress(console, FWUPD_STATUS_DECOMPRESSING, i); |
| g_usleep(10000); |
| } |
| fu_console_set_progress(console, FWUPD_STATUS_IDLE, 0); |
| for (guint i = 0; i < 100; i++) { |
| guint pc = (i > 25 && i < 75) ? 0 : i; |
| fu_console_set_progress(console, FWUPD_STATUS_LOADING, pc); |
| g_usleep(10000); |
| } |
| fu_console_set_progress(console, FWUPD_STATUS_IDLE, 0); |
| |
| for (guint i = 0; i < 5000; i++) { |
| fu_console_set_progress(console, FWUPD_STATUS_LOADING, 0); |
| g_usleep(1000); |
| } |
| fu_console_set_progress(console, FWUPD_STATUS_IDLE, 0); |
| } |
| |
| static gint |
| fu_release_compare_func_cb(gconstpointer a, gconstpointer b) |
| { |
| FuRelease *release1 = *((FuRelease **)a); |
| FuRelease *release2 = *((FuRelease **)b); |
| return fu_release_compare(release1, release2); |
| } |
| |
| static void |
| fu_release_compare_func(gconstpointer user_data) |
| { |
| g_autoptr(GPtrArray) releases = g_ptr_array_new(); |
| g_autoptr(FuDevice) device1 = fu_device_new(NULL); |
| g_autoptr(FuDevice) device2 = fu_device_new(NULL); |
| g_autoptr(FuDevice) device3 = fu_device_new(NULL); |
| g_autoptr(FuRelease) release1 = fu_release_new(); |
| g_autoptr(FuRelease) release2 = fu_release_new(); |
| g_autoptr(FuRelease) release3 = fu_release_new(); |
| |
| fu_device_set_order(device1, 33); |
| fu_release_set_device(release1, device1); |
| fu_release_set_priority(release1, 0); |
| fu_release_set_branch(release1, "1"); |
| |
| fu_device_set_order(device2, 11); |
| fu_release_set_device(release2, device2); |
| fu_release_set_priority(release2, 0); |
| fu_release_set_branch(release2, "2"); |
| |
| fu_device_set_order(device3, 11); |
| fu_release_set_device(release3, device3); |
| fu_release_set_priority(release3, 99); |
| fu_release_set_branch(release3, "3"); |
| |
| g_ptr_array_add(releases, release1); |
| g_ptr_array_add(releases, release2); |
| g_ptr_array_add(releases, release3); |
| |
| /* order the install tasks */ |
| g_ptr_array_sort(releases, fu_release_compare_func_cb); |
| g_assert_cmpint(releases->len, ==, 3); |
| g_assert_cmpstr(fu_release_get_branch(g_ptr_array_index(releases, 0)), ==, "3"); |
| g_assert_cmpstr(fu_release_get_branch(g_ptr_array_index(releases, 1)), ==, "2"); |
| g_assert_cmpstr(fu_release_get_branch(g_ptr_array_index(releases, 2)), ==, "1"); |
| } |
| |
| static void |
| fu_release_uri_scheme_func(void) |
| { |
| struct { |
| const gchar *in; |
| const gchar *op; |
| } strs[] = {{"https://foo.bar/baz", "https"}, |
| {"HTTP://FOO.BAR/BAZ", "http"}, |
| {"ftp://", "ftp"}, |
| {"ftp:", "ftp"}, |
| {"foobarbaz", NULL}, |
| {"", NULL}, |
| {NULL, NULL}}; |
| for (guint i = 0; strs[i].in != NULL; i++) { |
| g_autofree gchar *tmp = fu_release_uri_get_scheme(strs[i].in); |
| g_assert_cmpstr(tmp, ==, strs[i].op); |
| } |
| } |
| |
| static void |
| fu_release_trusted_report_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FwupdRelease *rel; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FwupdRemote) remote = fwupd_remote_new(); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GFile) file = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderNode) custom = xb_builder_node_new("custom"); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| |
| /* add fake LVFS remote */ |
| fwupd_remote_set_id(remote, "lvfs"); |
| fu_engine_add_remote(engine, remote); |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* metadata with vendor id=123 */ |
| filename = g_test_build_filename(G_TEST_DIST, "tests", "metadata-report1.xml", NULL); |
| file = g_file_new_for_path(filename); |
| ret = xb_builder_source_load_file(source, file, XB_BUILDER_SOURCE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_node_insert_text(custom, "value", "lvfs", "key", "fwupd::RemoteId", NULL); |
| xb_builder_source_set_info(source, custom); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "dummy"); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_engine_add_device(engine, device); |
| |
| /* ensure we set this as trusted */ |
| releases = fu_engine_get_releases_for_device(engine, request, device, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases); |
| g_assert_cmpint(releases->len, ==, 1); |
| rel = g_ptr_array_index(releases, 0); |
| g_assert_true(fwupd_release_has_flag(rel, FWUPD_RELEASE_FLAG_TRUSTED_REPORT)); |
| } |
| |
| static void |
| fu_release_trusted_report_oem_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FwupdRelease *rel; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GFile) file = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* metadata with FromOEM */ |
| filename = g_test_build_filename(G_TEST_DIST, "tests", "metadata-report2.xml", NULL); |
| file = g_file_new_for_path(filename); |
| ret = xb_builder_source_load_file(source, file, XB_BUILDER_SOURCE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "dummy"); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_engine_add_device(engine, device); |
| |
| /* ensure we set this as trusted */ |
| releases = fu_engine_get_releases_for_device(engine, request, device, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases); |
| g_assert_cmpint(releases->len, ==, 1); |
| rel = g_ptr_array_index(releases, 0); |
| g_assert_true(fwupd_release_has_flag(rel, FWUPD_RELEASE_FLAG_TRUSTED_REPORT)); |
| } |
| |
| static void |
| fu_release_no_trusted_report_upgrade_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FwupdRelease *rel; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GFile) file = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* metadata with FromOEM, but *NOT* an upgrade */ |
| filename = g_test_build_filename(G_TEST_DIST, "tests", "metadata-report4.xml", NULL); |
| file = g_file_new_for_path(filename); |
| ret = xb_builder_source_load_file(source, file, XB_BUILDER_SOURCE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "dummy"); |
| fu_device_set_version(device, "1.2.3"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_engine_add_device(engine, device); |
| |
| /* ensure we set this as trusted */ |
| releases = fu_engine_get_releases_for_device(engine, request, device, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases); |
| g_assert_cmpint(releases->len, ==, 1); |
| rel = g_ptr_array_index(releases, 0); |
| g_assert_false(fwupd_release_has_flag(rel, FWUPD_RELEASE_FLAG_TRUSTED_REPORT)); |
| } |
| |
| static void |
| fu_release_no_trusted_report_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| FwupdRelease *rel; |
| gboolean ret; |
| g_autofree gchar *filename = NULL; |
| g_autoptr(FuDevice) device = fu_device_new(self->ctx); |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GFile) file = NULL; |
| g_autoptr(XbBuilder) builder = xb_builder_new(); |
| g_autoptr(XbBuilderSource) source = xb_builder_source_new(); |
| g_autoptr(XbSilo) silo = NULL; |
| g_autoptr(GPtrArray) releases = NULL; |
| g_autoptr(FuEngineRequest) request = fu_engine_request_new(NULL); |
| |
| /* load engine to get FuConfig set up */ |
| ret = fu_engine_load(engine, FU_ENGINE_LOAD_FLAG_NO_CACHE, progress, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* metadata without OEM or valid VendorId as per tests/fwupd.conf */ |
| filename = g_test_build_filename(G_TEST_DIST, "tests", "metadata-report3.xml", NULL); |
| file = g_file_new_for_path(filename); |
| ret = xb_builder_source_load_file(source, file, XB_BUILDER_SOURCE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| xb_builder_import_source(builder, source); |
| silo = xb_builder_compile(builder, XB_BUILDER_COMPILE_FLAG_NONE, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(silo); |
| fu_engine_set_silo(engine, silo); |
| |
| /* add a dummy device */ |
| fu_device_set_id(device, "dummy"); |
| fu_device_set_version(device, "1.2.2"); |
| fu_device_build_vendor_id_u16(device, "USB", 0xFFFF); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UPDATABLE); |
| fu_device_add_flag(device, FWUPD_DEVICE_FLAG_UNSIGNED_PAYLOAD); |
| fu_device_add_protocol(device, "com.acme"); |
| fu_device_add_instance_id(device, "2d47f29b-83a2-4f31-a2e8-63474f4d4c2e"); |
| fu_device_set_version_format(device, FWUPD_VERSION_FORMAT_TRIPLET); |
| fu_engine_add_device(engine, device); |
| |
| /* ensure trusted reports flag is not set */ |
| releases = fu_engine_get_releases_for_device(engine, request, device, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(releases); |
| g_assert_cmpint(releases->len, ==, 1); |
| rel = g_ptr_array_index(releases, 0); |
| g_assert_false(fwupd_release_has_flag(rel, FWUPD_RELEASE_FLAG_TRUSTED_REPORT)); |
| } |
| |
| static void |
| fu_common_store_cab_func(void) |
| { |
| gboolean ret; |
| GBytes *blob_tmp; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbNode) csum = NULL; |
| g_autoptr(XbNode) rel = NULL; |
| g_autoptr(XbNode) req = NULL; |
| g_autoptr(XbQuery) query = NULL; |
| |
| /* create silo */ |
| blob = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <name>ACME Firmware</name>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">ae56e3fb-6528-5bc4-8b03-012f124075d7</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\" date=\"2017-09-06\">\n" |
| " <size type=\"installed\">5</size>\n" |
| " <checksum filename=\"firmware.dfu\" target=\"content\" " |
| "type=\"sha1\">7c211433f02071597741e6ff5a8ea34789abbf43</checksum>\n" |
| " <description><p>We fixed things</p></description>\n" |
| " </release>\n" |
| " </releases>\n" |
| " <requires>\n" |
| " <id compare=\"ge\" version=\"1.0.1\">org.freedesktop.fwupd</id>\n" |
| " </requires>\n" |
| "</component>", |
| "firmware.dfu", |
| "world", |
| "firmware.dfu.asc", |
| "signature", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_BLOB, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* verify */ |
| component = fu_cabinet_get_component(cabinet, "com.acme.example.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| query = xb_query_new_full(xb_node_get_silo(component), |
| "releases/release", |
| XB_QUERY_FLAG_FORCE_NODE_CACHE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(query); |
| rel = xb_node_query_first_full(component, query, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(rel); |
| g_assert_cmpstr(xb_node_get_attr(rel, "version"), ==, "1.2.3"); |
| csum = xb_node_query_first(rel, "checksum[@target='content']", &error); |
| g_assert_nonnull(csum); |
| g_assert_cmpstr(xb_node_get_text(csum), ==, "7c211433f02071597741e6ff5a8ea34789abbf43"); |
| blob_tmp = xb_node_get_data(rel, "fwupd::FirmwareBasename"); |
| g_assert_nonnull(blob_tmp); |
| req = xb_node_query_first(component, "requires/id", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(req); |
| } |
| |
| static void |
| fu_common_store_cab_artifact_func(void) |
| { |
| gboolean ret; |
| g_autoptr(GBytes) blob1 = NULL; |
| g_autoptr(GBytes) blob2 = NULL; |
| g_autoptr(GBytes) blob3 = NULL; |
| g_autoptr(GBytes) blob4 = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(FuCabinet) cabinet1 = fu_cabinet_new(); |
| g_autoptr(FuCabinet) cabinet2 = fu_cabinet_new(); |
| g_autoptr(FuCabinet) cabinet3 = fu_cabinet_new(); |
| g_autoptr(FuCabinet) cabinet4 = fu_cabinet_new(); |
| |
| /* create silo (sha256, using artifacts object) */ |
| blob1 = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\" date=\"2017-09-06\">\n" |
| " <artifacts>\n" |
| " <artifact type=\"source\">\n" |
| " <filename>firmware.dfu</filename>\n" |
| " <checksum " |
| "type=\"sha256\">486EA46224D1BB4FB680F34F7C9AD96A8F24EC88BE73EA8E5A6C65260E9CB8A7</" |
| "checksum>\n" |
| " </artifact>\n" |
| " </artifacts>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.dfu", |
| "world", |
| "firmware.dfu.asc", |
| "signature", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet1), |
| blob1, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_BLOB, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* create silo (sha1, using artifacts object; mixed case) */ |
| blob2 = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\" date=\"2017-09-06\">\n" |
| " <artifacts>\n" |
| " <artifact type=\"source\">\n" |
| " <filename>firmware.dfu</filename>\n" |
| " <checksum " |
| "type=\"sha1\">7c211433f02071597741e6ff5a8ea34789abbF43</" |
| "checksum>\n" |
| " </artifact>\n" |
| " </artifacts>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.dfu", |
| "world", |
| "firmware.dfu.asc", |
| "signature", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet2), |
| blob2, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_BLOB, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* create silo (sha512, using artifacts object; lower case) */ |
| blob3 = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\" date=\"2017-09-06\">\n" |
| " <artifacts>\n" |
| " <artifact type=\"source\">\n" |
| " <filename>firmware.dfu</filename>\n" |
| " <checksum " |
| "type=\"sha512\">" |
| "11853df40f4b2b919d3815f64792e58d08663767a494bcbb38c0b2389d9140bbb170281b" |
| "4a847be7757bde12c9cd0054ce3652d0ad3a1a0c92babb69798246ee</" |
| "checksum>\n" |
| " </artifact>\n" |
| " </artifacts>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.dfu", |
| "world", |
| "firmware.dfu.asc", |
| "signature", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet3), |
| blob3, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_BLOB, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* create silo (legacy release object) */ |
| blob4 = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\" date=\"2017-09-06\">\n" |
| " <checksum " |
| "target=\"content\" " |
| "filename=\"firmware.dfu\">" |
| "486EA46224D1BB4FB680F34F7C9AD96A8F24EC88BE73EA8E5A6C65260E9CB8A7</" |
| "checksum>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.dfu", |
| "world", |
| "firmware.dfu.asc", |
| "signature", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet4), |
| blob4, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_BLOB, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_common_store_cab_unsigned_func(void) |
| { |
| GBytes *blob_tmp; |
| gboolean ret; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbNode) csum = NULL; |
| g_autoptr(XbNode) rel = NULL; |
| g_autoptr(XbQuery) query = NULL; |
| |
| /* create silo */ |
| blob = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\"/>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.bin", |
| "world", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_BLOB, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* verify */ |
| component = fu_cabinet_get_component(cabinet, "com.acme.example.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| query = xb_query_new_full(xb_node_get_silo(component), |
| "releases/release", |
| XB_QUERY_FLAG_FORCE_NODE_CACHE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(query); |
| rel = xb_node_query_first_full(component, query, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(rel); |
| g_assert_cmpstr(xb_node_get_attr(rel, "version"), ==, "1.2.3"); |
| csum = xb_node_query_first(rel, "checksum[@target='content']", &error); |
| g_assert_null(csum); |
| blob_tmp = xb_node_get_data(rel, "fwupd::FirmwareBasename"); |
| g_assert_nonnull(blob_tmp); |
| } |
| |
| static void |
| fu_common_store_cab_sha256_func(void) |
| { |
| gboolean ret; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| /* create silo */ |
| blob = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\" date=\"2017-09-06\">\n" |
| " <checksum target=\"content\" " |
| "type=\"sha256\">486ea46224d1bb4fb680f34f7c9ad96a8f24ec88be73ea8e5a6c65260e9cb8a7</" |
| "checksum>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.bin", |
| "world", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_BLOB, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_common_store_cab_folder_func(void) |
| { |
| GBytes *blob_tmp; |
| gboolean ret; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(XbNode) component = NULL; |
| g_autoptr(XbNode) rel = NULL; |
| g_autoptr(XbQuery) query = NULL; |
| |
| /* create silo */ |
| blob = fu_test_build_cab( |
| FALSE, |
| "lvfs\\acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\"/>\n" |
| " </releases>\n" |
| "</component>", |
| "lvfs\\firmware.bin", |
| "world", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_CACHE_BLOB, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* verify */ |
| component = fu_cabinet_get_component(cabinet, "com.acme.example.firmware", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(component); |
| query = xb_query_new_full(xb_node_get_silo(component), |
| "releases/release", |
| XB_QUERY_FLAG_FORCE_NODE_CACHE, |
| &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(query); |
| rel = xb_node_query_first_full(component, query, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(rel); |
| g_assert_cmpstr(xb_node_get_attr(rel, "version"), ==, "1.2.3"); |
| blob_tmp = xb_node_get_data(rel, "fwupd::FirmwareBasename"); |
| g_assert_nonnull(blob_tmp); |
| } |
| |
| static void |
| fu_common_store_cab_error_no_metadata_func(void) |
| { |
| gboolean ret; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| blob = fu_test_build_cab(FALSE, "foo.txt", "hello", "bar.txt", "world", NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_NONE, |
| &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_INVALID_FILE); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_common_store_cab_error_wrong_size_func(void) |
| { |
| gboolean ret; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| blob = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\">\n" |
| " <size type=\"installed\">7004701</size>\n" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"sha1\">deadbeef</checksum>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.bin", |
| "world", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_NONE, |
| &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_INVALID_FILE); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_common_store_cab_error_missing_file_func(void) |
| { |
| gboolean ret; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| blob = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\">\n" |
| " <checksum filename=\"firmware.dfu\" target=\"content\"/>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.bin", |
| "world", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_NONE, |
| &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_INVALID_FILE); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_common_store_cab_error_size_func(void) |
| { |
| gboolean ret; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| blob = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\"/>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.bin", |
| "world", |
| NULL); |
| fu_firmware_set_size_max(FU_FIRMWARE(cabinet), 123); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_NONE, |
| &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_INVALID_FILE); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_common_store_cab_error_wrong_checksum_func(void) |
| { |
| gboolean ret; |
| g_autoptr(FuCabinet) cabinet = fu_cabinet_new(); |
| g_autoptr(GBytes) blob = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| blob = fu_test_build_cab( |
| FALSE, |
| "acme.metainfo.xml", |
| "<component type=\"firmware\">\n" |
| " <id>com.acme.example.firmware</id>\n" |
| " <provides>\n" |
| " <firmware type=\"flashed\">b585990a-003e-5270-89d5-3705a17f9a43</firmware>\n" |
| " </provides>\n" |
| " <releases>\n" |
| " <release version=\"1.2.3\">\n" |
| " <checksum filename=\"firmware.bin\" target=\"content\" " |
| "type=\"sha1\">deadbeef</checksum>\n" |
| " </release>\n" |
| " </releases>\n" |
| "</component>", |
| "firmware.bin", |
| "world", |
| NULL); |
| ret = fu_firmware_parse_bytes(FU_FIRMWARE(cabinet), |
| blob, |
| 0x0, |
| FU_FIRMWARE_PARSE_FLAG_NONE, |
| &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_INVALID_FILE); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_engine_modify_bios_settings_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| const gchar *current; |
| FwupdBiosSetting *attr1; |
| FwupdBiosSetting *attr2; |
| FwupdBiosSetting *attr3; |
| FwupdBiosSetting *attr4; |
| g_autofree gchar *test_dir = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(FuBiosSettings) attrs = NULL; |
| g_autoptr(GPtrArray) items = NULL; |
| g_autoptr(GHashTable) bios_settings = |
| g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free); |
| |
| #ifdef _WIN32 |
| g_test_skip("BIOS settings not supported on Windows"); |
| return; |
| #endif |
| |
| /* load contrived attributes */ |
| test_dir = g_test_build_filename(G_TEST_DIST, "tests", "bios-attrs", NULL); |
| (void)g_setenv("FWUPD_SYSFSFWATTRIBDIR", test_dir, TRUE); |
| |
| ret = fu_context_reload_bios_settings(fu_engine_get_context(engine), &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| attrs = fu_context_get_bios_settings(fu_engine_get_context(engine)); |
| items = fu_bios_settings_get_all(attrs); |
| g_assert_cmpint(items->len, ==, 4); |
| |
| /* enumeration */ |
| attr1 = fu_context_get_bios_setting(fu_engine_get_context(engine), |
| "com.fwupd-internal.Absolute"); |
| g_assert_nonnull(attr1); |
| |
| current = fwupd_bios_setting_get_current_value(attr1); |
| g_assert_nonnull(current); |
| |
| g_hash_table_insert(bios_settings, g_strdup("Absolute"), g_strdup("Disabled")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOTHING_TO_DO); |
| g_assert_false(ret); |
| g_clear_error(&error); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("Absolute"), g_strdup("Enabled")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("Absolute"), g_strdup("off")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("Absolute"), g_strdup("FOO")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_false(ret); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_clear_error(&error); |
| |
| /* use BiosSettingId instead */ |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("com.fwupd-internal.Absolute"), g_strdup("on")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, |
| g_strdup("com.fwupd-internal.Absolute"), |
| g_strdup("off")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* string */ |
| attr2 = |
| fu_context_get_bios_setting(fu_engine_get_context(engine), "com.fwupd-internal.Asset"); |
| g_assert_nonnull(attr2); |
| |
| current = fwupd_bios_setting_get_current_value(attr2); |
| g_assert_nonnull(current); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("Asset"), g_strdup("0")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("Asset"), g_strdup("1")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert( |
| bios_settings, |
| g_strdup("Absolute"), |
| g_strdup("1234567891123456789112345678911234567891123456789112345678911111")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_false(ret); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_clear_error(&error); |
| |
| /* integer */ |
| attr3 = fu_context_get_bios_setting(fu_engine_get_context(engine), |
| "com.fwupd-internal.CustomChargeStop"); |
| g_assert_nonnull(attr3); |
| |
| current = fwupd_bios_setting_get_current_value(attr3); |
| g_assert_nonnull(current); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("CustomChargeStop"), g_strdup("75")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_true(ret); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("CustomChargeStop"), g_strdup("110")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_false(ret); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_clear_error(&error); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("CustomChargeStop"), g_strdup("1")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_false(ret); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_clear_error(&error); |
| |
| /* force it to read only */ |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("CustomChargeStop"), g_strdup("70")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, TRUE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* read only */ |
| attr4 = fu_context_get_bios_setting(fu_engine_get_context(engine), |
| "com.fwupd-internal.pending_reboot"); |
| g_assert_nonnull(attr4); |
| |
| current = fwupd_bios_setting_get_current_value(attr4); |
| g_assert_nonnull(current); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("pending_reboot"), g_strdup("foo")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_false(ret); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_clear_error(&error); |
| |
| g_hash_table_remove_all(bios_settings); |
| g_hash_table_insert(bios_settings, g_strdup("CustomChargeStop"), g_strdup("80")); |
| ret = fu_engine_modify_bios_settings(engine, bios_settings, FALSE, &error); |
| g_assert_false(ret); |
| g_assert_error(error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED); |
| g_clear_error(&error); |
| } |
| |
| GFileInputStream * |
| _g_local_file_input_stream_new(int fd); |
| |
| static void |
| fu_unix_seekable_input_stream_func(void) |
| { |
| #ifdef HAVE_GIO_UNIX |
| gssize ret; |
| gint fd; |
| guint8 buf[6] = {0}; |
| g_autofree gchar *fn = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GInputStream) stream = NULL; |
| |
| fn = g_test_build_filename(G_TEST_DIST, "tests", "metadata.xml", NULL); |
| g_assert_nonnull(fn); |
| fd = g_open(fn, O_RDONLY, 0); |
| g_assert_cmpint(fd, >=, 0); |
| |
| stream = fu_unix_seekable_input_stream_new(fd, TRUE); |
| g_assert_nonnull(stream); |
| |
| /* first chuck */ |
| ret = g_input_stream_read(stream, buf, sizeof(buf) - 1, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_cmpint(ret, ==, 5); |
| g_assert_cmpstr((const gchar *)buf, ==, "<?xml"); |
| |
| /* second chuck */ |
| ret = g_input_stream_read(stream, buf, sizeof(buf) - 1, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_cmpint(ret, ==, 5); |
| g_assert_cmpstr((const gchar *)buf, ==, " vers"); |
| |
| /* first chuck, again */ |
| ret = g_seekable_seek(G_SEEKABLE(stream), 0, G_SEEK_SET, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_cmpint(ret, ==, 1); |
| ret = g_input_stream_read(stream, buf, sizeof(buf) - 1, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_cmpint(ret, ==, 5); |
| g_assert_cmpstr((const gchar *)buf, ==, "<?xml"); |
| #else |
| g_test_skip("No gio-unix-2.0 support, skipping"); |
| #endif |
| } |
| |
| static void |
| fu_remote_download_func(void) |
| { |
| gboolean ret; |
| g_autofree gchar *fn = NULL; |
| g_autofree gchar *directory = NULL; |
| g_autofree gchar *expected_metadata = NULL; |
| g_autofree gchar *expected_signature = NULL; |
| g_autoptr(FwupdRemote) remote = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| remote = fwupd_remote_new(); |
| directory = g_build_filename(FWUPD_LOCALSTATEDIR, "lib", "fwupd", "remotes2.d", NULL); |
| expected_metadata = g_build_filename(FWUPD_LOCALSTATEDIR, |
| "lib", |
| "fwupd", |
| "remotes2.d", |
| "lvfs-testing", |
| "firmware.xml.gz", |
| NULL); |
| expected_signature = g_strdup_printf("%s.jcat", expected_metadata); |
| fwupd_remote_set_remotes_dir(remote, directory); |
| fn = g_test_build_filename(G_TEST_DIST, "tests", "remotes2.d", "lvfs-testing.conf", NULL); |
| ret = fu_remote_load_from_filename(remote, fn, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fwupd_remote_setup(remote, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(fwupd_remote_get_kind(remote), ==, FWUPD_REMOTE_KIND_DOWNLOAD); |
| g_assert_cmpint(fwupd_remote_get_priority(remote), ==, 0); |
| g_assert_false(fwupd_remote_has_flag(remote, FWUPD_REMOTE_FLAG_ENABLED)); |
| g_assert_nonnull(fwupd_remote_get_metadata_uri(remote)); |
| g_assert_nonnull(fwupd_remote_get_metadata_uri_sig(remote)); |
| g_assert_cmpstr(fwupd_remote_get_title(remote), |
| ==, |
| "Linux Vendor Firmware Service (testing)"); |
| g_assert_cmpstr(fwupd_remote_get_report_uri(remote), |
| ==, |
| "https://fwupd.org/lvfs/firmware/report"); |
| g_assert_cmpstr(fwupd_remote_get_filename_cache(remote), ==, expected_metadata); |
| g_assert_cmpstr(fwupd_remote_get_filename_cache_sig(remote), ==, expected_signature); |
| } |
| |
| /* verify we used the FirmwareBaseURI just for firmware */ |
| static void |
| fu_remote_baseuri_func(void) |
| { |
| gboolean ret; |
| g_autofree gchar *firmware_uri = NULL; |
| g_autofree gchar *fn = NULL; |
| g_autoptr(FwupdRemote) remote = NULL; |
| g_autofree gchar *directory = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| remote = fwupd_remote_new(); |
| directory = g_build_filename(FWUPD_LOCALSTATEDIR, "lib", "fwupd", "remotes2.d", NULL); |
| fwupd_remote_set_remotes_dir(remote, directory); |
| fn = g_test_build_filename(G_TEST_DIST, "tests", "firmware-base-uri.conf", NULL); |
| ret = fu_remote_load_from_filename(remote, fn, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(fwupd_remote_get_kind(remote), ==, FWUPD_REMOTE_KIND_DOWNLOAD); |
| g_assert_cmpint(fwupd_remote_get_priority(remote), ==, 0); |
| g_assert_true(fwupd_remote_has_flag(remote, FWUPD_REMOTE_FLAG_ENABLED)); |
| g_assert_cmpstr(fwupd_remote_get_firmware_base_uri(remote), ==, "https://my.fancy.cdn/"); |
| g_assert_cmpstr(fwupd_remote_get_agreement(remote), ==, NULL); |
| g_assert_cmpstr(fwupd_remote_get_checksum(remote), ==, NULL); |
| g_assert_cmpstr(fwupd_remote_get_metadata_uri(remote), |
| ==, |
| "https://s3.amazonaws.com/lvfsbucket/downloads/firmware.xml.gz"); |
| g_assert_cmpstr(fwupd_remote_get_metadata_uri_sig(remote), |
| ==, |
| "https://s3.amazonaws.com/lvfsbucket/downloads/firmware.xml.gz.jcat"); |
| firmware_uri = |
| fwupd_remote_build_firmware_uri(remote, "http://bbc.co.uk/firmware.cab", &error); |
| g_assert_no_error(error); |
| g_assert_cmpstr(firmware_uri, ==, "https://my.fancy.cdn/firmware.cab"); |
| } |
| |
| static void |
| fu_remote_auth_func(void) |
| { |
| gboolean ret; |
| gchar **order; |
| g_autofree gchar *fn = NULL; |
| g_autofree gchar *remotes_dir = NULL; |
| g_autofree gchar *json = NULL; |
| g_autoptr(FwupdRemote) remote = fwupd_remote_new(); |
| g_autoptr(FwupdRemote) remote2 = fwupd_remote_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GVariant) data = NULL; |
| |
| remotes_dir = g_test_build_filename(G_TEST_BUILT, "tests", NULL); |
| fwupd_remote_set_remotes_dir(remote, remotes_dir); |
| |
| fn = g_test_build_filename(G_TEST_DIST, "tests", "auth.conf", NULL); |
| ret = fu_remote_load_from_filename(remote, fn, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpstr(fwupd_remote_get_username(remote), ==, "user"); |
| g_assert_cmpstr(fwupd_remote_get_password(remote), ==, "pass"); |
| g_assert_cmpstr(fwupd_remote_get_report_uri(remote), |
| ==, |
| "https://fwupd.org/lvfs/firmware/report"); |
| g_assert_false(fwupd_remote_has_flag(remote, FWUPD_REMOTE_FLAG_APPROVAL_REQUIRED)); |
| g_assert_false(fwupd_remote_has_flag(remote, FWUPD_REMOTE_FLAG_AUTOMATIC_REPORTS)); |
| g_assert_true(fwupd_remote_has_flag(remote, FWUPD_REMOTE_FLAG_AUTOMATIC_SECURITY_REPORTS)); |
| |
| g_assert_true( |
| g_str_has_suffix(fwupd_remote_get_filename_source(remote), "tests/auth.conf")); |
| g_assert_true(g_str_has_suffix(fwupd_remote_get_remotes_dir(remote), "/src/tests")); |
| g_assert_cmpint(fwupd_remote_get_age(remote), >, 1000000); |
| |
| ret = fwupd_remote_setup(remote, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| order = fwupd_remote_get_order_before(remote); |
| g_assert_nonnull(order); |
| g_assert_cmpint(g_strv_length(order), ==, 1); |
| g_assert_cmpstr(order[0], ==, "before"); |
| order = fwupd_remote_get_order_after(remote); |
| g_assert_nonnull(order); |
| g_assert_cmpint(g_strv_length(order), ==, 1); |
| g_assert_cmpstr(order[0], ==, "after"); |
| |
| /* to/from GVariant */ |
| fwupd_remote_set_priority(remote, 999); |
| data = fwupd_codec_to_variant(FWUPD_CODEC(remote), FWUPD_CODEC_FLAG_NONE); |
| ret = fwupd_codec_from_variant(FWUPD_CODEC(remote2), data, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpstr(fwupd_remote_get_username(remote2), ==, "user"); |
| g_assert_cmpint(fwupd_remote_get_priority(remote2), ==, 999); |
| |
| /* jcat-tool is not a hard dep, and the tests create an empty file if unfound */ |
| ret = fwupd_remote_load_signature(remote, |
| fwupd_remote_get_filename_cache_sig(remote), |
| &error); |
| if (!ret) { |
| if (g_error_matches(error, FWUPD_ERROR, FWUPD_ERROR_READ)) { |
| g_test_skip("no jcat-tool, so skipping test"); |
| return; |
| } |
| } |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* to JSON */ |
| fwupd_remote_set_filename_source(remote2, NULL); |
| fwupd_remote_set_checksum_sig( |
| remote2, |
| "dd1b4fd2a59bb0e4d9ea760c658ac3cf9336c7b6729357bab443485b5cf071b2"); |
| fwupd_remote_set_filename_cache(remote2, "./libfwupd/tests/auth/firmware.xml.gz"); |
| json = fwupd_codec_to_json_string(FWUPD_CODEC(remote2), FWUPD_CODEC_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(json); |
| ret = |
| g_strcmp0( |
| json, |
| "{\n" |
| " \"Id\" : \"auth\",\n" |
| " \"Kind\" : \"download\",\n" |
| " \"ReportUri\" : \"https://fwupd.org/lvfs/firmware/report\",\n" |
| " \"MetadataUri\" : \"https://cdn.fwupd.org/downloads/firmware.xml.gz\",\n" |
| " \"MetadataUriSig\" : \"https://cdn.fwupd.org/downloads/firmware.xml.gz.jcat\",\n" |
| " \"FirmwareBaseUri\" : \"https://my.fancy.cdn/\",\n" |
| " \"Username\" : \"user\",\n" |
| " \"Password\" : \"pass\",\n" |
| " \"ChecksumSig\" : " |
| "\"dd1b4fd2a59bb0e4d9ea760c658ac3cf9336c7b6729357bab443485b5cf071b2\",\n" |
| " \"FilenameCache\" : \"./libfwupd/tests/auth/firmware.xml.gz\",\n" |
| " \"FilenameCacheSig\" : \"./libfwupd/tests/auth/firmware.xml.gz.jcat\",\n" |
| " \"Flags\" : 9,\n" |
| " \"Enabled\" : true,\n" |
| " \"ApprovalRequired\" : false,\n" |
| " \"AutomaticReports\" : false,\n" |
| " \"AutomaticSecurityReports\" : true,\n" |
| " \"Priority\" : 999,\n" |
| " \"Mtime\" : 0,\n" |
| " \"RefreshInterval\" : 86400\n" |
| "}") == 0; |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_remote_duplicate_func(void) |
| { |
| gboolean ret; |
| g_autofree gchar *fn2 = NULL; |
| g_autofree gchar *fn = NULL; |
| g_autoptr(FwupdRemote) remote = fwupd_remote_new(); |
| g_autoptr(GError) error = NULL; |
| |
| fn = g_test_build_filename(G_TEST_DIST, "tests", "stable.conf", NULL); |
| ret = fu_remote_load_from_filename(remote, fn, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| fn2 = g_test_build_filename(G_TEST_DIST, "tests", "disabled.conf", NULL); |
| ret = fu_remote_load_from_filename(remote, fn2, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fwupd_remote_setup(remote, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fwupd_remote_setup(remote, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_false(fwupd_remote_has_flag(remote, FWUPD_REMOTE_FLAG_ENABLED)); |
| g_assert_cmpstr(fwupd_remote_get_username(remote), ==, NULL); |
| g_assert_cmpstr(fwupd_remote_get_password(remote), ==, ""); |
| g_assert_cmpstr(fwupd_remote_get_filename_cache(remote), |
| ==, |
| "/tmp/fwupd-self-test/stable.xml"); |
| } |
| |
| /* verify we used the metadata path for firmware */ |
| static void |
| fu_remote_nopath_func(void) |
| { |
| gboolean ret; |
| g_autofree gchar *firmware_uri = NULL; |
| g_autofree gchar *fn = NULL; |
| g_autoptr(FwupdRemote) remote = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autofree gchar *directory = NULL; |
| |
| remote = fwupd_remote_new(); |
| directory = g_build_filename(FWUPD_LOCALSTATEDIR, "lib", "fwupd", "remotes2.d", NULL); |
| fwupd_remote_set_remotes_dir(remote, directory); |
| fn = g_test_build_filename(G_TEST_DIST, "tests", "firmware-nopath.conf", NULL); |
| ret = fu_remote_load_from_filename(remote, fn, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(fwupd_remote_get_kind(remote), ==, FWUPD_REMOTE_KIND_DOWNLOAD); |
| g_assert_cmpint(fwupd_remote_get_priority(remote), ==, 0); |
| g_assert_true(fwupd_remote_has_flag(remote, FWUPD_REMOTE_FLAG_ENABLED)); |
| g_assert_cmpstr(fwupd_remote_get_checksum(remote), ==, NULL); |
| g_assert_cmpstr(fwupd_remote_get_metadata_uri(remote), |
| ==, |
| "https://s3.amazonaws.com/lvfsbucket/downloads/firmware.xml.gz"); |
| g_assert_cmpstr(fwupd_remote_get_metadata_uri_sig(remote), |
| ==, |
| "https://s3.amazonaws.com/lvfsbucket/downloads/firmware.xml.gz.jcat"); |
| firmware_uri = fwupd_remote_build_firmware_uri(remote, "firmware.cab", &error); |
| g_assert_no_error(error); |
| g_assert_cmpstr(firmware_uri, |
| ==, |
| "https://s3.amazonaws.com/lvfsbucket/downloads/firmware.cab"); |
| } |
| |
| static void |
| fu_remote_local_func(void) |
| { |
| gboolean ret; |
| g_autofree gchar *fn = NULL; |
| g_autofree gchar *json = NULL; |
| g_autoptr(FwupdRemote) remote = NULL; |
| g_autoptr(FwupdRemote) remote2 = fwupd_remote_new(); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GVariant) data = NULL; |
| |
| remote = fwupd_remote_new(); |
| fn = g_test_build_filename(G_TEST_DIST, "tests", "dell-esrt.conf", NULL); |
| ret = fu_remote_load_from_filename(remote, fn, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpint(fwupd_remote_get_kind(remote), ==, FWUPD_REMOTE_KIND_LOCAL); |
| g_assert_true(fwupd_remote_has_flag(remote, FWUPD_REMOTE_FLAG_ENABLED)); |
| g_assert_null(fwupd_remote_get_metadata_uri(remote)); |
| g_assert_null(fwupd_remote_get_metadata_uri_sig(remote)); |
| g_assert_null(fwupd_remote_get_report_uri(remote)); |
| g_assert_cmpstr(fwupd_remote_get_title(remote), |
| ==, |
| "Enable UEFI capsule updates on Dell systems"); |
| g_assert_cmpstr(fwupd_remote_get_filename_cache(remote), |
| ==, |
| "@datadir@/fwupd/remotes.d/dell-esrt/firmware.xml"); |
| g_assert_cmpstr(fwupd_remote_get_filename_cache_sig(remote), ==, NULL); |
| g_assert_cmpstr(fwupd_remote_get_checksum(remote), ==, NULL); |
| |
| /* to/from GVariant */ |
| data = fwupd_codec_to_variant(FWUPD_CODEC(remote), FWUPD_CODEC_FLAG_NONE); |
| ret = fwupd_codec_from_variant(FWUPD_CODEC(remote2), data, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_null(fwupd_remote_get_metadata_uri(remote)); |
| |
| /* to JSON */ |
| fwupd_remote_set_filename_source(remote2, NULL); |
| json = fwupd_codec_to_json_string(FWUPD_CODEC(remote2), FWUPD_CODEC_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(json); |
| ret = g_strcmp0( |
| json, |
| "{\n" |
| " \"Id\" : \"dell-esrt\",\n" |
| " \"Kind\" : \"local\",\n" |
| " \"Title\" : \"Enable UEFI capsule updates on Dell systems\",\n" |
| " \"FilenameCache\" : \"@datadir@/fwupd/remotes.d/dell-esrt/firmware.xml\",\n" |
| " \"Flags\" : 1,\n" |
| " \"Enabled\" : true,\n" |
| " \"ApprovalRequired\" : false,\n" |
| " \"AutomaticReports\" : false,\n" |
| " \"AutomaticSecurityReports\" : false,\n" |
| " \"Priority\" : 0,\n" |
| " \"Mtime\" : 0,\n" |
| " \"RefreshInterval\" : 0\n" |
| "}") == 0; |
| g_assert_true(ret); |
| } |
| |
| static void |
| fu_remote_list_repair_func(void) |
| { |
| FwupdRemote *remote; |
| gboolean ret; |
| g_autoptr(FuRemoteList) remote_list = fu_remote_list_new(); |
| g_autoptr(GError) error = NULL; |
| |
| fu_remote_list_set_lvfs_metadata_format(remote_list, "zst"); |
| ret = fu_remote_list_load(remote_list, FU_REMOTE_LIST_LOAD_FLAG_FIX_METADATA_URI, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* check .gz converted to .zst */ |
| remote = fu_remote_list_get_by_id(remote_list, "legacy-lvfs"); |
| g_assert_nonnull(remote); |
| g_assert_cmpstr(fwupd_remote_get_metadata_uri(remote), |
| ==, |
| "http://localhost/stable.xml.zst"); |
| |
| /* check .xz converted to .zst */ |
| remote = fu_remote_list_get_by_id(remote_list, "legacy-lvfs-xz"); |
| g_assert_nonnull(remote); |
| g_assert_cmpstr(fwupd_remote_get_metadata_uri(remote), |
| ==, |
| "http://localhost/stable.xml.zst"); |
| |
| /* check non-LVFS remote NOT .gz converted to .xz */ |
| remote = fu_remote_list_get_by_id(remote_list, "legacy"); |
| g_assert_nonnull(remote); |
| g_assert_cmpstr(fwupd_remote_get_metadata_uri(remote), |
| ==, |
| "http://localhost/stable.xml.gz"); |
| } |
| |
| static void |
| fu_config_migrate_1_9_func(void) |
| { |
| const gchar *fake_localconf_fn = "/tmp/fwupd-self-test/var/etc/fwupd/fwupd.conf"; |
| const gchar *fake_sysconf_fn = "/tmp/fwupd-self-test/fwupd/fwupd.conf"; |
| gboolean ret; |
| g_autoptr(FuConfig) config = FU_CONFIG(fu_engine_config_new()); |
| g_autoptr(GError) error = NULL; |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| g_unsetenv("CONFIGURATION_DIRECTORY"); |
| (void)g_setenv("FWUPD_SYSCONFDIR", "/tmp/fwupd-self-test", TRUE); |
| |
| ret = fu_path_mkdir_parent(fake_sysconf_fn, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = g_file_set_contents(fake_sysconf_fn, |
| "# use `man 5 fwupd.conf` for documentation\n" |
| "[fwupd]\n" |
| "DisabledPlugins=test;test_ble\n" |
| "OnlyTrusted=true\n" |
| "AllowEmulation=false\n", |
| -1, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| ret = fu_config_load(config, FU_CONFIG_LOAD_FLAG_NONE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* ensure that all keys migrated */ |
| ret = g_file_test(fake_localconf_fn, G_FILE_TEST_EXISTS); |
| g_assert_false(ret); |
| } |
| |
| static void |
| fu_config_set_plugin_defaults(FuConfig *config) |
| { |
| /* these are correct for v2.0.0 */ |
| fu_config_set_default(config, "msr", "MinimumSmeKernelVersion", "5.18.0"); |
| fu_config_set_default(config, "redfish", "CACheck", "false"); |
| fu_config_set_default(config, "redfish", "IpmiDisableCreateUser", "false"); |
| fu_config_set_default(config, "redfish", "ManagerResetTimeout", "1800"); /* seconds */ |
| fu_config_set_default(config, "redfish", "Password", NULL); |
| fu_config_set_default(config, "redfish", "Uri", NULL); |
| fu_config_set_default(config, "redfish", "Username", NULL); |
| fu_config_set_default(config, "redfish", "UserUri", NULL); |
| fu_config_set_default(config, "thunderbolt", "DelayedActivation", "false"); |
| fu_config_set_default(config, "thunderbolt", "MinimumKernelVersion", "4.13.0"); |
| fu_config_set_default(config, "uefi-capsule", "DisableCapsuleUpdateOnDisk", "false"); |
| fu_config_set_default(config, "uefi-capsule", "DisableShimForSecureBoot", "false"); |
| fu_config_set_default(config, "uefi-capsule", "EnableEfiDebugging", "false"); |
| fu_config_set_default(config, "uefi-capsule", "EnableGrubChainLoad", "false"); |
| fu_config_set_default(config, "uefi-capsule", "OverrideESPMountPoint", NULL); |
| fu_config_set_default(config, "uefi-capsule", "RebootCleanup", "true"); |
| fu_config_set_default(config, "uefi-capsule", "RequireESPFreeSpace", "0"); |
| fu_config_set_default(config, "uefi-capsule", "ScreenWidth", "0"); |
| fu_config_set_default(config, "uefi-capsule", "ScreenHeight", "0"); |
| } |
| |
| static void |
| fu_config_migrate_1_7_func(void) |
| { |
| const gchar *sysconfdir = "/tmp/fwupd-self-test/conf-migration-1.7/var/etc"; |
| gboolean ret; |
| const gchar *fn_merge[] = {"daemon.conf", |
| "msr.conf", |
| "redfish.conf", |
| "thunderbolt.conf", |
| "uefi_capsule.conf", |
| NULL}; |
| g_autofree gchar *localconf_data = NULL; |
| g_autofree gchar *fn_mut = NULL; |
| g_autofree gchar *testdatadir = NULL; |
| g_autoptr(FuConfig) config = FU_CONFIG(fu_engine_config_new()); |
| g_autoptr(GError) error = NULL; |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| /* source directory and data */ |
| testdatadir = g_test_build_filename(G_TEST_DIST, "tests", "conf-migration-1.7", NULL); |
| if (!g_file_test(testdatadir, G_FILE_TEST_EXISTS)) { |
| g_test_skip("missing fwupd 1.7.x migration test data"); |
| return; |
| } |
| |
| /* working directory */ |
| (void)g_setenv("FWUPD_SYSCONFDIR", sysconfdir, TRUE); |
| g_unsetenv("CONFIGURATION_DIRECTORY"); |
| |
| fn_mut = g_build_filename(sysconfdir, "fwupd", "fwupd.conf", NULL); |
| g_assert_nonnull(fn_mut); |
| ret = fu_path_mkdir_parent(fn_mut, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* copy all files to working directory */ |
| for (guint i = 0; fn_merge[i] != NULL; i++) { |
| g_autofree gchar *source = |
| g_build_filename(testdatadir, "fwupd", fn_merge[i], NULL); |
| g_autofree gchar *target = g_build_filename(sysconfdir, "fwupd", fn_merge[i], NULL); |
| fu_test_copy_file(source, target); |
| } |
| |
| /* we don't want to run all the plugins just to get the _init() defaults */ |
| fu_config_set_plugin_defaults(config); |
| ret = fu_config_load(config, FU_CONFIG_LOAD_FLAG_MIGRATE_FILES, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* make sure all migrated files were renamed */ |
| for (guint i = 0; fn_merge[i] != NULL; i++) { |
| g_autofree gchar *old = g_build_filename(sysconfdir, "fwupd", fn_merge[i], NULL); |
| g_autofree gchar *new = g_strdup_printf("%s.old", old); |
| ret = g_file_test(old, G_FILE_TEST_EXISTS); |
| g_assert_false(ret); |
| ret = g_file_test(new, G_FILE_TEST_EXISTS); |
| g_assert_true(ret); |
| } |
| |
| /* ensure all default keys migrated */ |
| ret = g_file_get_contents(fn_mut, &localconf_data, NULL, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| g_assert_cmpstr(localconf_data, ==, ""); |
| } |
| |
| static void |
| fu_engine_report_metadata_func(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(GError) error = NULL; |
| g_autoptr(GHashTable) metadata = NULL; |
| g_autoptr(GList) keys = NULL; |
| const gchar *keys_exist[] = { |
| "BatteryLevel", |
| "BatteryThreshold", |
| "CompileVersion(org.freedesktop.fwupd)", |
| "CpuArchitecture", |
| "DistroId", |
| "FwupdSupported", |
| "RuntimeVersion(org.freedesktop.fwupd)", |
| "SELinux", |
| }; |
| |
| /* check report metadata */ |
| metadata = fu_engine_get_report_metadata(engine, &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(metadata); |
| |
| keys = g_list_sort(g_hash_table_get_keys(metadata), (GCompareFunc)g_strcmp0); |
| for (GList *l = keys; l != NULL; l = l->next) { |
| const gchar *key = l->data; |
| const gchar *value = g_hash_table_lookup(metadata, key); |
| g_debug("%s=%s", key, value); |
| } |
| for (guint i = 0; i < G_N_ELEMENTS(keys_exist); i++) { |
| const gchar *value = g_hash_table_lookup(metadata, keys_exist[i]); |
| if (value == NULL) |
| g_warning("no %s in metadata", keys_exist[i]); |
| } |
| } |
| |
| static void |
| fu_engine_machine_hash_func(void) |
| { |
| gsize sz = 0; |
| g_autofree gchar *buf = NULL; |
| g_autofree gchar *mhash1 = NULL; |
| g_autofree gchar *mhash2 = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| if (!g_file_test("/etc/machine-id", G_FILE_TEST_EXISTS)) { |
| g_test_skip("Missing /etc/machine-id"); |
| return; |
| } |
| if (!g_file_get_contents("/etc/machine-id", &buf, &sz, &error)) { |
| g_test_skip("/etc/machine-id is unreadable"); |
| return; |
| } |
| |
| if (sz == 0) { |
| g_test_skip("Empty /etc/machine-id"); |
| return; |
| } |
| |
| mhash1 = fu_engine_build_machine_id("salt1", &error); |
| g_assert_no_error(error); |
| g_assert_cmpstr(mhash1, !=, NULL); |
| mhash2 = fu_engine_build_machine_id("salt2", &error); |
| g_assert_no_error(error); |
| g_assert_cmpstr(mhash2, !=, NULL); |
| g_assert_cmpstr(mhash2, !=, mhash1); |
| } |
| |
| static void |
| fu_test_engine_fake_hidraw(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autofree gchar *value2 = NULL; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(FuUdevDevice) udev_device2 = NULL; |
| g_autoptr(FuUdevDevice) udev_device3 = NULL; |
| g_autoptr(GError) error = NULL; |
| |
| /* non-linux */ |
| if (!fu_context_has_backend(self->ctx, "udev")) { |
| g_test_skip("no Udev backend"); |
| return; |
| } |
| |
| /* load engine and check the device was found */ |
| fu_engine_add_plugin_filter(engine, "pixart_rf"); |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_COLDPLUG | FU_ENGINE_LOAD_FLAG_BUILTIN_PLUGINS | |
| FU_ENGINE_LOAD_FLAG_READONLY, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* hidraw -> pixart_rf */ |
| device = fu_engine_get_device(engine, "6acd27f1feb25ba3b604063de4c13b604776b2f5", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_udev_device_get_subsystem(FU_UDEV_DEVICE(device)), ==, "hidraw"); |
| g_assert_cmpstr(fu_udev_device_get_devtype(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpstr(fu_udev_device_get_driver(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpint(fu_device_get_vid(device), ==, 0x093a); |
| g_assert_cmpint(fu_device_get_pid(device), ==, 0x2862); |
| g_assert_cmpstr(fu_device_get_plugin(device), ==, "pixart_rf"); |
| g_assert_cmpstr(fu_device_get_name(device), ==, "PIXART Pixart dual-mode mouse"); |
| g_assert_cmpstr(fu_device_get_physical_id(device), ==, "usb-0000:00:14.0-1/input1"); |
| g_assert_cmpstr(fu_device_get_logical_id(device), ==, NULL); |
| |
| /* check can read random files */ |
| value2 = fu_udev_device_read_sysfs(FU_UDEV_DEVICE(device), |
| "dev", |
| FU_UDEV_DEVICE_ATTR_READ_TIMEOUT_DEFAULT, |
| &error); |
| g_assert_no_error(error); |
| g_assert_cmpstr(value2, ==, "241:1"); |
| |
| /* get child, both specified */ |
| udev_device2 = FU_UDEV_DEVICE( |
| fu_device_get_backend_parent_with_subsystem(device, "usb:usb_interface", &error)); |
| g_assert_no_error(error); |
| g_assert_nonnull(udev_device2); |
| g_assert_cmpstr(fu_udev_device_get_subsystem(udev_device2), ==, "usb"); |
| |
| /* get child, initially unprobed */ |
| udev_device3 = |
| FU_UDEV_DEVICE(fu_device_get_backend_parent_with_subsystem(device, "usb", &error)); |
| g_assert_no_error(error); |
| g_assert_nonnull(udev_device3); |
| g_assert_cmpstr(fu_udev_device_get_subsystem(udev_device3), ==, "usb"); |
| g_assert_cmpstr(fu_udev_device_get_driver(udev_device3), ==, "usb"); |
| } |
| |
| static void |
| fu_test_engine_fake_usb(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| |
| /* non-linux */ |
| if (!fu_context_has_backend(self->ctx, "udev")) { |
| g_test_skip("no Udev backend"); |
| return; |
| } |
| |
| /* load engine and check the device was found */ |
| fu_engine_add_plugin_filter(engine, "hughski_colorhug"); |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_COLDPLUG | FU_ENGINE_LOAD_FLAG_BUILTIN_PLUGINS | |
| FU_ENGINE_LOAD_FLAG_READONLY, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* USB -> colorhug */ |
| device = fu_engine_get_device(engine, "d787669ee4a103fe0b361fe31c10ea037c72f27c", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_udev_device_get_subsystem(FU_UDEV_DEVICE(device)), ==, "usb"); |
| g_assert_cmpstr(fu_udev_device_get_devtype(FU_UDEV_DEVICE(device)), ==, "usb_device"); |
| g_assert_cmpstr(fu_udev_device_get_driver(FU_UDEV_DEVICE(device)), ==, "usb"); |
| g_assert_cmpint(fu_device_get_vid(device), ==, 0x093a); |
| g_assert_cmpint(fu_device_get_pid(device), ==, 0x2862); |
| g_assert_cmpstr(fu_device_get_plugin(device), ==, "hughski_colorhug"); |
| g_assert_cmpstr(fu_device_get_physical_id(device), ==, "1-1"); |
| g_assert_cmpstr(fu_device_get_logical_id(device), ==, NULL); |
| } |
| |
| static void |
| fu_test_engine_fake_v4l(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| |
| /* non-linux */ |
| if (!fu_context_has_backend(self->ctx, "udev")) { |
| g_test_skip("no Udev backend"); |
| return; |
| } |
| |
| /* load engine and check the device was found */ |
| fu_engine_add_plugin_filter(engine, "logitech_tap"); |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_COLDPLUG | FU_ENGINE_LOAD_FLAG_BUILTIN_PLUGINS | |
| FU_ENGINE_LOAD_FLAG_READONLY, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* no -Dplugin_logitech_tap=enabled */ |
| if (fu_engine_get_plugin_by_name(engine, "logitech_tap", &error) == NULL) { |
| g_test_skip(error->message); |
| return; |
| } |
| |
| /* v4l -> logitech_tap */ |
| device = fu_engine_get_device(engine, "d787669ee4a103fe0b361fe31c10ea037c72f27c", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_udev_device_get_subsystem(FU_UDEV_DEVICE(device)), ==, "video4linux"); |
| g_assert_cmpstr(fu_udev_device_get_devtype(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpstr(fu_udev_device_get_driver(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpint(fu_device_get_vid(device), ==, 0x093A); |
| g_assert_cmpint(fu_device_get_pid(device), ==, 0x2862); |
| g_assert_cmpint(fu_v4l_device_get_index(FU_V4L_DEVICE(device)), ==, 0); |
| g_assert_cmpint(fu_v4l_device_get_caps(FU_V4L_DEVICE(device)), ==, FU_V4L_CAP_NONE); |
| g_assert_cmpstr(fu_device_get_name(device), ==, "Integrated Camera: Integrated C"); |
| g_assert_cmpstr(fu_device_get_plugin(device), ==, "logitech_tap"); |
| } |
| |
| static void |
| fu_test_engine_fake_nvme(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| |
| /* non-linux */ |
| if (!fu_context_has_backend(self->ctx, "udev")) { |
| g_test_skip("no Udev backend"); |
| return; |
| } |
| |
| /* load engine and check the device was found */ |
| fu_engine_add_plugin_filter(engine, "nvme"); |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_COLDPLUG | FU_ENGINE_LOAD_FLAG_BUILTIN_PLUGINS | |
| FU_ENGINE_LOAD_FLAG_READONLY | FU_ENGINE_LOAD_FLAG_NO_CACHE, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* no linux/nvme_ioctl.h */ |
| if (fu_engine_get_plugin_by_name(engine, "nvme", &error) == NULL) { |
| g_test_skip(error->message); |
| return; |
| } |
| |
| /* NVMe -> nvme */ |
| device = fu_engine_get_device(engine, "4c263c95f596030b430d65dc934f6722bcee5720", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_udev_device_get_subsystem(FU_UDEV_DEVICE(device)), ==, "nvme"); |
| g_assert_cmpstr(fu_udev_device_get_devtype(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpint(fu_udev_device_get_number(FU_UDEV_DEVICE(device)), ==, 1); |
| g_assert_cmpstr(fu_udev_device_get_driver(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpstr(fu_udev_device_get_device_file(FU_UDEV_DEVICE(device)), ==, "/dev/nvme1"); |
| g_assert_cmpint(fu_device_get_vid(device), ==, 0x1179); |
| g_assert_cmpint(fu_device_get_pid(device), ==, 0x010F); |
| g_assert_true(fu_device_has_vendor_id(device, "PCI:0x1179")); |
| g_assert_cmpstr(fu_device_get_vendor(device), ==, NULL); |
| g_assert_cmpstr(fu_device_get_plugin(device), ==, "nvme"); |
| g_assert_cmpstr(fu_device_get_physical_id(device), ==, "PCI_SLOT_NAME=0000:00:1b.0"); |
| g_assert_cmpstr(fu_device_get_logical_id(device), ==, NULL); |
| } |
| |
| static void |
| fu_test_engine_fake_serio(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| |
| /* non-linux */ |
| if (!fu_context_has_backend(self->ctx, "udev")) { |
| g_test_skip("no Udev backend"); |
| return; |
| } |
| |
| /* load engine and check the device was found */ |
| fu_engine_add_plugin_filter(engine, "synaptics_rmi"); |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_COLDPLUG | FU_ENGINE_LOAD_FLAG_BUILTIN_PLUGINS | |
| FU_ENGINE_LOAD_FLAG_READONLY, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* no gnutls */ |
| if (fu_engine_get_plugin_by_name(engine, "synaptics_rmi", &error) == NULL) { |
| g_test_skip(error->message); |
| return; |
| } |
| |
| /* serio */ |
| device = fu_engine_get_device(engine, "d8419b7614e50c6fb6162b5dca34df5236a62a8d", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_udev_device_get_subsystem(FU_UDEV_DEVICE(device)), ==, "serio"); |
| g_assert_cmpstr(fu_udev_device_get_devtype(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpstr(fu_udev_device_get_driver(FU_UDEV_DEVICE(device)), ==, "psmouse"); |
| g_assert_cmpstr(fu_udev_device_get_device_file(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpint(fu_device_get_vid(device), ==, 0x0); |
| g_assert_cmpint(fu_device_get_pid(device), ==, 0x0); |
| g_assert_cmpstr(fu_device_get_name(device), ==, "TouchStyk"); |
| g_assert_cmpstr(fu_device_get_plugin(device), ==, "synaptics_rmi"); |
| g_assert_cmpstr(fu_device_get_physical_id(device), |
| ==, |
| "DEVPATH=/devices/platform/i8042/serio1"); |
| g_assert_cmpstr(fu_device_get_logical_id(device), ==, NULL); |
| g_assert_true(fu_device_has_instance_id(device, |
| "SERIO\\FWID_LEN0305-PNP0F13", |
| FU_DEVICE_INSTANCE_FLAG_VISIBLE)); |
| } |
| |
| static void |
| fu_test_engine_fake_tpm(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| |
| /* non-linux */ |
| if (!fu_context_has_backend(self->ctx, "udev")) { |
| g_test_skip("no Udev backend"); |
| return; |
| } |
| |
| /* load engine and check the device was found */ |
| fu_engine_add_plugin_filter(engine, "tpm"); |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_COLDPLUG | FU_ENGINE_LOAD_FLAG_BUILTIN_PLUGINS | |
| FU_ENGINE_LOAD_FLAG_READONLY, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* no tss2-esys */ |
| if (fu_engine_get_plugin_by_name(engine, "tpm", &error) == NULL) { |
| g_test_skip(error->message); |
| return; |
| } |
| |
| /* tpm */ |
| device = fu_engine_get_device(engine, "1d8d50a4dbc65618f5c399c2ae827b632b3ccc11", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_udev_device_get_subsystem(FU_UDEV_DEVICE(device)), ==, "tpm"); |
| g_assert_cmpstr(fu_udev_device_get_devtype(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpstr(fu_udev_device_get_driver(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpstr(fu_udev_device_get_device_file(FU_UDEV_DEVICE(device)), ==, "/dev/tpm0"); |
| g_assert_cmpint(fu_device_get_vid(device), ==, 0x0); |
| g_assert_cmpint(fu_device_get_pid(device), ==, 0x0); |
| g_assert_cmpstr(fu_device_get_plugin(device), ==, "tpm"); |
| g_assert_cmpstr(fu_device_get_physical_id(device), ==, "DEVNAME=tpm0"); |
| g_assert_cmpstr(fu_device_get_logical_id(device), ==, NULL); |
| } |
| |
| static void |
| fu_test_engine_fake_block(gconstpointer user_data) |
| { |
| FuTest *self = (FuTest *)user_data; |
| gboolean ret; |
| g_autoptr(FuDevice) device = NULL; |
| g_autoptr(FuEngine) engine = fu_engine_new(self->ctx); |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(GError) error = NULL; |
| |
| /* non-linux */ |
| if (!fu_context_has_backend(self->ctx, "udev")) { |
| g_test_skip("no Udev backend"); |
| return; |
| } |
| |
| /* load engine and check the device was found */ |
| fu_engine_add_plugin_filter(engine, "scsi"); |
| ret = fu_engine_load(engine, |
| FU_ENGINE_LOAD_FLAG_COLDPLUG | FU_ENGINE_LOAD_FLAG_BUILTIN_PLUGINS | |
| FU_ENGINE_LOAD_FLAG_READONLY, |
| progress, |
| &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* no Udev */ |
| if (fu_engine_get_plugin_by_name(engine, "scsi", &error) == NULL) { |
| g_test_skip(error->message); |
| return; |
| } |
| |
| /* block */ |
| device = fu_engine_get_device(engine, "7772d9fe9419e3ea564216e12913a16e233378a6", &error); |
| g_assert_no_error(error); |
| g_assert_nonnull(device); |
| g_assert_cmpstr(fu_udev_device_get_subsystem(FU_UDEV_DEVICE(device)), ==, "block"); |
| g_assert_cmpstr(fu_udev_device_get_devtype(FU_UDEV_DEVICE(device)), ==, "disk"); |
| g_assert_cmpstr(fu_udev_device_get_driver(FU_UDEV_DEVICE(device)), ==, NULL); |
| g_assert_cmpstr(fu_udev_device_get_device_file(FU_UDEV_DEVICE(device)), ==, "/dev/sde"); |
| g_assert_cmpstr(fu_device_get_plugin(device), ==, "scsi"); |
| g_assert_cmpstr(fu_device_get_vendor(device), ==, "IBM-ESXS"); |
| } |
| |
| int |
| main(int argc, char **argv) |
| { |
| gboolean ret; |
| g_autofree gchar *sysfsdir = NULL; |
| g_autofree gchar *testdatadir = NULL; |
| g_autoptr(GError) error = NULL; |
| g_autoptr(FuProgress) progress = fu_progress_new(G_STRLOC); |
| g_autoptr(FuTest) self = g_new0(FuTest, 1); |
| |
| (void)g_setenv("G_TEST_SRCDIR", SRCDIR, FALSE); |
| g_test_init(&argc, &argv, NULL); |
| |
| /* only critical and error are fatal */ |
| g_log_set_fatal_mask(NULL, G_LOG_LEVEL_ERROR | G_LOG_LEVEL_CRITICAL); |
| (void)g_setenv("G_MESSAGES_DEBUG", "all", TRUE); |
| testdatadir = g_test_build_filename(G_TEST_DIST, "tests", NULL); |
| (void)g_setenv("FWUPD_DATADIR", testdatadir, TRUE); |
| (void)g_setenv("FWUPD_LIBDIR_PKG", testdatadir, TRUE); |
| (void)g_setenv("FWUPD_SYSCONFDIR", testdatadir, TRUE); |
| (void)g_setenv("FWUPD_SYSFSFWDIR", testdatadir, TRUE); |
| (void)g_setenv("CONFIGURATION_DIRECTORY", testdatadir, TRUE); |
| (void)g_setenv("FWUPD_LOCALSTATEDIR", "/tmp/fwupd-self-test/var", TRUE); |
| (void)g_setenv("FWUPD_RUNDIR", "/tmp/fwupd-self-test/run", TRUE); |
| (void)g_setenv("FWUPD_SYSFSFWATTRIBDIR", testdatadir, TRUE); |
| (void)g_setenv("FWUPD_PROCFS", testdatadir, TRUE); |
| sysfsdir = g_test_build_filename(G_TEST_DIST, "tests", "sys", NULL); |
| (void)g_setenv("FWUPD_SYSFSDIR", sysfsdir, TRUE); |
| (void)g_setenv("FWUPD_SELF_TEST", "1", TRUE); |
| (void)g_setenv("FWUPD_MACHINE_ID", "test", TRUE); |
| (void)g_setenv("FWUPD_EFIVARS", "dummy", TRUE); |
| |
| /* ensure empty tree */ |
| fu_self_test_mkroot(); |
| |
| /* do not save silo */ |
| self->ctx = fu_context_new(); |
| fu_context_add_flag(self->ctx, FU_CONTEXT_FLAG_NO_IDLE_SOURCES); |
| ret = fu_context_load_quirks(self->ctx, FU_QUIRKS_LOAD_FLAG_NO_CACHE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* build a plausible EFI system */ |
| ret = fu_efivars_set_secure_boot(fu_context_get_efivars(self->ctx), TRUE, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| ret = fu_efivars_set_boot_current(fu_context_get_efivars(self->ctx), 0x0000, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* do not allow mounting disks in the self tests */ |
| fu_context_add_flag(self->ctx, FU_CONTEXT_FLAG_INHIBIT_VOLUME_MOUNT); |
| |
| /* load dummy hwids */ |
| ret = fu_context_load_hwinfo(self->ctx, progress, FU_CONTEXT_HWID_FLAG_LOAD_CONFIG, &error); |
| g_assert_no_error(error); |
| g_assert_true(ret); |
| |
| /* tests go here */ |
| if (g_test_slow()) |
| g_test_add_data_func("/fwupd/console", self, fu_console_func); |
| g_test_add_func("/fwupd/idle", fu_idle_func); |
| g_test_add_func("/fwupd/client-list", fu_client_list_func); |
| g_test_add_func("/fwupd/remote{download}", fu_remote_download_func); |
| g_test_add_func("/fwupd/remote{base-uri}", fu_remote_baseuri_func); |
| g_test_add_func("/fwupd/remote{no-path}", fu_remote_nopath_func); |
| g_test_add_func("/fwupd/remote{local}", fu_remote_local_func); |
| g_test_add_func("/fwupd/remote{duplicate}", fu_remote_duplicate_func); |
| g_test_add_func("/fwupd/remote{auth}", fu_remote_auth_func); |
| g_test_add_func("/fwupd/remote-list{repair}", fu_remote_list_repair_func); |
| g_test_add_func("/fwupd/unix-seekable-input-stream", fu_unix_seekable_input_stream_func); |
| g_test_add_data_func("/fwupd/backend{usb}", self, fu_backend_usb_func); |
| g_test_add_data_func("/fwupd/backend{usb-invalid}", self, fu_backend_usb_invalid_func); |
| g_test_add_data_func("/fwupd/plugin{module}", self, fu_plugin_module_func); |
| g_test_add_data_func("/fwupd/memcpy", self, fu_memcpy_func); |
| g_test_add_func("/fwupd/cabinet", fu_common_cabinet_func); |
| g_test_add_data_func("/fwupd/security-attr", self, fu_security_attr_func); |
| g_test_add_data_func("/fwupd/device-list", self, fu_device_list_func); |
| g_test_add_data_func("/fwupd/device-list{unconnected-no-delay}", |
| self, |
| fu_device_list_unconnected_no_delay_func); |
| g_test_add_data_func("/fwupd/device-list{equivalent-id}", |
| self, |
| fu_device_list_equivalent_id_func); |
| g_test_add_data_func("/fwupd/device-list{delay}", self, fu_device_list_delay_func); |
| g_test_add_data_func("/fwupd/device-list{explicit-order}", |
| self, |
| fu_device_list_explicit_order_func); |
| g_test_add_data_func("/fwupd/device-list{explicit-order-post}", |
| self, |
| fu_device_list_explicit_order_post_func); |
| g_test_add_data_func("/fwupd/device-list{install-parent-first}", |
| self, |
| fu_device_list_install_parent_first_func); |
| g_test_add_data_func("/fwupd/device-list{install-parent-first-child}", |
| self, |
| fu_device_list_install_parent_first_child_func); |
| g_test_add_data_func("/fwupd/device-list{no-auto-remove-children}", |
| self, |
| fu_device_list_no_auto_remove_children_func); |
| g_test_add_data_func("/fwupd/device-list{compatible}", |
| self, |
| fu_device_list_compatible_func); |
| g_test_add_data_func("/fwupd/device-list{remove-chain}", |
| self, |
| fu_device_list_remove_chain_func); |
| g_test_add_data_func("/fwupd/device-list{counterpart}", |
| self, |
| fu_device_list_counterpart_func); |
| g_test_add_data_func("/fwupd/device-list{better-than}", |
| self, |
| fu_device_list_better_than_func); |
| g_test_add_data_func("/fwupd/release{compare}", self, fu_release_compare_func); |
| g_test_add_func("/fwupd/release{uri-scheme}", fu_release_uri_scheme_func); |
| g_test_add_data_func("/fwupd/release{trusted-report}", |
| self, |
| fu_release_trusted_report_func); |
| g_test_add_data_func("/fwupd/release{trusted-report-oem}", |
| self, |
| fu_release_trusted_report_oem_func); |
| g_test_add_data_func("/fwupd/release{no-trusted-report-upgrade}", |
| self, |
| fu_release_no_trusted_report_upgrade_func); |
| g_test_add_data_func("/fwupd/release{no-trusted-report}", |
| self, |
| fu_release_no_trusted_report_func); |
| g_test_add_data_func("/fwupd/engine{get-details-added}", |
| self, |
| fu_engine_get_details_added_func); |
| g_test_add_data_func("/fwupd/engine{get-details-missing}", |
| self, |
| fu_engine_get_details_missing_func); |
| g_test_add_data_func("/fwupd/engine{device-unlock}", self, fu_engine_device_unlock_func); |
| g_test_add_data_func("/fwupd/engine{device-equivalent}", |
| self, |
| fu_engine_device_equivalent_func); |
| g_test_add_data_func("/fwupd/engine{device-md-set-flags}", |
| self, |
| fu_engine_device_md_set_flags_func); |
| g_test_add_data_func("/fwupd/engine{device-md-checksum-set-version}", |
| self, |
| fu_engine_device_md_checksum_set_version_func); |
| g_test_add_data_func("/fwupd/engine{device-md-checksum-set-version-wrong-proto}", |
| self, |
| fu_engine_device_md_checksum_set_version_wrong_proto_func); |
| g_test_add_data_func("/fwupd/engine{multiple-releases}", |
| self, |
| fu_engine_multiple_rels_func); |
| g_test_add_data_func("/fwupd/engine{install-loop-restart}", |
| self, |
| fu_engine_install_loop_restart_func); |
| g_test_add_data_func("/fwupd/engine{install-request}", self, fu_engine_install_request); |
| g_test_add_data_func("/fwupd/engine{history-success}", self, fu_engine_history_func); |
| g_test_add_data_func("/fwupd/engine{history-verfmt}", self, fu_engine_history_verfmt_func); |
| g_test_add_data_func("/fwupd/engine{history-modify}", self, fu_engine_history_modify_func); |
| g_test_add_data_func("/fwupd/engine{history-error}", self, fu_engine_history_error_func); |
| g_test_add_data_func("/fwupd/engine{fake-hidraw}", self, fu_test_engine_fake_hidraw); |
| g_test_add_data_func("/fwupd/engine{fake-usb}", self, fu_test_engine_fake_usb); |
| g_test_add_data_func("/fwupd/engine{fake-serio}", self, fu_test_engine_fake_serio); |
| g_test_add_data_func("/fwupd/engine{fake-nvme}", self, fu_test_engine_fake_nvme); |
| g_test_add_data_func("/fwupd/engine{fake-block}", self, fu_test_engine_fake_block); |
| g_test_add_data_func("/fwupd/engine{fake-tpm}", self, fu_test_engine_fake_tpm); |
| g_test_add_data_func("/fwupd/engine{fake-v4l}", self, fu_test_engine_fake_v4l); |
| if (g_test_slow()) { |
| g_test_add_data_func("/fwupd/device-list{replug-auto}", |
| self, |
| fu_device_list_replug_auto_func); |
| } |
| g_test_add_data_func("/fwupd/device-list{replug-user}", |
| self, |
| fu_device_list_replug_user_func); |
| g_test_add_func("/fwupd/engine{machine-hash}", fu_engine_machine_hash_func); |
| g_test_add_data_func("/fwupd/engine{report-metadata}", |
| self, |
| fu_engine_report_metadata_func); |
| g_test_add_data_func("/fwupd/engine{require-hwid}", self, fu_engine_require_hwid_func); |
| g_test_add_data_func("/fwupd/engine{requires-reboot}", |
| self, |
| fu_engine_install_needs_reboot); |
| g_test_add_data_func("/fwupd/engine{history-inherit}", self, fu_engine_history_inherit); |
| g_test_add_data_func("/fwupd/engine{partial-hash}", self, fu_engine_partial_hash_func); |
| g_test_add_data_func("/fwupd/engine{downgrade}", self, fu_engine_downgrade_func); |
| g_test_add_data_func("/fwupd/engine{md-verfmt}", self, fu_engine_md_verfmt_func); |
| g_test_add_data_func("/fwupd/engine{requirements-success}", |
| self, |
| fu_engine_requirements_func); |
| g_test_add_data_func("/fwupd/engine{requirements-soft}", |
| self, |
| fu_engine_requirements_soft_func); |
| g_test_add_data_func("/fwupd/engine{requirements-missing}", |
| self, |
| fu_engine_requirements_missing_func); |
| g_test_add_data_func("/fwupd/engine{requirements-client-fail}", |
| self, |
| fu_engine_requirements_client_fail_func); |
| g_test_add_data_func("/fwupd/engine{requirements-client-invalid}", |
| self, |
| fu_engine_requirements_client_invalid_func); |
| g_test_add_data_func("/fwupd/engine{requirements-client-pass}", |
| self, |
| fu_engine_requirements_client_pass_func); |
| g_test_add_data_func("/fwupd/engine{requirements-not-hardware}", |
| self, |
| fu_engine_requirements_not_hardware_func); |
| g_test_add_data_func("/fwupd/engine{requirements-phased}", |
| self, |
| fu_engine_requirements_phased_func); |
| g_test_add_data_func("/fwupd/engine{requirements-phased-old-fwpud}", |
| self, |
| fu_engine_requirements_phased_old_fwupd_func); |
| g_test_add_data_func("/fwupd/engine{requirements-version-require}", |
| self, |
| fu_engine_requirements_version_require_func); |
| g_test_add_data_func("/fwupd/engine{requirements-version-lowest}", |
| self, |
| fu_engine_requirements_version_lowest_func); |
| g_test_add_data_func("/fwupd/engine{requirements-parent-device}", |
| self, |
| fu_engine_requirements_parent_device_func); |
| g_test_add_data_func("/fwupd/engine{requirements-child-device}", |
| self, |
| fu_engine_requirements_child_device_func); |
| g_test_add_data_func("/fwupd/engine{requirements_protocol_check_func}", |
| self, |
| fu_engine_requirements_protocol_check_func); |
| g_test_add_data_func("/fwupd/engine{requirements-not-child}", |
| self, |
| fu_engine_requirements_child_func); |
| g_test_add_data_func("/fwupd/engine{requirements-not-child-fail}", |
| self, |
| fu_engine_requirements_child_fail_func); |
| g_test_add_data_func("/fwupd/engine{requirements-unsupported}", |
| self, |
| fu_engine_requirements_unsupported_func); |
| g_test_add_data_func("/fwupd/engine{requirements-device}", |
| self, |
| fu_engine_requirements_device_func); |
| g_test_add_data_func("/fwupd/engine{requirements-device-plain}", |
| self, |
| fu_engine_requirements_device_plain_func); |
| g_test_add_data_func("/fwupd/engine{requirements-version-format}", |
| self, |
| fu_engine_requirements_version_format_func); |
| g_test_add_data_func("/fwupd/engine{requirements-only-upgrade}", |
| self, |
| fu_engine_requirements_only_upgrade_func); |
| g_test_add_data_func("/fwupd/engine{requirements-only-upgrade-reinstall}", |
| self, |
| fu_engine_requirements_only_upgrade_reinstall_func); |
| g_test_add_data_func("/fwupd/engine{device-auto-parent-id}", |
| self, |
| fu_engine_device_parent_id_func); |
| g_test_add_data_func("/fwupd/engine{device-auto-parent-guid}", |
| self, |
| fu_engine_device_parent_guid_func); |
| g_test_add_data_func("/fwupd/engine{install-duration}", |
| self, |
| fu_engine_install_duration_func); |
| g_test_add_data_func("/fwupd/engine{get-results-appstream-id}", |
| self, |
| fu_plugin_engine_get_results_appstream_id_func); |
| g_test_add_data_func("/fwupd/engine{release-dedupe}", self, fu_engine_release_dedupe_func); |
| g_test_add_data_func("/fwupd/engine{generate-md}", self, fu_engine_generate_md_func); |
| g_test_add_data_func("/fwupd/engine{requirements-other-device}", |
| self, |
| fu_engine_requirements_other_device_func); |
| g_test_add_data_func("/fwupd/engine{requirements-sibling-device}", |
| self, |
| fu_engine_requirements_sibling_device_func); |
| g_test_add_data_func("/fwupd/engine{requirements-vercmp-glob}", |
| self, |
| fu_engine_requirements_vercmp_glob_func); |
| g_test_add_data_func("/fwupd/engine{requirements-vercmp-glob-fallback}", |
| self, |
| fu_engine_requirements_vercmp_glob_fallback_func); |
| g_test_add_data_func("/fwupd/engine{plugin-gtypes}", self, fu_engine_plugin_gtypes_func); |
| g_test_add_data_func("/fwupd/plugin/mutable", |
| self, |
| fu_engine_test_plugin_mutable_enumeration); |
| g_test_add_data_func("/fwupd/plugin{composite}", self, fu_plugin_composite_func); |
| g_test_add_data_func("/fwupd/plugin{composite-multistep}", |
| self, |
| fu_plugin_composite_multistep_func); |
| g_test_add_data_func("/fwupd/history", self, fu_history_func); |
| g_test_add_data_func("/fwupd/history{migrate-v1}", self, fu_history_migrate_v1_func); |
| g_test_add_data_func("/fwupd/history{migrate-v2}", self, fu_history_migrate_v2_func); |
| g_test_add_data_func("/fwupd/plugin-list", self, fu_plugin_list_func); |
| g_test_add_data_func("/fwupd/plugin-list{depsolve}", self, fu_plugin_list_depsolve_func); |
| g_test_add_func("/fwupd/common{cab-success}", fu_common_store_cab_func); |
| g_test_add_func("/fwupd/common{cab-success-artifact}", fu_common_store_cab_artifact_func); |
| g_test_add_func("/fwupd/common{cab-success-unsigned}", fu_common_store_cab_unsigned_func); |
| g_test_add_func("/fwupd/common{cab-success-folder}", fu_common_store_cab_folder_func); |
| g_test_add_func("/fwupd/common{cab-success-sha256}", fu_common_store_cab_sha256_func); |
| g_test_add_func("/fwupd/common{cab-error-no-metadata}", |
| fu_common_store_cab_error_no_metadata_func); |
| g_test_add_func("/fwupd/common{cab-error-wrong-size}", |
| fu_common_store_cab_error_wrong_size_func); |
| g_test_add_func("/fwupd/common{cab-error-wrong-checksum}", |
| fu_common_store_cab_error_wrong_checksum_func); |
| g_test_add_func("/fwupd/common{cab-error-missing-file}", |
| fu_common_store_cab_error_missing_file_func); |
| g_test_add_func("/fwupd/common{cab-error-size}", fu_common_store_cab_error_size_func); |
| g_test_add_data_func("/fwupd/write-bios-attrs", self, fu_engine_modify_bios_settings_func); |
| |
| /* these need to be last as they overwrite stuff in the mkroot */ |
| g_test_add_func("/fwupd/config_migrate_1_7", fu_config_migrate_1_7_func); |
| g_test_add_func("/fwupd/config_migrate_1_9", fu_config_migrate_1_9_func); |
| return g_test_run(); |
| } |