summaryrefslogtreecommitdiff
path: root/src/libostree/ostree-sign.c
diff options
context:
space:
mode:
authorDenis Pynkin <denis.pynkin@collabora.com>2019-07-29 02:32:28 +0300
committerDenis Pynkin <denis.pynkin@collabora.com>2020-03-25 15:23:54 +0300
commitedbbe1c4f2267c0d95d4ee14f6cbc516a1d31dca (patch)
treee3c6c25522ff009865283668ec1ee28fabe4d511 /src/libostree/ostree-sign.c
parent84c8164610ee3df9bbd06f0be9e37a873708ec2d (diff)
downloadostree-edbbe1c4f2267c0d95d4ee14f6cbc516a1d31dca.tar.gz
lib/sign: initial implementation
Added the initial version of signing interface allowing to allowing to sign and verify commits. Implemented initial signing modules: - dummy -- simple module allowing to sign/verify with ASCII string - ed25519 -- module allowing to sign/verify commit with ed25519 (EdDSA) signature scheme provided by libsodium library. Signed-off-by: Denis Pynkin <denis.pynkin@collabora.com>
Diffstat (limited to 'src/libostree/ostree-sign.c')
-rw-r--r--src/libostree/ostree-sign.c347
1 files changed, 347 insertions, 0 deletions
diff --git a/src/libostree/ostree-sign.c b/src/libostree/ostree-sign.c
new file mode 100644
index 00000000..96455f86
--- /dev/null
+++ b/src/libostree/ostree-sign.c
@@ -0,0 +1,347 @@
+/* vim:set et sw=2 cin cino=t0,f0,(0,{s,>2s,n-s,^-s,e2s: */
+
+/*
+ * Copyright © 2019 Collabora Ltd.
+ *
+ * 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, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ *
+ */
+
+#include "config.h"
+
+#include <unistd.h>
+#include <sys/types.h>
+#include <fcntl.h>
+#include "libglnx.h"
+#include "otutil.h"
+
+#include "ostree-autocleanups.h"
+#include "ostree-core.h"
+#include "ostree-sign.h"
+#include "ostree-sign-dummy.h"
+#ifdef HAVE_LIBSODIUM
+#include "ostree-sign-ed25519.h"
+#endif
+
+#define G_LOG_DOMAIN "OSTreeSign"
+
+G_DEFINE_INTERFACE (OstreeSign, ostree_sign, G_TYPE_OBJECT)
+
+static void
+ostree_sign_default_init (OstreeSignInterface *iface)
+{
+ g_debug ("OstreeSign initialization");
+}
+
+gchar * ostree_sign_metadata_key (OstreeSign *self)
+{
+ g_debug ("%s enter", __FUNCTION__);
+
+ g_return_val_if_fail (OSTREE_SIGN_GET_IFACE (self)->metadata_key != NULL, NULL);
+ return OSTREE_SIGN_GET_IFACE (self)->metadata_key (self);
+}
+
+gchar * ostree_sign_metadata_format (OstreeSign *self)
+{
+ g_debug ("%s enter", __FUNCTION__);
+
+ g_return_val_if_fail (OSTREE_SIGN_GET_IFACE (self)->metadata_format != NULL, NULL);
+ return OSTREE_SIGN_GET_IFACE (self)->metadata_format (self);
+}
+
+gboolean ostree_sign_set_sk (OstreeSign *self,
+ GVariant *secret_key,
+ GError **error)
+{
+ g_debug ("%s enter", __FUNCTION__);
+
+ if (OSTREE_SIGN_GET_IFACE (self)->set_sk == NULL)
+ return TRUE;
+
+ return OSTREE_SIGN_GET_IFACE (self)->set_sk (self, secret_key, error);
+}
+
+gboolean ostree_sign_set_pk (OstreeSign *self,
+ GVariant *public_key,
+ GError **error)
+{
+ g_debug ("%s enter", __FUNCTION__);
+
+ if (OSTREE_SIGN_GET_IFACE (self)->set_pk == NULL)
+ return TRUE;
+
+ return OSTREE_SIGN_GET_IFACE (self)->set_pk (self, public_key, error);
+}
+
+/* Load private keys for verification from anywhere.
+ * No need to have the same function for secret keys -- the signing SW must do it in it's own way
+ * */
+gboolean
+ostree_sign_load_pk (OstreeSign *self,
+ gchar *remote_name,
+ GError **error)
+{
+ g_debug ("%s enter", __FUNCTION__);
+
+ g_return_val_if_fail (OSTREE_SIGN_GET_IFACE (self)->load_pk != NULL, FALSE);
+
+ if (remote_name == NULL)
+ remote_name = OSTREE_SIGN_ALL_REMOTES;
+
+ return OSTREE_SIGN_GET_IFACE (self)->load_pk (self, remote_name, error);
+}
+
+gboolean ostree_sign_data (OstreeSign *self,
+ GBytes *data,
+ GBytes **signature,
+ GCancellable *cancellable,
+ GError **error)
+{
+
+ g_debug ("%s enter", __FUNCTION__);
+ g_return_val_if_fail (OSTREE_IS_SIGN (self), FALSE);
+ g_return_val_if_fail (OSTREE_SIGN_GET_IFACE (self)->data != NULL, FALSE);
+
+ return OSTREE_SIGN_GET_IFACE (self)->data (self, data, signature, cancellable, error);
+}
+
+/*
+ * Adopted version of _ostree_detached_metadata_append_gpg_sig ()
+ */
+GVariant *
+ostree_sign_detached_metadata_append (OstreeSign *self,
+ GVariant *existing_metadata,
+ GBytes *signature_bytes)
+{
+ g_debug ("%s enter", __FUNCTION__);
+ g_return_val_if_fail (signature_bytes != NULL, FALSE);
+
+ GVariantDict metadata_dict;
+ g_autoptr(GVariant) signature_data = NULL;
+ g_autoptr(GVariantBuilder) signature_builder = NULL;
+
+ g_variant_dict_init (&metadata_dict, existing_metadata);
+
+ g_autofree gchar *signature_key = ostree_sign_metadata_key(self);
+ g_autofree GVariantType *signature_format = (GVariantType *) ostree_sign_metadata_format(self);
+
+ signature_data = g_variant_dict_lookup_value (&metadata_dict,
+ signature_key,
+ (GVariantType*)signature_format);
+
+ /* signature_data may be NULL */
+ signature_builder = ot_util_variant_builder_from_variant (signature_data, signature_format);
+
+ g_variant_builder_add (signature_builder, "@ay", ot_gvariant_new_ay_bytes (signature_bytes));
+
+ g_variant_dict_insert_value (&metadata_dict,
+ signature_key,
+ g_variant_builder_end (signature_builder));
+
+ return g_variant_dict_end (&metadata_dict);
+}
+
+
+gboolean
+ostree_sign_metadata_verify (OstreeSign *self,
+ GBytes *data,
+ GVariant *signatures,
+ GError **error)
+{
+ g_debug ("%s enter", __FUNCTION__);
+ g_return_val_if_fail (OSTREE_IS_SIGN (self), FALSE);
+ g_return_val_if_fail (OSTREE_SIGN_GET_IFACE (self)->metadata_verify != NULL, FALSE);
+
+ return OSTREE_SIGN_GET_IFACE (self)->metadata_verify(self, data, signatures, error);
+}
+
+gboolean
+ostree_sign_commit_verify (OstreeSign *self,
+ OstreeRepo *repo,
+ const gchar *commit_checksum,
+ GCancellable *cancellable,
+ GError **error)
+
+{
+ g_debug ("%s enter", __FUNCTION__);
+ g_return_val_if_fail (OSTREE_IS_SIGN (self), FALSE);
+
+ g_autoptr(GVariant) commit_variant = NULL;
+ /* Load the commit */
+ if (!ostree_repo_load_variant (repo, OSTREE_OBJECT_TYPE_COMMIT,
+ commit_checksum, &commit_variant,
+ error))
+ return glnx_prefix_error (error, "Failed to read commit");
+
+ /* Load the metadata */
+ g_autoptr(GVariant) metadata = NULL;
+ if (!ostree_repo_read_commit_detached_metadata (repo,
+ commit_checksum,
+ &metadata,
+ cancellable,
+ error))
+ return glnx_prefix_error (error, "Failed to read detached metadata");
+
+ g_autoptr(GBytes) signed_data = g_variant_get_data_as_bytes (commit_variant);
+
+ /* XXX This is a hackish way to indicate to use ALL remote-specific
+ * keyrings in the signature verification. We want this when
+ * verifying a signed commit that's already been pulled. */
+/*
+ if (remote_name == NULL)
+ remote_name = OSTREE_ALL_REMOTES;
+*/
+
+ g_autoptr(GVariant) signatures = NULL;
+
+ g_autofree gchar *signature_key = ostree_sign_metadata_key(self);
+ g_autofree GVariantType *signature_format = (GVariantType *) ostree_sign_metadata_format(self);
+
+ if (metadata)
+ signatures = g_variant_lookup_value (metadata,
+ signature_key,
+ signature_format);
+
+
+ return ostree_sign_metadata_verify (self,
+ signed_data,
+ signatures,
+ error);
+}
+
+const gchar * ostree_sign_get_name (OstreeSign *self)
+{
+ g_debug ("%s enter", __FUNCTION__);
+ g_return_val_if_fail (OSTREE_IS_SIGN (self), FALSE);
+ g_return_val_if_fail (OSTREE_SIGN_GET_IFACE (self)->get_name != NULL, FALSE);
+
+ return OSTREE_SIGN_GET_IFACE (self)->get_name (self);
+}
+
+OstreeSign * ostree_sign_get_by_name (const gchar *name, GError **error)
+{
+ g_debug ("%s enter", __FUNCTION__);
+
+ GType types [] = {
+#if defined(HAVE_LIBSODIUM)
+ OSTREE_TYPE_SIGN_ED25519,
+#endif
+ OSTREE_TYPE_SIGN_DUMMY
+ };
+ OstreeSign *ret = NULL;
+
+ for (gint i=0; i < G_N_ELEMENTS(types); i++)
+ {
+ g_autoptr (OstreeSign) sign = g_object_new (types[i], NULL);
+ g_autofree gchar *sign_name = OSTREE_SIGN_GET_IFACE (sign)->get_name(sign);
+
+ g_debug ("Found '%s' signing module", sign_name);
+
+ if (g_strcmp0 (name, sign_name) == 0)
+ {
+ ret = g_steal_pointer (&sign);
+ break;
+ }
+ }
+
+ if (ret == NULL)
+ g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_FAILED,
+ "Requested signature type is not implemented");
+
+ return ret;
+}
+
+
+/**
+ * ostree_sign_commit:
+ * @self: Self
+ * @commit_checksum: SHA256 of given commit to sign
+ * @cancellable: A #GCancellable
+ * @error: a #GError
+ *
+ * Add a GPG signature to a commit.
+ */
+gboolean
+ostree_sign_commit (OstreeSign *self,
+ OstreeRepo *repo,
+ const gchar *commit_checksum,
+ GCancellable *cancellable,
+ GError **error)
+{
+ g_debug ("%s enter", __FUNCTION__);
+
+ g_autoptr(GBytes) commit_data = NULL;
+ g_autoptr(GBytes) signature = NULL;
+ g_autoptr(GVariant) commit_variant = NULL;
+ g_autoptr(GVariant) old_metadata = NULL;
+ g_autoptr(GVariant) new_metadata = NULL;
+
+ if (!ostree_repo_load_variant (repo, OSTREE_OBJECT_TYPE_COMMIT,
+ commit_checksum, &commit_variant, error))
+ return glnx_prefix_error (error, "Failed to read commit");
+
+ if (!ostree_repo_read_commit_detached_metadata (repo,
+ commit_checksum,
+ &old_metadata,
+ cancellable,
+ error))
+ return glnx_prefix_error (error, "Failed to read detached metadata");
+
+ // TODO: d4s: check if already signed?
+
+ commit_data = g_variant_get_data_as_bytes (commit_variant);
+
+ if (!ostree_sign_data (self, commit_data, &signature,
+ cancellable, error))
+ return glnx_prefix_error (error, "Not able to sign the cobject");
+
+ new_metadata =
+ ostree_sign_detached_metadata_append (self, old_metadata, signature);
+
+ if (!ostree_repo_write_commit_detached_metadata (repo,
+ commit_checksum,
+ new_metadata,
+ cancellable,
+ error))
+ return FALSE;
+
+ return TRUE;
+}
+
+GStrv ostree_sign_list_names(void)
+{
+ g_debug ("%s enter", __FUNCTION__);
+
+ GType types [] = {
+#if defined(HAVE_LIBSODIUM)
+ OSTREE_TYPE_SIGN_ED25519,
+#endif
+ OSTREE_TYPE_SIGN_DUMMY
+ };
+ GStrv names = g_new0 (char *, G_N_ELEMENTS(types)+1);
+ gint i = 0;
+
+ for (i=0; i < G_N_ELEMENTS(types); i++)
+ {
+ g_autoptr (OstreeSign) sign = g_object_new (types[i], NULL);
+ names[i] = OSTREE_SIGN_GET_IFACE (sign)->get_name(sign);
+ g_debug ("Found '%s' signing module", names[i]);
+ }
+
+ return names;
+}