Radish alpha
H
rad:z3QDZAW2FAfuLvihrhiyDC9fAD8G9
HardenedBSD Package Manager
Radicle
Git
HardenedBSD-pkg libpkg pkg_key.c
/*-
 * Copyright (c) 2021 Kyle Evans <kevans@FreeBSD.org>
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer
 *    in this position and unchanged.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
 * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

#include <assert.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>

#include "pkg.h"
#include "xmalloc.h"
#include "private/pkg.h"
#include "private/pkgsign.h"

int
pkg_key_new(struct pkg_key **key, const char *keytype, const char *keypath,
    pkg_password_cb *cb)
{
	struct pkg_key *nkey;
	struct pkgsign_ctx *nctx = NULL;
	int ret;

	assert(*key == NULL);
	assert(keytype != NULL);	/* XXX for now. */
	if (*keypath == '\0')
		return (EPKG_FATAL);

	ret = pkgsign_new_sign(keytype, &nctx);
	if (ret != 0)
		return (EPKG_FATAL);

	pkgsign_set(nctx, cb, keypath);

	nkey = xcalloc(1, sizeof(*nkey));
	nkey->ctx = nctx;

	*key = nkey;
	return (EPKG_OK);
}

void
pkg_key_free(struct pkg_key *key)
{

	pkgsign_free(key->ctx);
	free(key);
}

/*
 * Key generation callbacks may take any number of options, so we handle those
 * with an iovec.  The pkg_key layer does not discriminate, beyond enforcing
 * that options come in pairs.  The intention is that the first option in every
 * pair names the option.
 */
int
pkg_key_create(struct pkg_key *key, const struct iovec *iov, int niov)
{

	/* Malformed arguments; must come in pairs. */
	if ((niov % 2) != 0)
		return (EPKG_FATAL);

	return (pkgsign_generate(key->ctx, iov, niov));
}

int
pkg_key_sign_data(struct pkg_key *key, const unsigned char *msg, size_t msgsz,
    unsigned char **sig, size_t *siglen)
{

	return (pkgsign_sign_data(key->ctx, msg, msgsz, sig, siglen));
}

int
pkg_key_info(struct pkg_key *key, struct iovec **iov, int *niov)
{
	int rc;
	struct iovec *kiov;
	int nkiov;

	kiov = NULL;
	rc = pkgsign_keyinfo(key->ctx, &kiov, &nkiov);
	if (rc != EPKG_OK)
		return (rc);
	if ((nkiov % 2) != 0) {
		free(kiov);
		return (EPKG_FATAL);
	}

	*iov = kiov;
	*niov = nkiov;

	return (EPKG_OK);
}

int
pkg_key_pubkey(struct pkg_key *key, char **pubkey, size_t *len)
{

	return (pkgsign_pubkey(key->ctx, pubkey, len));
}