core: Add generic "commit filter" API, use it to implement statoverride

The builder wants the ability to mark a given file as e.g. setuid.  To
implement this, the repo now has a callback-based API when importing a
directory to modify or remove items.

The commit tool accepts a "statoverride" file as input which looks like:

+mode /path/to/file
This commit is contained in:
Colin Walters 2012-01-22 13:27:38 -05:00
parent a322ecacba
commit e8dbaa3c07
4 changed files with 303 additions and 101 deletions

View File

@ -1465,21 +1465,58 @@ create_modified_file_info (GFileInfo *info,
ret = g_file_info_dup (info); ret = g_file_info_dup (info);
if (modifier->uid >= 0)
g_file_info_set_attribute_uint32 (ret, "unix::uid", modifier->uid);
if (modifier->gid >= 0)
g_file_info_set_attribute_uint32 (ret, "unix::gid", modifier->gid);
return ret; return ret;
} }
gboolean static OstreeRepoCommitFilterResult
ostree_repo_stage_directory_to_mtree (OstreeRepo *self, apply_commit_filter (OstreeRepo *self,
GFile *dir, OstreeRepoCommitModifier *modifier,
OstreeMutableTree *mtree, GPtrArray *path,
OstreeRepoCommitModifier *modifier, GFileInfo *file_info,
GCancellable *cancellable, GFileInfo **out_modified_info)
GError **error) {
GString *path_buf;
guint i;
OstreeRepoCommitFilterResult result;
GFileInfo *modified_info;
if (modifier == NULL || modifier->filter == NULL)
{
*out_modified_info = g_object_ref (file_info);
return OSTREE_REPO_COMMIT_FILTER_ALLOW;
}
path_buf = g_string_new ("");
if (path->len == 0)
g_string_append_c (path_buf, '/');
else
{
for (i = 0; i < path->len; i++)
{
const char *elt = path->pdata[i];
g_string_append_c (path_buf, '/');
g_string_append (path_buf, elt);
}
}
modified_info = g_file_info_dup (file_info);
result = modifier->filter (self, path_buf->str, modified_info, modifier->user_data);
*out_modified_info = modified_info;
g_string_free (path_buf, TRUE);
return result;
}
static gboolean
stage_directory_to_mtree_internal (OstreeRepo *self,
GFile *dir,
OstreeMutableTree *mtree,
OstreeRepoCommitModifier *modifier,
GPtrArray *path,
GCancellable *cancellable,
GError **error)
{ {
gboolean ret = FALSE; gboolean ret = FALSE;
OstreeRepoFile *repo_dir = NULL; OstreeRepoFile *repo_dir = NULL;
@ -1493,6 +1530,7 @@ ostree_repo_stage_directory_to_mtree (OstreeRepo *self,
GVariant *xattrs = NULL; GVariant *xattrs = NULL;
GInputStream *file_input = NULL; GInputStream *file_input = NULL;
gboolean repo_dir_was_empty = FALSE; gboolean repo_dir_was_empty = FALSE;
OstreeRepoCommitFilterResult filter_result;
/* We can only reuse checksums directly if there's no modifier */ /* We can only reuse checksums directly if there's no modifier */
if (OSTREE_IS_REPO_FILE (dir) && modifier == NULL) if (OSTREE_IS_REPO_FILE (dir) && modifier == NULL)
@ -1507,6 +1545,8 @@ ostree_repo_stage_directory_to_mtree (OstreeRepo *self,
repo_dir_was_empty = repo_dir_was_empty =
g_hash_table_size (ostree_mutable_tree_get_files (mtree)) == 0 g_hash_table_size (ostree_mutable_tree_get_files (mtree)) == 0
&& g_hash_table_size (ostree_mutable_tree_get_subdirs (mtree)) == 0; && g_hash_table_size (ostree_mutable_tree_get_subdirs (mtree)) == 0;
filter_result = OSTREE_REPO_COMMIT_FILTER_ALLOW;
} }
else else
{ {
@ -1516,96 +1556,108 @@ ostree_repo_stage_directory_to_mtree (OstreeRepo *self,
if (!child_info) if (!child_info)
goto out; goto out;
modified_info = create_modified_file_info (child_info, modifier); filter_result = apply_commit_filter (self, modifier, path, child_info, &modified_info);
if (!(modifier && modifier->skip_xattrs)) if (filter_result == OSTREE_REPO_COMMIT_FILTER_ALLOW)
{ {
xattrs = ostree_get_xattrs_for_file (dir, error);
if (!xattrs)
goto out;
}
if (!stage_directory_meta (self, modified_info, xattrs, &child_file_checksum,
cancellable, error))
goto out;
ostree_mutable_tree_set_metadata_checksum (mtree, g_checksum_get_string (child_file_checksum));
g_clear_object (&child_info);
g_clear_object (&modified_info);
}
dir_enum = g_file_enumerate_children ((GFile*)dir, OSTREE_GIO_FAST_QUERYINFO,
G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS,
cancellable,
error);
if (!dir_enum)
goto out;
while ((child_info = g_file_enumerator_next_file (dir_enum, cancellable, &temp_error)) != NULL)
{
const char *name = g_file_info_get_name (child_info);
g_clear_object (&modified_info);
modified_info = create_modified_file_info (child_info, modifier);
g_clear_object (&child);
child = g_file_get_child (dir, name);
if (g_file_info_get_file_type (child_info) == G_FILE_TYPE_DIRECTORY)
{
g_clear_object (&child_mtree);
if (!ostree_mutable_tree_ensure_dir (mtree, name, &child_mtree, error))
goto out;
if (!ostree_repo_stage_directory_to_mtree (self, child, child_mtree,
modifier, cancellable, error))
goto out;
}
else if (repo_dir)
{
if (!ostree_mutable_tree_replace_file (mtree, name,
ostree_repo_file_get_checksum ((OstreeRepoFile*) child),
error))
goto out;
}
else
{
ot_clear_checksum (&child_file_checksum);
ot_clear_gvariant (&xattrs);
g_clear_object (&file_input);
if (g_file_info_get_file_type (modified_info) == G_FILE_TYPE_REGULAR)
{
file_input = (GInputStream*)g_file_read (child, cancellable, error);
if (!file_input)
goto out;
}
if (!(modifier && modifier->skip_xattrs)) if (!(modifier && modifier->skip_xattrs))
{ {
xattrs = ostree_get_xattrs_for_file (child, error); xattrs = ostree_get_xattrs_for_file (dir, error);
if (!xattrs) if (!xattrs)
goto out; goto out;
} }
if (!stage_object_impl (self, OSTREE_OBJECT_TYPE_RAW_FILE, if (!stage_directory_meta (self, modified_info, xattrs, &child_file_checksum,
modified_info, xattrs, file_input, NULL, cancellable, error))
&child_file_checksum, cancellable, error))
goto out;
if (!ostree_mutable_tree_replace_file (mtree, name,
g_checksum_get_string (child_file_checksum),
error))
goto out; goto out;
ostree_mutable_tree_set_metadata_checksum (mtree, g_checksum_get_string (child_file_checksum));
g_clear_object (&child_info);
g_clear_object (&modified_info);
} }
g_clear_object (&child_info);
} }
if (temp_error != NULL)
if (filter_result == OSTREE_REPO_COMMIT_FILTER_ALLOW)
{ {
g_propagate_error (error, temp_error); dir_enum = g_file_enumerate_children ((GFile*)dir, OSTREE_GIO_FAST_QUERYINFO,
goto out; G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS,
cancellable,
error);
if (!dir_enum)
goto out;
while ((child_info = g_file_enumerator_next_file (dir_enum, cancellable, &temp_error)) != NULL)
{
const char *name = g_file_info_get_name (child_info);
g_clear_object (&modified_info);
g_ptr_array_add (path, (char*)name);
filter_result = apply_commit_filter (self, modifier, path, child_info, &modified_info);
if (filter_result == OSTREE_REPO_COMMIT_FILTER_ALLOW)
{
g_clear_object (&child);
child = g_file_get_child (dir, name);
if (g_file_info_get_file_type (child_info) == G_FILE_TYPE_DIRECTORY)
{
g_clear_object (&child_mtree);
if (!ostree_mutable_tree_ensure_dir (mtree, name, &child_mtree, error))
goto out;
if (!stage_directory_to_mtree_internal (self, child, child_mtree,
modifier, path, cancellable, error))
goto out;
}
else if (repo_dir)
{
if (!ostree_mutable_tree_replace_file (mtree, name,
ostree_repo_file_get_checksum ((OstreeRepoFile*) child),
error))
goto out;
}
else
{
ot_clear_checksum (&child_file_checksum);
ot_clear_gvariant (&xattrs);
g_clear_object (&file_input);
if (g_file_info_get_file_type (modified_info) == G_FILE_TYPE_REGULAR)
{
file_input = (GInputStream*)g_file_read (child, cancellable, error);
if (!file_input)
goto out;
}
if (!(modifier && modifier->skip_xattrs))
{
xattrs = ostree_get_xattrs_for_file (child, error);
if (!xattrs)
goto out;
}
if (!stage_object_impl (self, OSTREE_OBJECT_TYPE_RAW_FILE,
modified_info, xattrs, file_input, NULL,
&child_file_checksum, cancellable, error))
goto out;
if (!ostree_mutable_tree_replace_file (mtree, name,
g_checksum_get_string (child_file_checksum),
error))
goto out;
}
g_ptr_array_remove_index (path, path->len - 1);
}
g_clear_object (&child_info);
}
if (temp_error != NULL)
{
g_propagate_error (error, temp_error);
goto out;
}
} }
if (repo_dir && repo_dir_was_empty) if (repo_dir && repo_dir_was_empty)
@ -1624,6 +1676,28 @@ ostree_repo_stage_directory_to_mtree (OstreeRepo *self,
return ret; return ret;
} }
gboolean
ostree_repo_stage_directory_to_mtree (OstreeRepo *self,
GFile *dir,
OstreeMutableTree *mtree,
OstreeRepoCommitModifier *modifier,
GCancellable *cancellable,
GError **error)
{
gboolean ret = FALSE;
GPtrArray *path = NULL;
path = g_ptr_array_new ();
if (!stage_directory_to_mtree_internal (self, dir, mtree, modifier, path, cancellable, error))
goto out;
ret = TRUE;
out:
if (path)
g_ptr_array_free (path, TRUE);
return ret;
}
gboolean gboolean
ostree_repo_stage_mtree (OstreeRepo *self, ostree_repo_stage_mtree (OstreeRepo *self,
OstreeMutableTree *mtree, OstreeMutableTree *mtree,
@ -2020,8 +2094,6 @@ OstreeRepoCommitModifier *
ostree_repo_commit_modifier_new (void) ostree_repo_commit_modifier_new (void)
{ {
OstreeRepoCommitModifier *modifier = g_new0 (OstreeRepoCommitModifier, 1); OstreeRepoCommitModifier *modifier = g_new0 (OstreeRepoCommitModifier, 1);
modifier->uid = -1;
modifier->gid = -1;
modifier->refcount = 1; modifier->refcount = 1;

View File

@ -135,15 +135,25 @@ gboolean ostree_repo_load_variant (OstreeRepo *self,
GVariant **out_variant, GVariant **out_variant,
GError **error); GError **error);
typedef enum {
OSTREE_REPO_COMMIT_FILTER_ALLOW,
OSTREE_REPO_COMMIT_FILTER_SKIP
} OstreeRepoCommitFilterResult;
typedef OstreeRepoCommitFilterResult (*OstreeRepoCommitFilter) (OstreeRepo *repo,
const char *path,
GFileInfo *file_info,
gpointer user_data);
typedef struct { typedef struct {
volatile gint refcount; volatile gint refcount;
gint uid;
gint gid;
guint reserved_flags : 31; guint reserved_flags : 31;
guint skip_xattrs : 1; guint skip_xattrs : 1;
OstreeRepoCommitFilter filter;
gpointer user_data;
gpointer reserved[3]; gpointer reserved[3];
} OstreeRepoCommitModifier; } OstreeRepoCommitModifier;

View File

@ -36,6 +36,7 @@ static char *body;
static char *parent; static char *parent;
static char *branch; static char *branch;
static char **metadata_strings; static char **metadata_strings;
static char *statoverride_file;
static gboolean skip_if_unchanged; static gboolean skip_if_unchanged;
static gboolean tar_autocreate_parents; static gboolean tar_autocreate_parents;
static gboolean no_xattrs; static gboolean no_xattrs;
@ -57,9 +58,93 @@ static GOptionEntry options[] = {
{ "no-xattrs", 0, 0, G_OPTION_ARG_NONE, &no_xattrs, "Do not import extended attributes", NULL }, { "no-xattrs", 0, 0, G_OPTION_ARG_NONE, &no_xattrs, "Do not import extended attributes", NULL },
{ "tar-autocreate-parents", 0, 0, G_OPTION_ARG_NONE, &tar_autocreate_parents, "When loading tar archives, automatically create parent directories as needed", NULL }, { "tar-autocreate-parents", 0, 0, G_OPTION_ARG_NONE, &tar_autocreate_parents, "When loading tar archives, automatically create parent directories as needed", NULL },
{ "skip-if-unchanged", 0, 0, G_OPTION_ARG_NONE, &skip_if_unchanged, "If the contents are unchanged from previous commit, do nothing", NULL }, { "skip-if-unchanged", 0, 0, G_OPTION_ARG_NONE, &skip_if_unchanged, "If the contents are unchanged from previous commit, do nothing", NULL },
{ "statoverride", 0, 0, G_OPTION_ARG_FILENAME, &statoverride_file, "File containing list of modifications to make to permissions", "path" },
{ NULL } { NULL }
}; };
static gboolean
parse_statoverride_file (GHashTable **out_mode_add,
GCancellable *cancellable,
GError **error)
{
gboolean ret = FALSE;
GFile *path = NULL;
char *contents = NULL;
gsize len;
GHashTable *ret_hash = NULL;
char **lines = NULL;
char **iter = NULL;
path = ot_gfile_new_for_path (statoverride_file);
if (!g_file_load_contents (path, cancellable, &contents, &len, NULL,
error))
goto out;
ret_hash = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
lines = g_strsplit (contents, "\n", -1);
for (iter = lines; iter && *iter; iter++)
{
const char *line = *iter;
if (*line == '+')
{
const char *spc;
guint mode_add;
spc = strchr (line + 1, ' ');
if (!spc)
{
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
"Malformed statoverride file");
goto out;
}
mode_add = (guint32)(gint32)g_ascii_strtod (line + 1, NULL);
g_hash_table_insert (ret_hash,
g_strdup (spc + 1),
GUINT_TO_POINTER((gint32)mode_add));
}
}
ret = TRUE;
ot_transfer_out_value (out_mode_add, &ret_hash);
out:
if (ret_hash)
g_hash_table_unref (ret_hash);
g_free (contents);
g_strfreev (lines);
g_clear_object (&path);
return ret;
}
static OstreeRepoCommitFilterResult
commit_filter (OstreeRepo *self,
const char *path,
GFileInfo *file_info,
gpointer user_data)
{
GHashTable *mode_adds = user_data;
gpointer value;
if (owner_uid >= 0)
g_file_info_set_attribute_uint32 (file_info, "unix::uid", owner_uid);
if (owner_gid >= 0)
g_file_info_set_attribute_uint32 (file_info, "unix::gid", owner_gid);
if (mode_adds && g_hash_table_lookup_extended (mode_adds, path, NULL, &value))
{
guint current_mode = g_file_info_get_attribute_uint32 (file_info, "unix::mode");
guint mode_add = GPOINTER_TO_UINT (value);
g_file_info_set_attribute_uint32 (file_info, "unix::mode",
current_mode | mode_add);
g_hash_table_remove (mode_adds, path);
}
return OSTREE_REPO_COMMIT_FILTER_ALLOW;
}
gboolean gboolean
ostree_builtin_commit (int argc, char **argv, GFile *repo_path, GError **error) ostree_builtin_commit (int argc, char **argv, GFile *repo_path, GError **error)
{ {
@ -82,6 +167,7 @@ ostree_builtin_commit (int argc, char **argv, GFile *repo_path, GError **error)
gboolean metadata_builder_initialized = FALSE; gboolean metadata_builder_initialized = FALSE;
gboolean skip_commit = FALSE; gboolean skip_commit = FALSE;
gboolean in_transaction = FALSE; gboolean in_transaction = FALSE;
GHashTable *mode_adds = NULL;
context = g_option_context_new ("[ARG] - Commit a new revision"); context = g_option_context_new ("[ARG] - Commit a new revision");
g_option_context_add_main_entries (context, options, NULL); g_option_context_add_main_entries (context, options, NULL);
@ -143,6 +229,12 @@ ostree_builtin_commit (int argc, char **argv, GFile *repo_path, GError **error)
metadata_builder_initialized = FALSE; metadata_builder_initialized = FALSE;
} }
if (statoverride_file)
{
if (!parse_statoverride_file (&mode_adds, cancellable, error))
goto out;
}
repo = ostree_repo_new (repo_path); repo = ostree_repo_new (repo_path);
if (!ostree_repo_check (repo, error)) if (!ostree_repo_check (repo, error))
goto out; goto out;
@ -161,11 +253,13 @@ ostree_builtin_commit (int argc, char **argv, GFile *repo_path, GError **error)
goto out; goto out;
} }
if (owner_uid >= 0 || owner_gid >= 0) if (owner_uid >= 0 || owner_gid >= 0 || statoverride_file != NULL
|| no_xattrs)
{ {
modifier = ostree_repo_commit_modifier_new (); modifier = ostree_repo_commit_modifier_new ();
modifier->uid = owner_uid; modifier->skip_xattrs = no_xattrs;
modifier->gid = owner_gid; modifier->filter = commit_filter;
modifier->user_data = mode_adds;
} }
if (!ostree_repo_resolve_rev (repo, branch, TRUE, &parent, error)) if (!ostree_repo_resolve_rev (repo, branch, TRUE, &parent, error))
@ -249,6 +343,22 @@ ostree_builtin_commit (int argc, char **argv, GFile *repo_path, GError **error)
} }
} }
} }
if (mode_adds && g_hash_table_size (mode_adds) > 0)
{
GHashTableIter hash_iter;
gpointer key, value;
g_hash_table_iter_init (&hash_iter, mode_adds);
while (g_hash_table_iter_next (&hash_iter, &key, &value))
{
g_printerr ("Unmatched statoverride path: %s\n", (char*)key);
}
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
"Unmatched statoverride paths");
goto out;
}
if (!ostree_repo_stage_mtree (repo, mtree, &contents_checksum, cancellable, error)) if (!ostree_repo_stage_mtree (repo, mtree, &contents_checksum, cancellable, error))
goto out; goto out;
@ -316,6 +426,8 @@ ostree_builtin_commit (int argc, char **argv, GFile *repo_path, GError **error)
g_free (contents_checksum); g_free (contents_checksum);
g_free (parent); g_free (parent);
ot_clear_gvariant(&parent_commit); ot_clear_gvariant(&parent_commit);
if (mode_adds)
g_hash_table_unref (mode_adds);
g_free (tree_type); g_free (tree_type);
if (metadata_mappedf) if (metadata_mappedf)
g_mapped_file_unref (metadata_mappedf); g_mapped_file_unref (metadata_mappedf);

View File

@ -19,7 +19,7 @@
set -e set -e
echo "1..23" echo "1..24"
. libtest.sh . libtest.sh
@ -173,3 +173,11 @@ echo "ok metadata commit with strings"
cd ${test_tmpdir}/checkout-test2-4 cd ${test_tmpdir}/checkout-test2-4
$OSTREE commit -b test2 -s "no xattrs" --no-xattrs $OSTREE commit -b test2 -s "no xattrs" --no-xattrs
echo "ok commit with no xattrs" echo "ok commit with no xattrs"
cd ${test_tmpdir}
cat > test-statoverride.txt <<EOF
+2048 /a/nested/3
EOF
cd ${test_tmpdir}/checkout-test2-4
$OSTREE commit -b test2 -s "with statoverride" --statoverride=../test-statoverride.txt
echo "ok commit statoverridde"