core: Add ability for repositories to have a "parent"

This will be useful for ostbuild; a user can create their own archive
mode repository which transparently inherits objects from the
root-owned one in /ostree.
This commit is contained in:
Colin Walters 2012-04-16 21:21:50 -04:00
parent 2ecc0cdef1
commit 5947b5b145
8 changed files with 339 additions and 48 deletions

View File

@ -22,6 +22,7 @@ bin_PROGRAMS += ostree
ostree_SOURCES = src/ostree/main.c \ ostree_SOURCES = src/ostree/main.c \
src/ostree/ot-builtins.h \ src/ostree/ot-builtins.h \
src/ostree/ot-builtin-cat.c \ src/ostree/ot-builtin-cat.c \
src/ostree/ot-builtin-config.c \
src/ostree/ot-builtin-checkout.c \ src/ostree/ot-builtin-checkout.c \
src/ostree/ot-builtin-checksum.c \ src/ostree/ot-builtin-checksum.c \
src/ostree/ot-builtin-commit.c \ src/ostree/ot-builtin-commit.c \

View File

@ -40,6 +40,16 @@
#include "ostree-libarchive-input-stream.h" #include "ostree-libarchive-input-stream.h"
#endif #endif
static gboolean
repo_find_object (OstreeRepo *self,
OstreeObjectType objtype,
const char *checksum,
GFile **out_stored_path,
char **out_pack_checksum,
guint64 *out_pack_offset,
GCancellable *cancellable,
GError **error);
enum { enum {
PROP_0, PROP_0,
@ -73,6 +83,8 @@ struct _OstreeRepoPrivate {
GKeyFile *config; GKeyFile *config;
OstreeRepoMode mode; OstreeRepoMode mode;
OstreeRepo *parent_repo;
GHashTable *pack_index_mappings; GHashTable *pack_index_mappings;
GHashTable *pack_data_mappings; GHashTable *pack_data_mappings;
}; };
@ -83,6 +95,8 @@ ostree_repo_finalize (GObject *object)
OstreeRepo *self = OSTREE_REPO (object); OstreeRepo *self = OSTREE_REPO (object);
OstreeRepoPrivate *priv = GET_PRIVATE (self); OstreeRepoPrivate *priv = GET_PRIVATE (self);
g_clear_object (&priv->parent_repo);
g_clear_object (&priv->repodir); g_clear_object (&priv->repodir);
g_clear_object (&priv->tmp_dir); g_clear_object (&priv->tmp_dir);
g_clear_object (&priv->pending_dir); g_clear_object (&priv->pending_dir);
@ -397,7 +411,14 @@ ostree_repo_resolve_rev (OstreeRepo *self,
if (child == NULL) if (child == NULL)
{ {
if (!allow_noent) if (priv->parent_repo)
{
if (!ostree_repo_resolve_rev (priv->parent_repo, rev,
allow_noent, &ret_rev,
error))
goto out;
}
else if (!allow_noent)
{ {
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED, g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
"Rev '%s' not found", rev); "Rev '%s' not found", rev);
@ -641,8 +662,9 @@ ostree_repo_check (OstreeRepo *self, GError **error)
gboolean ret = FALSE; gboolean ret = FALSE;
OstreeRepoPrivate *priv = GET_PRIVATE (self); OstreeRepoPrivate *priv = GET_PRIVATE (self);
gboolean is_archive; gboolean is_archive;
ot_lfree char *version = NULL;; ot_lfree char *version = NULL;
ot_lfree char *mode = NULL;; ot_lfree char *mode = NULL;
ot_lfree char *parent_repo_path = NULL;
g_return_val_if_fail (error == NULL || *error == NULL, FALSE); g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
@ -702,6 +724,24 @@ ostree_repo_check (OstreeRepo *self, GError **error)
} }
} }
if (!keyfile_get_value_with_default (priv->config, "core", "parent",
NULL, &parent_repo_path, error))
goto out;
if (parent_repo_path && parent_repo_path[0])
{
ot_lobj GFile *parent_repo_f = ot_gfile_new_for_path (parent_repo_path);
priv->parent_repo = ostree_repo_new (parent_repo_f);
if (!ostree_repo_check (priv->parent_repo, error))
{
g_prefix_error (error, "While checking parent repository '%s': ",
ot_gfile_get_path_cached (parent_repo_f));
goto out;
}
}
priv->inited = TRUE; priv->inited = TRUE;
ret = TRUE; ret = TRUE;
@ -793,6 +833,7 @@ impl_stage_archive_file_object (OstreeRepo *self,
GFileInfo *file_info, GFileInfo *file_info,
GVariant *xattrs, GVariant *xattrs,
GInputStream *input, GInputStream *input,
gboolean store_if_packed,
const char *expected_checksum, const char *expected_checksum,
guchar **out_csum, guchar **out_csum,
GCancellable *cancellable, GCancellable *cancellable,
@ -801,6 +842,8 @@ impl_stage_archive_file_object (OstreeRepo *self,
gboolean ret = FALSE; gboolean ret = FALSE;
OstreeRepoPrivate *priv = GET_PRIVATE (self); OstreeRepoPrivate *priv = GET_PRIVATE (self);
const char *actual_checksum; const char *actual_checksum;
gboolean have_obj;
gboolean do_commit;
ot_lvariant GVariant *archive_metadata = NULL; ot_lvariant GVariant *archive_metadata = NULL;
ot_lobj GFileInfo *temp_info = NULL; ot_lobj GFileInfo *temp_info = NULL;
ot_lobj GFile *temp_file = NULL; ot_lobj GFile *temp_file = NULL;
@ -850,16 +893,34 @@ impl_stage_archive_file_object (OstreeRepo *self,
else else
actual_checksum = g_checksum_get_string (checksum); actual_checksum = g_checksum_get_string (checksum);
if (!store_if_packed)
{
if (!ostree_repo_has_object (self, OSTREE_OBJECT_TYPE_FILE, actual_checksum, &have_obj,
cancellable, error))
goto out;
do_commit = !have_obj;
}
else
do_commit = TRUE;
if (do_commit)
{
if (!commit_tmpfile_trusted (self, actual_checksum, OSTREE_OBJECT_TYPE_FILE, if (!commit_tmpfile_trusted (self, actual_checksum, OSTREE_OBJECT_TYPE_FILE,
temp_file, cancellable, error)) temp_file, cancellable, error))
goto out; goto out;
g_clear_object (&temp_file);
}
if (checksum) if (checksum)
ret_csum = ot_csum_from_gchecksum (checksum); ret_csum = ot_csum_from_gchecksum (checksum);
ret = TRUE; ret = TRUE;
ot_transfer_out_value (out_csum, &ret_csum); ot_transfer_out_value (out_csum, &ret_csum);
out: out:
if (temp_file)
(void) unlink (ot_gfile_get_path_cached (temp_file));
ot_clear_checksum (&checksum); ot_clear_checksum (&checksum);
return ret; return ret;
} }
@ -880,6 +941,7 @@ stage_object_impl (OstreeRepo *self,
OstreeRepoPrivate *priv = GET_PRIVATE (self); OstreeRepoPrivate *priv = GET_PRIVATE (self);
guint64 pack_offset; guint64 pack_offset;
const char *actual_checksum; const char *actual_checksum;
gboolean do_commit;
ot_lobj GFileInfo *temp_info = NULL; ot_lobj GFileInfo *temp_info = NULL;
ot_lobj GFile *temp_file = NULL; ot_lobj GFile *temp_file = NULL;
ot_lobj GFile *stored_path = NULL; ot_lobj GFile *stored_path = NULL;
@ -900,14 +962,14 @@ stage_object_impl (OstreeRepo *self,
{ {
if (!store_if_packed) if (!store_if_packed)
{ {
if (!ostree_repo_find_object (self, objtype, expected_checksum, if (!repo_find_object (self, objtype, expected_checksum,
&stored_path, &pack_checksum, &pack_offset, &stored_path, &pack_checksum, &pack_offset,
cancellable, error)) cancellable, error))
goto out; goto out;
} }
else else
{ {
if (!ostree_repo_find_object (self, objtype, expected_checksum, if (!repo_find_object (self, objtype, expected_checksum,
&stored_path, NULL, NULL, &stored_path, NULL, NULL,
cancellable, error)) cancellable, error))
goto out; goto out;
@ -933,6 +995,7 @@ stage_object_impl (OstreeRepo *self,
if (objtype == OSTREE_OBJECT_TYPE_FILE && priv->mode == OSTREE_REPO_MODE_ARCHIVE) if (objtype == OSTREE_OBJECT_TYPE_FILE && priv->mode == OSTREE_REPO_MODE_ARCHIVE)
{ {
if (!impl_stage_archive_file_object (self, file_info, xattrs, input, if (!impl_stage_archive_file_object (self, file_info, xattrs, input,
store_if_packed,
expected_checksum, expected_checksum,
out_csum ? &ret_csum : NULL, out_csum ? &ret_csum : NULL,
cancellable, error)) cancellable, error))
@ -977,12 +1040,29 @@ stage_object_impl (OstreeRepo *self,
expected_checksum, actual_checksum); expected_checksum, actual_checksum);
goto out; goto out;
} }
} }
if (!store_if_packed)
{
gboolean have_obj;
if (!ostree_repo_has_object (self, objtype, actual_checksum, &have_obj,
cancellable, error))
goto out;
do_commit = !have_obj;
}
else
do_commit = TRUE;
if (do_commit)
{
if (!commit_tmpfile_trusted (self, actual_checksum, objtype, if (!commit_tmpfile_trusted (self, actual_checksum, objtype,
temp_file, cancellable, error)) temp_file, cancellable, error))
goto out; goto out;
g_clear_object (&temp_file); g_clear_object (&temp_file);
}
if (checksum) if (checksum)
ret_csum = ot_csum_from_gchecksum (checksum); ret_csum = ot_csum_from_gchecksum (checksum);
@ -3140,6 +3220,7 @@ ostree_repo_load_file (OstreeRepo *self,
GError **error) GError **error)
{ {
gboolean ret = FALSE; gboolean ret = FALSE;
OstreeRepoPrivate *priv = GET_PRIVATE (self);
guchar *pack_data; guchar *pack_data;
guint64 pack_len; guint64 pack_len;
guint64 pack_offset; guint64 pack_offset;
@ -3154,7 +3235,7 @@ ostree_repo_load_file (OstreeRepo *self,
ot_lobj GFileInfo *ret_file_info = NULL; ot_lobj GFileInfo *ret_file_info = NULL;
ot_lvariant GVariant *ret_xattrs = NULL; ot_lvariant GVariant *ret_xattrs = NULL;
if (!ostree_repo_find_object (self, OSTREE_OBJECT_TYPE_FILE, if (!repo_find_object (self, OSTREE_OBJECT_TYPE_FILE,
checksum, &loose_path, checksum, &loose_path,
&pack_checksum, &pack_offset, &pack_checksum, &pack_offset,
cancellable, error)) cancellable, error))
@ -3212,6 +3293,15 @@ ostree_repo_load_file (OstreeRepo *self,
cancellable, error)) cancellable, error))
goto out; goto out;
} }
else if (priv->parent_repo)
{
if (!ostree_repo_load_file (priv->parent_repo, checksum,
out_input ? &ret_input : NULL,
out_file_info ? &ret_file_info : NULL,
out_xattrs ? &ret_xattrs : NULL,
cancellable, error))
goto out;
}
else else
{ {
g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND, g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND,
@ -3399,8 +3489,8 @@ find_object_in_packs (OstreeRepo *self,
return ret; return ret;
} }
gboolean static gboolean
ostree_repo_find_object (OstreeRepo *self, repo_find_object (OstreeRepo *self,
OstreeObjectType objtype, OstreeObjectType objtype,
const char *checksum, const char *checksum,
GFile **out_stored_path, GFile **out_stored_path,
@ -3445,6 +3535,41 @@ out:
return ret; return ret;
} }
gboolean
ostree_repo_has_object (OstreeRepo *self,
OstreeObjectType objtype,
const char *checksum,
gboolean *out_have_object,
GCancellable *cancellable,
GError **error)
{
gboolean ret = FALSE;
OstreeRepoPrivate *priv = GET_PRIVATE (self);
gboolean ret_have_object;
ot_lobj GFile *loose_path = NULL;
ot_lfree char *pack_checksum = NULL;
if (!repo_find_object (self, objtype, checksum, &loose_path,
&pack_checksum, NULL,
cancellable, error))
goto out;
ret_have_object = (loose_path != NULL) || (pack_checksum != NULL);
if (!ret_have_object && priv->parent_repo)
{
if (!ostree_repo_has_object (priv->parent_repo, objtype, checksum,
&ret_have_object, cancellable, error))
goto out;
}
ret = TRUE;
if (out_have_object)
*out_have_object = ret_have_object;
out:
return ret;
}
gboolean gboolean
ostree_repo_load_variant_c (OstreeRepo *self, ostree_repo_load_variant_c (OstreeRepo *self,
OstreeObjectType objtype, OstreeObjectType objtype,
@ -3473,6 +3598,7 @@ ostree_repo_load_variant (OstreeRepo *self,
GError **error) GError **error)
{ {
gboolean ret = FALSE; gboolean ret = FALSE;
OstreeRepoPrivate *priv = GET_PRIVATE (self);
guchar *pack_data; guchar *pack_data;
guint64 pack_len; guint64 pack_len;
guint64 object_offset; guint64 object_offset;
@ -3484,7 +3610,7 @@ ostree_repo_load_variant (OstreeRepo *self,
g_return_val_if_fail (OSTREE_OBJECT_TYPE_IS_META (objtype), FALSE); g_return_val_if_fail (OSTREE_OBJECT_TYPE_IS_META (objtype), FALSE);
if (!ostree_repo_find_object (self, objtype, sha256, &object_path, if (!repo_find_object (self, objtype, sha256, &object_path,
&pack_checksum, &object_offset, &pack_checksum, &object_offset,
cancellable, error)) cancellable, error))
goto out; goto out;
@ -3508,6 +3634,11 @@ ostree_repo_load_variant (OstreeRepo *self,
g_variant_get_child (packed_object, 2, "v", &ret_variant); g_variant_get_child (packed_object, 2, "v", &ret_variant);
} }
else if (priv->parent_repo)
{
if (!ostree_repo_load_variant (priv->parent_repo, objtype, sha256, &ret_variant, error))
goto out;
}
else else
{ {
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED, g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
@ -3562,12 +3693,22 @@ ostree_repo_list_objects (OstreeRepo *self,
{ {
if (!list_loose_objects (self, ret_objects, cancellable, error)) if (!list_loose_objects (self, ret_objects, cancellable, error))
goto out; goto out;
if (priv->parent_repo)
{
if (!list_loose_objects (priv->parent_repo, ret_objects, cancellable, error))
goto out;
}
} }
if (flags & OSTREE_REPO_LIST_OBJECTS_PACKED) if (flags & OSTREE_REPO_LIST_OBJECTS_PACKED)
{ {
if (!list_packed_objects (self, ret_objects, cancellable, error)) if (!list_packed_objects (self, ret_objects, cancellable, error))
goto out; goto out;
if (priv->parent_repo)
{
if (!list_packed_objects (priv->parent_repo, ret_objects, cancellable, error))
goto out;
}
} }
ret = TRUE; ret = TRUE;

View File

@ -92,12 +92,10 @@ gboolean ostree_repo_abort_transaction (OstreeRepo *self,
GCancellable *cancellable, GCancellable *cancellable,
GError **error); GError **error);
gboolean ostree_repo_find_object (OstreeRepo *self, gboolean ostree_repo_has_object (OstreeRepo *self,
OstreeObjectType objtype, OstreeObjectType objtype,
const char *checksum, const char *checksum,
GFile **out_stored_path, gboolean *out_have_object,
char **out_pack_checksum,
guint64 *out_pack_offset,
GCancellable *cancellable, GCancellable *cancellable,
GError **error); GError **error);

View File

@ -31,6 +31,7 @@
static OstreeBuiltin builtins[] = { static OstreeBuiltin builtins[] = {
{ "cat", ostree_builtin_cat, 0 }, { "cat", ostree_builtin_cat, 0 },
{ "config", ostree_builtin_config, 0 },
{ "checkout", ostree_builtin_checkout, 0 }, { "checkout", ostree_builtin_checkout, 0 },
{ "checksum", ostree_builtin_checksum, OSTREE_BUILTIN_FLAG_NO_REPO }, { "checksum", ostree_builtin_checksum, OSTREE_BUILTIN_FLAG_NO_REPO },
{ "diff", ostree_builtin_diff, 0 }, { "diff", ostree_builtin_diff, 0 },

View File

@ -514,17 +514,12 @@ find_object_ensure_indexes (OtPullData *pull_data,
{ {
gboolean ret = FALSE; gboolean ret = FALSE;
gboolean ret_is_stored; gboolean ret_is_stored;
ot_lobj GFile *stored_path = NULL;
ot_lfree char *local_pack_checksum = NULL;
ot_lfree char *ret_remote_pack_checksum = NULL; ot_lfree char *ret_remote_pack_checksum = NULL;
if (!ostree_repo_find_object (pull_data->repo, objtype, checksum, if (!ostree_repo_has_object (pull_data->repo, objtype, checksum, &ret_is_stored,
&stored_path, &local_pack_checksum, NULL,
cancellable, error)) cancellable, error))
goto out; goto out;
ret_is_stored = (stored_path != NULL || local_pack_checksum != NULL);
if (!ret_is_stored) if (!ret_is_stored)
{ {
if (!pull_data->fetched_packs) if (!pull_data->fetched_packs)

View File

@ -0,0 +1,144 @@
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
*
* Copyright (C) 2011 Colin Walters <walters@verbum.org>
*
* 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 <walters@verbum.org>
*/
#include "config.h"
#include "ot-builtins.h"
#include "ostree.h"
#include <glib/gi18n.h>
static GOptionEntry options[] = {
{ NULL }
};
static gboolean
split_key_string (const char *k,
char **out_section,
char **out_value,
GError **error)
{
const char *dot = strchr (k, '.');
if (!dot)
{
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
"Key must be of the form \"sectionname.keyname\"");
return FALSE;
}
*out_section = g_strndup (k, dot - k);
*out_value = g_strdup (dot + 1);
return TRUE;
}
gboolean
ostree_builtin_config (int argc, char **argv, GFile *repo_path, GError **error)
{
GOptionContext *context = NULL;
gboolean ret = FALSE;
const char *op;
const char *section_key;
const char *value;
ot_lobj OstreeRepo *repo = NULL;
ot_lfree char *section = NULL;
ot_lfree char *key = NULL;
GKeyFile *config = NULL;
context = g_option_context_new ("- Change configuration settings");
g_option_context_add_main_entries (context, options, NULL);
if (!g_option_context_parse (context, &argc, &argv, error))
goto out;
repo = ostree_repo_new (repo_path);
if (!ostree_repo_check (repo, error))
goto out;
if (argc < 2)
{
ot_util_usage_error (context, "OPERATION must be specified", error);
goto out;
}
op = argv[1];
if (!strcmp (op, "set"))
{
if (argc < 4)
{
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
"KEY and VALUE must be specified");
goto out;
}
section_key = argv[2];
value = argv[3];
if (!split_key_string (section_key, &section, &key, error))
goto out;
config = ostree_repo_copy_config (repo);
g_key_file_set_string (config, section, key, value);
if (!ostree_repo_write_config (repo, config, error))
goto out;
}
else if (!strcmp (op, "set"))
{
ot_lfree char *value = NULL;
if (argc < 3)
{
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
"KEY must be specified");
goto out;
}
section_key = argv[2];
if (!split_key_string (section_key, &section, &key, error))
goto out;
config = g_key_file_ref (ostree_repo_get_config (repo));
value = g_key_file_get_string (config, section, key, error);
if (value == NULL)
goto out;
g_print ("%s\n", value);
}
else
{
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
"Unknown operation %s", op);
goto out;
}
ret = TRUE;
out:
if (config)
g_key_file_free (config);
if (context)
g_option_context_free (context);
return ret;
}

View File

@ -28,6 +28,7 @@
G_BEGIN_DECLS G_BEGIN_DECLS
gboolean ostree_builtin_cat (int argc, char **argv, GFile *repo_path, GError **error); gboolean ostree_builtin_cat (int argc, char **argv, GFile *repo_path, GError **error);
gboolean ostree_builtin_config (int argc, char **argv, GFile *repo_path, GError **error);
gboolean ostree_builtin_checkout (int argc, char **argv, GFile *repo_path, GError **error); gboolean ostree_builtin_checkout (int argc, char **argv, GFile *repo_path, GError **error);
gboolean ostree_builtin_checksum (int argc, char **argv, GFile *repo_path, GError **error); gboolean ostree_builtin_checksum (int argc, char **argv, GFile *repo_path, GError **error);
gboolean ostree_builtin_commit (int argc, char **argv, GFile *repo_path, GError **error); gboolean ostree_builtin_commit (int argc, char **argv, GFile *repo_path, GError **error);

View File

@ -19,7 +19,7 @@
set -e set -e
echo "1..30" echo "1..31"
. libtest.sh . libtest.sh
@ -219,3 +219,13 @@ $OSTREE checkout --link-cache=linkcache test2 test2-checkout-from-link-cache
cd test2-checkout-from-link-cache cd test2-checkout-from-link-cache
assert_file_has_content ./yet/another/tree/green "leaf" assert_file_has_content ./yet/another/tree/green "leaf"
echo "ok checkout link cache" echo "ok checkout link cache"
cd ${test_tmpdir}
rm -rf shadow-repo
mkdir shadow-repo
ostree --repo=shadow-repo init
ostree --repo=shadow-repo config set core.parent $(pwd)/repo
rm -rf test2-checkout
parent_rev_test2=$(ostree --repo=repo rev-parse test2)
ostree --repo=shadow-repo checkout "${parent_rev_test2}" test2-checkout
echo "ok checkout from shadow repo"