mirror of
https://github.com/ostreedev/ostree.git
synced 2025-01-03 05:18:24 +03:00
a6d07b6cc3
This fixes a truly horrific performance bug when composefs is enabled, but fsverity is not supported by the filesystem. We'd fall back to doing *userspace* checksumming of all files at deployment time which was absolutely not expected or required. There's really an immense amount of technical debt here, such as the confusion between `ex-integity.composefs` vs the prepare-root config, how we handle "torn" states where some objects don't have verity enabled but some do, etc. The ostree composefs state has two modes: - signed: We need to enforce fsverity - unsigned: Best effort resilience So we fix this by making the deploy path to make verity "opportunistic" - if the ioctl gives us the data, then we add it to the composefs. However, this code path is also invoked when we're computing the expected composefs digest to inject as commit metadata, and *that* API must work regardless of whether the target repo has fsverity enabled as it may operate on a build server. One lucky thing in all of this: When I went to add the "checkout composefs" API I added a stub `GVariant` for options extensibility, which we now use. Signed-off-by: Colin Walters <walters@verbum.org>
110 lines
3.3 KiB
C
110 lines
3.3 KiB
C
/*
|
|
* Copyright (C) 2015 Red Hat, Inc.
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.0+
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "libglnx.h"
|
|
#include "ot-gio-utils.h"
|
|
#include "ot-unix-utils.h"
|
|
#include <glib.h>
|
|
|
|
static void
|
|
test_ot_util_path_split_validate (void)
|
|
{
|
|
const char *paths[] = { "foo/bar", "test", "foo/bar:", "a/b/c/d/e/f/g/h/i/l/m/n/o/p", NULL };
|
|
int n_components[] = { 2, 1, 2, 14, 0 };
|
|
int i;
|
|
for (i = 0; paths[i]; i++)
|
|
{
|
|
GError *error = NULL;
|
|
g_autoptr (GPtrArray) components = NULL;
|
|
if (!ot_util_path_split_validate (paths[i], &components, &error))
|
|
{
|
|
int j;
|
|
g_assert_cmpint (components->len, ==, n_components[i]);
|
|
for (j = 0; j < components->len; j++)
|
|
{
|
|
g_assert (strcmp (components->pdata[i], ".."));
|
|
g_assert_null (strchr (components->pdata[i], '/'));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_ot_util_filename_validate (void)
|
|
{
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
/* Check for valid inputs. */
|
|
g_assert (ot_util_filename_validate ("valid", &error));
|
|
g_assert_no_error (error);
|
|
g_assert (ot_util_filename_validate ("valid_file_name", &error));
|
|
g_assert_no_error (error);
|
|
g_assert (ot_util_filename_validate ("file.name", &error));
|
|
g_assert_no_error (error);
|
|
g_assert (ot_util_filename_validate ("foo..", &error));
|
|
g_assert_no_error (error);
|
|
g_assert (ot_util_filename_validate ("..bar", &error));
|
|
g_assert_no_error (error);
|
|
g_assert (ot_util_filename_validate ("baz:", &error));
|
|
g_assert_no_error (error);
|
|
|
|
/* Check for invalid inputs. */
|
|
g_assert_false (ot_util_filename_validate ("not/valid/file/name", &error));
|
|
g_clear_error (&error);
|
|
g_assert_false (ot_util_filename_validate (".", &error));
|
|
g_clear_error (&error);
|
|
g_assert_false (ot_util_filename_validate ("..", &error));
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
static void
|
|
test_ot_human_duration (void)
|
|
{
|
|
struct tcase
|
|
{
|
|
guint64 v;
|
|
const char *expected;
|
|
};
|
|
const struct tcase test_cases[] = {
|
|
{ 0, "0ns" }, { 590, "590ns" }, { 1590, "1ms" },
|
|
{ 9001, "9ms" }, { 1597249, "1.6s" }, { 10597249, "10.6s" },
|
|
};
|
|
|
|
for (guint i = 0; i < G_N_ELEMENTS (test_cases); i++)
|
|
{
|
|
const struct tcase *tcase = &test_cases[i];
|
|
g_autofree char *buf = ot_format_human_duration (tcase->v);
|
|
g_assert_cmpstr (buf, ==, tcase->expected);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
g_test_init (&argc, &argv, NULL);
|
|
g_test_add_func ("/ot_util_path_split_validate", test_ot_util_path_split_validate);
|
|
g_test_add_func ("/ot_util_filename_validate", test_ot_util_filename_validate);
|
|
g_test_add_func ("/ot_human_duration", test_ot_human_duration);
|
|
return g_test_run ();
|
|
}
|