/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- * * Copyright (C) 2011 Colin Walters * * 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, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. * * Author: Colin Walters */ #include "config.h" #include "otutil.h" #include "ostree.h" /** * SECTION:ostree-mutable-tree * @title: In-memory modifiable filesystem tree * @short_description: Modifiable filesystem tree * * In order to commit content into an #OstreeRepo, it must first be * imported into an #OstreeMutableTree. There are several high level * APIs to create an initiable #OstreeMutableTree from a physical * filesystem directory, but they may also be computed * programmatically. */ /** * OstreeMutableTree: * * Private instance structure. */ struct OstreeMutableTree { GObject parent_instance; char *contents_checksum; char *metadata_checksum; GHashTable *files; GHashTable *subdirs; }; G_DEFINE_TYPE (OstreeMutableTree, ostree_mutable_tree, G_TYPE_OBJECT) static void ostree_mutable_tree_finalize (GObject *object) { OstreeMutableTree *self; self = OSTREE_MUTABLE_TREE (object); g_free (self->contents_checksum); g_free (self->metadata_checksum); g_hash_table_destroy (self->files); g_hash_table_destroy (self->subdirs); G_OBJECT_CLASS (ostree_mutable_tree_parent_class)->finalize (object); } static void ostree_mutable_tree_class_init (OstreeMutableTreeClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); gobject_class->finalize = ostree_mutable_tree_finalize; } static void ostree_mutable_tree_init (OstreeMutableTree *self) { self->files = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free); self->subdirs = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, (GDestroyNotify)g_object_unref); } void ostree_mutable_tree_set_metadata_checksum (OstreeMutableTree *self, const char *checksum) { g_free (self->metadata_checksum); self->metadata_checksum = g_strdup (checksum); } const char * ostree_mutable_tree_get_metadata_checksum (OstreeMutableTree *self) { return self->metadata_checksum; } void ostree_mutable_tree_set_contents_checksum (OstreeMutableTree *self, const char *checksum) { g_free (self->contents_checksum); self->contents_checksum = g_strdup (checksum); } const char * ostree_mutable_tree_get_contents_checksum (OstreeMutableTree *self) { if (!self->contents_checksum) return NULL; /* Ensure the cache is valid; this implementation is a bit * lame in that we walk the whole tree every time this * getter is called; a better approach would be to invalidate * all of the parents whenever a child is modified. * * However, we only call this function once right now. */ GLNX_HASH_TABLE_FOREACH_V (self->subdirs, OstreeMutableTree*, subdir) { if (!ostree_mutable_tree_get_contents_checksum (subdir)) { g_free (self->contents_checksum); self->contents_checksum = NULL; return NULL; } } return self->contents_checksum; } static gboolean set_error_noent (GError **error, const char *path) { g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND, "No such file or directory: %s", path); return FALSE; } gboolean ostree_mutable_tree_replace_file (OstreeMutableTree *self, const char *name, const char *checksum, GError **error) { gboolean ret = FALSE; g_return_val_if_fail (name != NULL, FALSE); if (!ot_util_filename_validate (name, error)) goto out; if (g_hash_table_lookup (self->subdirs, name)) { g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED, "Can't replace directory with file: %s", name); goto out; } ostree_mutable_tree_set_contents_checksum (self, NULL); g_hash_table_replace (self->files, g_strdup (name), g_strdup (checksum)); ret = TRUE; out: return ret; } gboolean ostree_mutable_tree_ensure_dir (OstreeMutableTree *self, const char *name, OstreeMutableTree **out_subdir, GError **error) { gboolean ret = FALSE; g_autoptr(OstreeMutableTree) ret_dir = NULL; g_return_val_if_fail (name != NULL, FALSE); if (!ot_util_filename_validate (name, error)) goto out; if (g_hash_table_lookup (self->files, name)) { g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED, "Can't replace file with directory: %s", name); goto out; } ret_dir = ot_gobject_refz (g_hash_table_lookup (self->subdirs, name)); if (!ret_dir) { ret_dir = ostree_mutable_tree_new (); ostree_mutable_tree_set_contents_checksum (self, NULL); g_hash_table_insert (self->subdirs, g_strdup (name), g_object_ref (ret_dir)); } ret = TRUE; ot_transfer_out_value (out_subdir, &ret_dir); out: return ret; } gboolean ostree_mutable_tree_lookup (OstreeMutableTree *self, const char *name, char **out_file_checksum, OstreeMutableTree **out_subdir, GError **error) { gboolean ret = FALSE; g_autoptr(OstreeMutableTree) ret_subdir = NULL; g_autofree char *ret_file_checksum = NULL; ret_subdir = ot_gobject_refz (g_hash_table_lookup (self->subdirs, name)); if (!ret_subdir) { ret_file_checksum = g_strdup (g_hash_table_lookup (self->files, name)); if (!ret_file_checksum) { set_error_noent (error, name); goto out; } } ret = TRUE; ot_transfer_out_value (out_file_checksum, &ret_file_checksum); ot_transfer_out_value (out_subdir, &ret_subdir); out: return ret; } /** * ostree_mutable_tree_ensure_parent_dirs: * @self: Tree * @split_path: (element-type utf8): File path components * @metadata_checksum: SHA256 checksum for metadata * @out_parent: (out) (transfer full): The parent tree * @error: a #GError * * Create all parent trees necessary for the given @split_path to * exist. */ gboolean ostree_mutable_tree_ensure_parent_dirs (OstreeMutableTree *self, GPtrArray *split_path, const char *metadata_checksum, OstreeMutableTree **out_parent, GError **error) { gboolean ret = FALSE; int i; OstreeMutableTree *subdir = self; /* nofree */ g_autoptr(OstreeMutableTree) ret_parent = NULL; g_assert (metadata_checksum != NULL); if (!self->metadata_checksum) ostree_mutable_tree_set_metadata_checksum (self, metadata_checksum); for (i = 0; i+1 < split_path->len; i++) { OstreeMutableTree *next; const char *name = split_path->pdata[i]; if (g_hash_table_lookup (subdir->files, name)) { g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED, "Can't replace file with directory: %s", name); goto out; } next = g_hash_table_lookup (subdir->subdirs, name); if (!next) { next = ostree_mutable_tree_new (); ostree_mutable_tree_set_metadata_checksum (next, metadata_checksum); g_hash_table_insert (subdir->subdirs, g_strdup (name), next); } subdir = next; } ret_parent = g_object_ref (subdir); ret = TRUE; ot_transfer_out_value (out_parent, &ret_parent); out: return ret; } /** * ostree_mutable_tree_walk: * @self: Tree * @split_path: (element-type utf8): Split pathname * @start: Descend from this number of elements in @split_path * @out_subdir: (out) (transfer full): Target parent * @error: Error * * Traverse @start number of elements starting from @split_path; the * child will be returned in @out_subdir. */ gboolean ostree_mutable_tree_walk (OstreeMutableTree *self, GPtrArray *split_path, guint start, OstreeMutableTree **out_subdir, GError **error) { g_return_val_if_fail (start < split_path->len, FALSE); if (start == split_path->len - 1) { *out_subdir = g_object_ref (self); return TRUE; } else { OstreeMutableTree *subdir; subdir = g_hash_table_lookup (self->subdirs, split_path->pdata[start]); if (!subdir) return set_error_noent (error, (char*)split_path->pdata[start]); return ostree_mutable_tree_walk (subdir, split_path, start + 1, out_subdir, error); } } /** * ostree_mutable_tree_get_subdirs: * @self: * * Returns: (transfer none) (element-type utf8 OstreeMutableTree): All children directories */ GHashTable * ostree_mutable_tree_get_subdirs (OstreeMutableTree *self) { return self->subdirs; } /** * ostree_mutable_tree_get_files: * @self: * * Returns: (transfer none) (element-type utf8 utf8): All children files (the value is a checksum) */ GHashTable * ostree_mutable_tree_get_files (OstreeMutableTree *self) { return self->files; } /** * ostree_mutable_tree_new: * * Returns: (transfer full): A new tree */ OstreeMutableTree * ostree_mutable_tree_new (void) { return (OstreeMutableTree*)g_object_new (OSTREE_TYPE_MUTABLE_TREE, NULL); }