Radish alpha
H
rad:z3QDZAW2FAfuLvihrhiyDC9fAD8G9
HardenedBSD Package Manager
Radicle
Git
HardenedBSD-pkg libpkg pkg_manifest.c
/*-
 * Copyright (c) 2011-2025 Baptiste Daroussin <bapt@FreeBSD.org>
 * Copyright (c) 2011-2012 Julien Laffaye <jlaffaye@FreeBSD.org>
 * Copyright (c) 2013-2014 Vsevolod Stakhov <vsevolod@FreeBSD.org>
 *
 * SPDX-License-Identifier: BSD-2-Clause
 */

#include <sys/types.h>
#include <stddef.h>

#include <assert.h>
#include <ctype.h>
#include <errno.h>
#include <stdbool.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <ucl.h>

#include "pkg.h"
#include "private/event.h"
#include "private/pkg.h"
#include "private/utils.h"

#define dbg(x, ...) pkg_dbg(PKG_DBG_MANIFEST, x, __VA_ARGS__)

enum {
	MANIFEST_ANNOTATIONS,
	MANIFEST_CATEGORIES,
	MANIFEST_CONFIG_FILES,
	MANIFEST_CONFLICTS,
	MANIFEST_DEPS,
	MANIFEST_DIRECTORIES,
	MANIFEST_DIRS,
	MANIFEST_FILES,
	MANIFEST_GROUPS,
	MANIFEST_LICENSES,
	MANIFEST_LUA_SCRIPTS,
	MANIFEST_OPTIONS,
	MANIFEST_PROVIDES,
	MANIFEST_REQUIRES,
	MANIFEST_SCRIPTS,
	MANIFEST_SHLIBS_PROVIDED,
	MANIFEST_SHLIBS_REQUIRED,
	MANIFEST_USERS,
	MANIFEST_SHLIBS_REQUIRED_IGNORE,
	MANIFEST_SHLIBS_PROVIDED_IGNORE,
};

#define PKG_MESSAGE_LEGACY	1
#define PKG_MESSAGE_NEW 2

static int pkg_string(struct pkg *, const ucl_object_t *, uint32_t);
static int pkg_obj(struct pkg *, const ucl_object_t *, uint32_t);
static int pkg_array(struct pkg *, const ucl_object_t *, uint32_t);
static int pkg_int(struct pkg *, const ucl_object_t *, uint32_t);
static int pkg_boolean(struct pkg *, const ucl_object_t *, uint32_t);
static int pkg_message(struct pkg *, const ucl_object_t *, uint32_t);
static int pkg_set_deps_from_object(struct pkg *, const ucl_object_t *);
static int pkg_set_files_from_object(struct pkg *, const ucl_object_t *);
static int pkg_set_dirs_from_object(struct pkg *, const ucl_object_t *);

/*
 * Keep sorted
 */
#define TYPE_SHIFT(x) (1 << (x))
#define STRING_FLAG_LICENSE (1U << 31)
#define STRING_FLAG_URLDECODE (1U << 30)
#define STRING_FLAG_MASK ~(STRING_FLAG_LICENSE|STRING_FLAG_URLDECODE)

static const struct pkg_manifest_key {
	const char *key;
	uint32_t type;
	uint16_t valid_type;
	int (*parse_data)(struct pkg *, const ucl_object_t *, uint32_t);
} manifest_keys[] = {
	{ "annotations",         MANIFEST_ANNOTATIONS,
			TYPE_SHIFT(UCL_OBJECT), pkg_obj},

	{ "abi",                 offsetof(struct pkg, abi),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "arch",                offsetof(struct pkg, altabi),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "categories",          MANIFEST_CATEGORIES,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "comment",             offsetof(struct pkg, comment),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "conflicts",           MANIFEST_CONFLICTS,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "config",              MANIFEST_CONFIG_FILES,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "dep_formula",         offsetof(struct pkg, dep_formula),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "deps",                MANIFEST_DEPS,
			TYPE_SHIFT(UCL_OBJECT), pkg_obj},

	{ "desc",                offsetof(struct pkg, desc) | STRING_FLAG_URLDECODE,
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "directories",         MANIFEST_DIRECTORIES,
			TYPE_SHIFT(UCL_OBJECT), pkg_obj},

	{ "dirs",                MANIFEST_DIRS,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "files",               MANIFEST_FILES,
			TYPE_SHIFT(UCL_OBJECT), pkg_obj},

	{ "flatsize",            offsetof(struct pkg, flatsize),
			TYPE_SHIFT(UCL_INT),    pkg_int},

	{ "groups",              MANIFEST_GROUPS,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "licenselogic",        offsetof(struct pkg, licenselogic) | STRING_FLAG_LICENSE,
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "licenses",            MANIFEST_LICENSES,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "lua_scripts",         MANIFEST_LUA_SCRIPTS,
			TYPE_SHIFT(UCL_OBJECT), pkg_obj},

	{ "maintainer",          offsetof(struct pkg, maintainer),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "messages",            PKG_MESSAGE_NEW,
			TYPE_SHIFT(UCL_STRING)|TYPE_SHIFT(UCL_ARRAY), pkg_message},

	{ "message",             PKG_MESSAGE_LEGACY,
			TYPE_SHIFT(UCL_STRING)|TYPE_SHIFT(UCL_ARRAY), pkg_message},

	{ "name",                offsetof(struct pkg, name),
			TYPE_SHIFT(UCL_STRING)|TYPE_SHIFT(UCL_INT), pkg_string},

	{ "options",             MANIFEST_OPTIONS,
			TYPE_SHIFT(UCL_OBJECT), pkg_obj},

	{ "origin",              offsetof(struct pkg, origin),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "path",                offsetof(struct pkg, repopath),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "repopath",            offsetof(struct pkg, repopath),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "pkgsize",             offsetof(struct pkg, pkgsize),
			TYPE_SHIFT(UCL_INT),    pkg_int},

	{ "prefix",              offsetof(struct pkg, prefix),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "provides",            MANIFEST_PROVIDES,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "requires",            MANIFEST_REQUIRES,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "scripts",             MANIFEST_SCRIPTS,
			TYPE_SHIFT(UCL_OBJECT), pkg_obj},

	{ "shlibs",              MANIFEST_SHLIBS_REQUIRED,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array}, /* Backwards compat with 1.0.x packages */

	{ "shlibs_provided",     MANIFEST_SHLIBS_PROVIDED,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "shlibs_provided_ignore",     MANIFEST_SHLIBS_PROVIDED_IGNORE,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "shlibs_required",     MANIFEST_SHLIBS_REQUIRED,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "shlibs_required_ignore",     MANIFEST_SHLIBS_REQUIRED_IGNORE,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "sum",                 offsetof(struct pkg, sum),
			TYPE_SHIFT(UCL_STRING), pkg_string},

	{ "users",               MANIFEST_USERS,
			TYPE_SHIFT(UCL_ARRAY),  pkg_array},

	{ "version",             offsetof(struct pkg, version),
			TYPE_SHIFT(UCL_STRING)|TYPE_SHIFT(UCL_INT), pkg_string},

	{ "vital",            offsetof(struct pkg, vital),
			TYPE_SHIFT(UCL_BOOLEAN),    pkg_boolean},

	{ "www",                 offsetof(struct pkg, www),
			TYPE_SHIFT(UCL_STRING), pkg_string},

};

static int
urlencode(const char *src, xstring **dest)
{
	size_t len;
	size_t i;

	xstring_renew(*dest);

	len = strlen(src);
	for (i = 0; i < len; i++) {
		if (!isascii(src[i]) || src[i] == '%')
			fprintf((*dest)->fp, "%%%.2x", (unsigned char)src[i]);
		else
			fputc(src[i], (*dest)->fp);
	}

	fflush((*dest)->fp);
	return (EPKG_OK);
}

static char*
url_decode(const char* src, size_t len)
{
	if (len == 0)
		len = strlen(src);

	char* dest = xmalloc(len + 1);
	char* p = dest;
	for (size_t i = 0; i < len; i++) {
		if (src[i] == '%' && i + 2 < len &&
		    isxdigit((unsigned char)src[i + 1]) &&
		    isxdigit((unsigned char)src[i + 2])) {
			int value;
			sscanf(src + i + 1, "%2x", &value);
			*p++ = (char)value;
			i += 2;
		} else {
			*p++ = src[i];
		}
	}
	*p = '\0';

	return (dest);
}

static int
lua_script_type_str(const char *str)
{
	if (STREQ(str, "pre-install"))
		return (PKG_LUA_PRE_INSTALL);
	if (STREQ(str, "post-install"))
		return (PKG_LUA_POST_INSTALL);
	if (STREQ(str, "pre-deinstall"))
		return (PKG_LUA_PRE_DEINSTALL);
	if (STREQ(str, "post-deinstall"))
		return (PKG_LUA_POST_DEINSTALL);
	return (PKG_LUA_UNKNOWN);
}

static int
script_type_str(const char *str)
{
	if (STREQ(str, "pre-install"))
		return (PKG_SCRIPT_PRE_INSTALL);
	if (STREQ(str, "install"))
		return (PKG_SCRIPT_INSTALL);
	if (STREQ(str, "post-install"))
		return (PKG_SCRIPT_POST_INSTALL);
	if (STREQ(str, "pre-deinstall"))
		return (PKG_SCRIPT_PRE_DEINSTALL);
	if (STREQ(str, "deinstall"))
		return (PKG_SCRIPT_DEINSTALL);
	if (STREQ(str, "post-deinstall"))
		return (PKG_SCRIPT_POST_DEINSTALL);
	return (PKG_SCRIPT_UNKNOWN);
}

static int
pkg_string(struct pkg *pkg, const ucl_object_t *obj, uint32_t offset)
{
	const char *str;
	char *tofree = NULL;
	char **dest;

	str = ucl_object_tostring_forced(obj);

	if (offset & STRING_FLAG_LICENSE) {
		if (STREQ(str, "single"))
			pkg->licenselogic = LICENSE_SINGLE;
		else if (STREQ(str, "or") ||
				STREQ(str, "dual"))
			pkg->licenselogic = LICENSE_OR;
		else if (STREQ(str, "and") ||
				STREQ(str, "multi"))
			pkg->licenselogic = LICENSE_AND;
		else {
			pkg_emit_error("Unknown license logic: %s", str);
			return (EPKG_FATAL);
		}
	}
	else {

		if (offset & STRING_FLAG_URLDECODE) {
			tofree = url_decode(str, 0);
			str = tofree;
		}

		/* Remove flags from the offset */
		offset &= STRING_FLAG_MASK;
		dest = (char **) ((unsigned char *)pkg + offset);
		*dest = xstrdup(str);

		free(tofree);
	}

	return (EPKG_OK);
}

static int
pkg_int(struct pkg *pkg, const ucl_object_t *obj, uint32_t offset)
{
	int64_t *dest;

	dest = (int64_t *)((unsigned char *)pkg + offset);
	*dest = ucl_object_toint(obj);

	return (EPKG_OK);
}

static int
pkg_boolean(struct pkg *pkg, const ucl_object_t *obj, uint32_t offset)
{
	bool *dest;

	dest = (bool *)((unsigned char *)pkg + offset);
	*dest = ucl_object_toboolean(obj);

	return (EPKG_OK);
}

static int
pkg_array(struct pkg *pkg, const ucl_object_t *obj, uint32_t attr)
{
	const ucl_object_t *cur;
	ucl_object_iter_t it = NULL;
	int ret;

	dbg(3, "%s", "parsing array");
	while ((cur = ucl_iterate_object(obj, &it, true))) {
		switch (attr) {
		case MANIFEST_CATEGORIES:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed category");
			else
				pkg_addstring(&pkg->categories,
				    ucl_object_tostring(cur), "category");
			break;
		case MANIFEST_LICENSES:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed license");
			else
				pkg_addstring(&pkg->licenses,
				    ucl_object_tostring(cur), "license");
			break;
		case MANIFEST_USERS:
			if (cur->type == UCL_STRING)
				pkg_adduser(pkg, ucl_object_tostring(cur));
			else if (cur->type == UCL_OBJECT)
				pkg_obj(pkg, cur, attr);
			else
				pkg_emit_error("Skipping malformed license");
			break;
		case MANIFEST_GROUPS:
			if (cur->type == UCL_STRING)
				pkg_addgroup(pkg, ucl_object_tostring(cur));
			else if (cur->type == UCL_OBJECT)
				pkg_obj(pkg, cur, attr);
			else
				pkg_emit_error("Skipping malformed license");
			break;
		case MANIFEST_DIRS:
			if (cur->type == UCL_STRING)
				pkg_adddir(pkg, ucl_object_tostring(cur), false);
			else if (cur->type == UCL_OBJECT)
				pkg_obj(pkg, cur, attr);
			else
				pkg_emit_error("Skipping malformed dirs");
			break;
		case MANIFEST_SHLIBS_REQUIRED:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed required shared library");
			else
				pkg_addshlib_required(pkg, ucl_object_tostring(cur), PKG_SHLIB_FLAGS_NONE);
			break;
		case MANIFEST_SHLIBS_REQUIRED_IGNORE:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed required shared library ignore");
			else
				pkg_addshlib_required_ignore(pkg, ucl_object_tostring(cur));
			break;
		case MANIFEST_SHLIBS_PROVIDED:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed provided shared library");
			else
				pkg_addshlib_provided(pkg, ucl_object_tostring(cur), PKG_SHLIB_FLAGS_NONE);
			break;
		case MANIFEST_SHLIBS_PROVIDED_IGNORE:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed provided shared library ignore");
			else
				pkg_addshlib_provided_ignore(pkg, ucl_object_tostring(cur));
			break;
		case MANIFEST_CONFLICTS:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed conflict name");
			else
				pkg_addconflict(pkg, ucl_object_tostring(cur));
			break;
		case MANIFEST_PROVIDES:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed provide name");
			else
				pkg_addprovide(pkg, ucl_object_tostring(cur));
			break;
		case MANIFEST_CONFIG_FILES:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed config file name");
			else {
				ret = pkg_addconfig_file(pkg, ucl_object_tostring(cur), NULL);
				if (ret != EPKG_OK)
					return (ret);
			}
			break;
		case MANIFEST_REQUIRES:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed require name");
			else
				pkg_addrequire(pkg, ucl_object_tostring(cur));
			break;
		}
	}

	return (EPKG_OK);
}

static int
pkg_obj(struct pkg *pkg, const ucl_object_t *obj, uint32_t attr)
{
	const ucl_object_t *cur;
	ucl_object_iter_t it = NULL;
	pkg_script script_type;
	pkg_lua_script lua_script_type;
	const char *key, *buf;
	size_t len;

	dbg(3, "%s", "parsing object");
	while ((cur = ucl_iterate_object(obj, &it, true))) {
		key = ucl_object_key(cur);
		if (key == NULL)
			continue;
		switch (attr) {
		case MANIFEST_DEPS:
			if (cur->type != UCL_OBJECT && cur->type != UCL_ARRAY)
				pkg_emit_error("Skipping malformed dependency %s",
				    key);
			else
				pkg_set_deps_from_object(pkg, cur);
			break;
		case MANIFEST_DIRS:
			if (cur->type != UCL_OBJECT)
				pkg_emit_error("Skipping malformed dirs %s",
				    key);
			else
				pkg_set_dirs_from_object(pkg, cur);
			break;
		case MANIFEST_DIRECTORIES:
			if (cur->type == UCL_BOOLEAN) {
				char *t = url_decode(key, 0);
				pkg_adddir(pkg, t, false);
				free(t);
			} else if (cur->type == UCL_OBJECT) {
				pkg_set_dirs_from_object(pkg, cur);
			} else if (cur->type == UCL_STRING) {
				char *t = url_decode(key, 0);
				pkg_adddir(pkg, t, false);
				free(t);
			} else {
				pkg_emit_error("Skipping malformed directories %s",
				    key);
			}
			break;
		case MANIFEST_FILES:
			if (cur->type == UCL_STRING) {
				char *t = url_decode(key, 0);
				buf = ucl_object_tolstring(cur, &len);
				pkg_addfile(pkg, t, len >= 2 ? buf : NULL, false);
				free(t);
			} else if (cur->type == UCL_OBJECT)
				pkg_set_files_from_object(pkg, cur);
			else
				pkg_emit_error("Skipping malformed files %s",
				   key);
			break;
		case MANIFEST_OPTIONS:
			if (cur->type == UCL_STRING)
				pkg_addoption(pkg, key, ucl_object_tostring(cur));
			else if (cur->type == UCL_BOOLEAN)
				pkg_addoption(pkg, key, ucl_object_toboolean(cur) ? "on" : "off");
			else
				pkg_emit_error("Skipping malformed option %s", key);
			break;
		case MANIFEST_SCRIPTS:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed scripts %s",
				    key);
			else {
				script_type = script_type_str(key);
				if (script_type == PKG_SCRIPT_UNKNOWN) {
					pkg_emit_error("Skipping unknown script "
					    "type: %s", key);
					break;
				}

				char *t = url_decode(ucl_object_tostring(cur), 0);
				pkg_addscript(pkg, t, script_type);
				free(t);
			}
			break;
		case MANIFEST_LUA_SCRIPTS:
			if (cur->type != UCL_ARRAY) {
				pkg_emit_error("Skipping malformed dependency %s",
				    key);
				break;
			}
			lua_script_type = lua_script_type_str(key);
			if (lua_script_type == PKG_LUA_UNKNOWN) {
				pkg_emit_error("Skipping unknown script "
				    "type: %s", key);
				break;
			}
			pkg_lua_script_from_ucl(pkg, cur, lua_script_type);
			break;
		case MANIFEST_ANNOTATIONS:
			if (cur->type != UCL_STRING)
				pkg_emit_error("Skipping malformed annotation %s",
				    key);
			else
				pkg_kv_add(&pkg->annotations, key, ucl_object_tostring(cur), "annotation");
			break;
		}
	}

	return (EPKG_OK);
}

static int
pkg_message(struct pkg *pkg, const ucl_object_t *obj, uint32_t attr __unused)
{
	return pkg_message_from_ucl(pkg, obj);
}

static int
pkg_set_files_from_object(struct pkg *pkg, const ucl_object_t *obj)
{
	const ucl_object_t *cur;
	ucl_object_iter_t it = NULL;
	const char *sum = NULL;
	const char *uname = NULL;
	const char *gname = NULL;
	const char *symlink_target = NULL;
	mode_t perm = 0;
	u_long fflags = 0;
	char *fname = NULL;
	const char *key, *okey;
	time_t mtime = 0;

	okey = ucl_object_key(obj);
	if (okey == NULL)
		return (EPKG_FATAL);
	fname = url_decode(okey, 0);
	while ((cur = ucl_iterate_object(obj, &it, true))) {
		key = ucl_object_key(cur);
		if (key == NULL)
			continue;
		if (STRIEQ(key, "uname") && cur->type == UCL_STRING)
			uname = ucl_object_tostring(cur);
		else if (STRIEQ(key, "gname") && cur->type == UCL_STRING)
			gname = ucl_object_tostring(cur);
		else if (STRIEQ(key, "sum") && cur->type == UCL_STRING)
			sum = ucl_object_tostring(cur);
		else if (STRIEQ(key, "perm") &&
			 (cur->type == UCL_STRING || cur->type == UCL_INT)) {
			void *set = 0;
			if ((set = setmode(ucl_object_tostring_forced(cur))) == NULL)
				pkg_emit_error("Not a valid mode: %s",
				    ucl_object_tostring(cur));
			else
				perm = getmode(set, 0);
			free(set);
		} else if (STRIEQ(key, "fflags") && cur->type == UCL_INT) {
			fflags = (u_long)ucl_object_toint(cur);
#ifdef HAVE_STRTOFFLAGS
		} else if (STRIEQ(key, "fflags") && cur->type == UCL_STRING) {
			char *str_flags = (char *)ucl_object_tostring(cur);
			u_long clearp;

			if (strtofflags(&str_flags, &fflags, &clearp) != 0 ||
			    clearp != 0) {
				fflags = 0;
				pkg_emit_error("Not valid file flags: %s", str_flags);
				continue;
			}
#endif
		} else if (STRIEQ(key, "symlink_target") && cur->type == UCL_STRING) {
			symlink_target = ucl_object_tostring(cur);
		} else if (STRIEQ(key, "mtime") &&
			   (cur->type == UCL_STRING || cur->type == UCL_INT)) {
			errno = 0;
			mtime = strtoll(ucl_object_tostring_forced(cur), NULL, 10);
			if (mtime == 0 && errno != 0) {
				pkg_emit_errno("strtoll: invalid epoch value",
					       ucl_object_tostring_forced(cur));
				continue;
			}
		} else {
			dbg(1, "Skipping unknown key for file(%s): %s",
			    fname, key);
		}
	}

	pkg_addfile_attr(pkg, fname, sum, uname, gname, perm, fflags,
			 mtime, symlink_target, false);
	free(fname);

	return (EPKG_OK);
}

static int
pkg_set_dirs_from_object(struct pkg *pkg, const ucl_object_t *obj)
{
	const ucl_object_t *cur;
	ucl_object_iter_t it = NULL;
	const char *uname = NULL;
	const char *gname = NULL;
	mode_t perm = 0;
	u_long fflags = 0;
	char *dirname = NULL;
	const char *key, *okey;

	okey = ucl_object_key(obj);
	if (okey == NULL)
		return (EPKG_FATAL);
	dirname = url_decode(okey, 0);
	while ((cur = ucl_iterate_object(obj, &it, true))) {
		key = ucl_object_key(cur);
		if (key == NULL)
			continue;
		if (STRIEQ(key, "uname") && cur->type == UCL_STRING)
			uname = ucl_object_tostring(cur);
		else if (STRIEQ(key, "gname") && cur->type == UCL_STRING)
			gname = ucl_object_tostring(cur);
		else if (STRIEQ(key, "perm") &&
			 (cur->type == UCL_STRING || cur->type == UCL_INT)) {
			void *set = 0;
			if ((set = setmode(ucl_object_tostring_forced(cur))) == NULL)
				pkg_emit_error("Not a valid mode: %s",
				    ucl_object_tostring(cur));
			else
				perm = getmode(set, 0);
			free(set);
		} else if (STRIEQ(key, "fflags") && cur->type == UCL_INT) {
			fflags = (u_long)ucl_object_toint(cur);
#ifdef HAVE_STRTOFFLAGS
		} else if (STRIEQ(key, "fflags") && cur->type == UCL_STRING) {
			char *str_flags = (char *)ucl_object_tostring(cur);
			u_long clearp;

			if (strtofflags(&str_flags, &fflags, &clearp) != 0 ||
			    clearp != 0) {
				fflags = 0;
				pkg_emit_error("Not valid file flags: %s", str_flags);
				continue;
			}
#endif
		} else if (STRIEQ(key, "try") && cur->type == UCL_BOOLEAN) {
			/* ignore on purpose : compatibility*/
		} else {
			dbg(1, "Skipping unknown key for dir(%s): %s",
			    dirname, key);
		}
	}

	pkg_adddir_attr(pkg, dirname, uname, gname, perm, fflags, false);
	free(dirname);

	return (EPKG_OK);
}

static int
pkg_set_deps_from_object(struct pkg *pkg, const ucl_object_t *obj)
{
	const ucl_object_t *cur, *self;
	ucl_object_iter_t it = NULL, it2;
	const char *origin = NULL;
	const char *version = NULL;
	const char *key, *okey;

	okey = ucl_object_key(obj);
	if (okey == NULL)
		return (EPKG_FATAL);
	dbg(2, "Found %s", okey);
	while ((self = ucl_iterate_object(obj, &it, (obj->type == UCL_ARRAY)))) {
		it2 = NULL;
		while ((cur = ucl_iterate_object(self, &it2, true))) {
			key = ucl_object_key(cur);
			if (key == NULL)
				continue;
			if (cur->type != UCL_STRING) {
				/* accept version to be an integer */
				if (cur->type == UCL_INT && STRIEQ(key, "version")) {
					if (!ctx.no_version_for_deps)
						version = ucl_object_tostring_forced(cur);
					continue;
				}

				pkg_emit_error("Skipping malformed dependency entry "
						"for %s", okey);
				continue;
			}
			if (STRIEQ(key, "origin"))
				origin = ucl_object_tostring(cur);
			if (STRIEQ(key, "version") && !ctx.no_version_for_deps)
				version = ucl_object_tostring(cur);
		}
		if (origin != NULL)
			pkg_adddep(pkg, okey, origin, version, false);
		else
			pkg_emit_error("Skipping malformed dependency %s", okey);
	}

	return (EPKG_OK);
}

static const struct pkg_manifest_key *
select_manifest_key(const char *key)
{
	for (int i = 0; i < NELEM(manifest_keys); i++)
		if (STREQ(manifest_keys[i].key, key))
			return (&(manifest_keys[i]));
	return (NULL);
}
static int
parse_manifest(struct pkg *pkg, ucl_object_t *obj)
{
	const ucl_object_t *cur;
	ucl_object_iter_t it = NULL;
	const struct pkg_manifest_key *sk;
	const char *key;
	int ret = EPKG_OK;

	while ((cur = ucl_iterate_object(obj, &it, true))) {
		key = ucl_object_key(cur);
		if (key == NULL)
			continue;
		dbg(3, "found key: '%s'", key);
		if ((sk = select_manifest_key(key)) == NULL) {
			dbg(1, "Skipping unknown key '%s'", key);
			continue;
		}
		if (TYPE_SHIFT(ucl_object_type(cur)) & sk->valid_type) {
			ret = sk->parse_data(pkg, cur, sk->type);
			if (ret != EPKG_OK)
				return (ret);
		} else {
			pkg_emit_error("Skipping malformed key '%s'", key);
		}
	}

	return (EPKG_OK);
}

int
pkg_parse_manifest_ucl(struct pkg *pkg, ucl_object_t *obj)
{
	const ucl_object_t *cur;
	ucl_object_iter_t it = NULL;
	const struct pkg_manifest_key *sk;
	const char *key;

	/* do a minimal validation */
	while ((cur = ucl_iterate_object(obj, &it, true))) {
		key = ucl_object_key(cur);
		if (key == NULL)
			continue;
		if ((sk = select_manifest_key(key)) == NULL)
			continue;
		if (!(sk->valid_type & TYPE_SHIFT(ucl_object_type(cur)))) {
			pkg_emit_error("Bad format in manifest for key:"
				" %s", key);
			UCL_FREE (sizeof (*it), it);
			return (EPKG_FATAL);
		}
	}

	return (parse_manifest(pkg, obj));
}

int
pkg_parse_manifest(struct pkg *pkg, const char *buf, size_t len)
{
	struct ucl_parser *p = NULL;
	ucl_object_t *obj = NULL;
	int rc;

	assert(pkg != NULL);
	assert(buf != NULL);

	dbg(2, "%s", "Parsing from buffer");

	p = ucl_parser_new(UCL_PARSER_NO_FILEVARS);
	if (!ucl_parser_add_chunk(p, buf, len)) {
		pkg_emit_error("Error parsing manifest: %s",
		    ucl_parser_get_error(p));
		ucl_parser_free(p);

		return (EPKG_FATAL);
	}

	if ((obj = ucl_parser_get_object(p)) == NULL) {
		ucl_parser_free(p);
		return (EPKG_FATAL);
	}

	ucl_parser_free(p);
	rc = pkg_parse_manifest_ucl(pkg, obj);
	ucl_object_unref(obj);

	return (rc);
}

int
pkg_parse_manifest_fileat(int dfd, struct pkg *pkg, const char *file)
{
	struct ucl_parser *p = NULL;
	ucl_object_t *obj = NULL;
	int rc;
	char *data;
	off_t sz = 0;

	assert(pkg != NULL);
	assert(file != NULL);

	dbg(1, "Parsing from '%s'", file);

	errno = 0;

	if (file_to_bufferat(dfd, file, &data, &sz) != EPKG_OK)
		return (EPKG_FATAL);

	p = ucl_parser_new(UCL_PARSER_NO_FILEVARS);
	if (!ucl_parser_add_string(p, data, sz)) {
		pkg_emit_error("manifest parsing error: %s", ucl_parser_get_error(p));
		ucl_parser_free(p);
		free(data);
		return (EPKG_FATAL);
	}

	if ((obj = ucl_parser_get_object(p)) == NULL) {
		ucl_parser_free(p);
		free(data);
		return (EPKG_FATAL);
	}
	ucl_parser_free(p);

	rc = pkg_parse_manifest_ucl(pkg, obj);
	ucl_object_unref(obj);
	free(data);

	return (rc);
}

int
pkg_parse_manifest_file(struct pkg *pkg, const char *file)
{
	return pkg_parse_manifest_fileat(AT_FDCWD, pkg, file);
}

#define MANIFEST_EXPORT_FIELD(result, o, field, type)	do { \
	ucl_object_insert_key((result), ucl_object_from ## type (o->field), #field, 0, false); \
	} while (0)

int
pkg_emit_filelist(struct pkg *pkg, FILE *f, pkghash **dirs, int *ndirs)
{
	struct pkg_file *file = NULL;
	int cur_idx = -1;

	fprintf(f, "%s %s\n", pkg->name, pkg->version);

	while (pkg_files(pkg, &file) == EPKG_OK) {
		char dpath[MAXPATHLEN];
		const char *dp = file->path;
		const char *last_slash, *base;
		size_t dir_len;
		char dirbuf[MAXPATHLEN];
		int idx;
		pkghash_entry *e;

		if (pkg->oprefix != NULL) {
			size_t l = strlen(pkg->prefix);
			if (strncmp(file->path, pkg->prefix, l) == 0 &&
			    (file->path[l] == '/' || l == 1)) {
				snprintf(dpath, sizeof(dpath), "%s%s%s",
				    pkg->oprefix, l == 1 ? "/" : "", file->path + l);
				dp = dpath;
			}
		}

		last_slash = strrchr(dp, '/');
		if (last_slash != NULL) {
			dir_len = last_slash - dp;
			if (dir_len == 0)
				dir_len = 1;
			base = last_slash + 1;
		} else {
			dir_len = 0;
			base = dp;
		}

		if (dir_len >= sizeof(dirbuf))
			continue;
		memcpy(dirbuf, dp, dir_len);
		dirbuf[dir_len] = '\0';

		/* Get or assign directory index */
		e = pkghash_get(*dirs, dirbuf);
		if (e != NULL) {
			idx = (int)(intptr_t)e->value;
		} else {
			idx = (*ndirs)++;
			if (*dirs == NULL)
				*dirs = pkghash_new();
			pkghash_add(*dirs, dirbuf, (void *)(intptr_t)idx, NULL);
		}

		if (idx != cur_idx) {
			fprintf(f, ">%d\n", idx);
			cur_idx = idx;
		}
		fprintf(f, "%s\n", base);
	}

	fprintf(f, "\n");

	return (EPKG_OK);
}

pkg_object*
pkg_emit_object(struct pkg *pkg, short flags)
{
	struct pkg_kv		*kv;
	struct pkg_dep		*dep      = NULL;
	struct pkg_file		*file     = NULL;
	struct pkg_dir		*dir      = NULL;
	struct pkg_conflict	*conflict = NULL;
	struct pkg_config_file	*cf       = NULL;
	xstring		*tmpsbuf  = NULL;
	const char *script_types = NULL;
	char legacyarch[BUFSIZ];
	char perm_str[sizeof("00000")];
	ucl_object_t *map, *seq, *submap;
	ucl_object_t *top = ucl_object_typed_new(UCL_OBJECT);

	if (pkg->abi == NULL && pkg->altabi != NULL)
		pkg->abi = xstrdup(pkg->altabi);
	pkg_arch_to_legacy(pkg->abi, legacyarch, BUFSIZ);
	free(pkg->altabi);
	pkg->altabi = xstrdup(legacyarch);
	pkg_lists_sort(pkg);
	dbg(4, "Emitting basic metadata");
	MANIFEST_EXPORT_FIELD(top, pkg, name, string);
	MANIFEST_EXPORT_FIELD(top, pkg, origin, string);
	MANIFEST_EXPORT_FIELD(top, pkg, version, string);
	MANIFEST_EXPORT_FIELD(top, pkg, comment, string);
	MANIFEST_EXPORT_FIELD(top, pkg, maintainer, string);
	MANIFEST_EXPORT_FIELD(top, pkg, www, string);
	MANIFEST_EXPORT_FIELD(top, pkg, abi, string);
	/* We need to keep altabi named arch in the manifest */
	ucl_object_insert_key(top, ucl_object_fromstring(pkg->altabi), "arch", 0, false);
	MANIFEST_EXPORT_FIELD(top, pkg, prefix, string);
	MANIFEST_EXPORT_FIELD(top, pkg, sum, string);
	MANIFEST_EXPORT_FIELD(top, pkg, flatsize, int);
	if (pkg->dep_formula != NULL)
		MANIFEST_EXPORT_FIELD(top, pkg, dep_formula, string);
	if (pkg->type == PKG_INSTALLED &&
	    (flags & PKG_MANIFEST_EMIT_LOCAL_METADATA) == PKG_MANIFEST_EMIT_LOCAL_METADATA) {
		MANIFEST_EXPORT_FIELD(top, pkg, timestamp, int);
	}

	/*
	 * XXX: dirty hack to be compatible with pkg 1.2
	 */
	if (pkg->repopath) {
		ucl_object_insert_key(top,
			ucl_object_fromstring(pkg->repopath), "path", 4, false);
		MANIFEST_EXPORT_FIELD(top, pkg, repopath, string);
	}

	switch (pkg->licenselogic) {
	case LICENSE_SINGLE:
		ucl_object_insert_key(top, ucl_object_fromlstring("single", 6), "licenselogic", 12, false);
		break;
	case LICENSE_AND:
		ucl_object_insert_key(top, ucl_object_fromlstring("and", 3), "licenselogic", 12, false);
		break;
	case LICENSE_OR:
		ucl_object_insert_key(top, ucl_object_fromlstring("or", 2), "licenselogic", 12, false);
		break;
	}

	dbg(4, "Emitting licenses");
	seq = NULL;
	vec_foreach(pkg->licenses, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->licenses.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "licenses", 8, false);

	if (pkg->pkgsize > 0)
		MANIFEST_EXPORT_FIELD(top, pkg, pkgsize, int);
	if (pkg->vital)
		MANIFEST_EXPORT_FIELD(top, pkg, vital, bool);

	if (pkg->desc != NULL) {
		urlencode(pkg->desc, &tmpsbuf);
		ucl_object_insert_key(top,
			ucl_object_fromstring_common(tmpsbuf->buf, strlen(tmpsbuf->buf), UCL_STRING_TRIM),
			"desc", 4, false);
	}

	dbg(4, "Emitting deps");
	map = NULL;
	while (pkg_deps(pkg, &dep) == EPKG_OK) {
		submap = ucl_object_typed_new(UCL_OBJECT);
		MANIFEST_EXPORT_FIELD(submap, dep, origin, string);
		MANIFEST_EXPORT_FIELD(submap, dep, version, string);
		if (map == NULL)
			map = ucl_object_typed_new(UCL_OBJECT);
		ucl_object_insert_key(map, submap, dep->name, 0, false);
	}
	if (map)
		ucl_object_insert_key(top, map, "deps", 4, false);

	dbg(4, "Emitting categories");
	seq = NULL;
	vec_foreach(pkg->categories, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->categories.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "categories", 10, false);

	dbg(4, "Emitting users");
	seq = NULL;
	vec_foreach(pkg->users, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->users.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "users", 5, false);

	dbg(4, "Emitting groups");
	seq = NULL;
	vec_foreach(pkg->groups, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->groups.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "groups", 6, false);

	dbg(4, "Emitting shibs_required");
	seq = NULL;
	vec_foreach(pkg->shlibs_required, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->shlibs_required.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "shlibs_required", 15, false);

	dbg(4, "Emitting shibs_required_ignore");
	seq = NULL;
	vec_foreach(pkg->shlibs_required_ignore, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->shlibs_required_ignore.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "shlibs_required_ignore", 22, false);

	dbg(4, "Emitting shlibs_provided");
	seq = NULL;
	vec_foreach(pkg->shlibs_provided, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->shlibs_provided.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "shlibs_provided", 15, false);

	dbg(4, "Emitting shlibs_provided_ignore");
	seq = NULL;
	vec_foreach(pkg->shlibs_provided_ignore, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->shlibs_provided_ignore.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "shlibs_provided_ignore", 22, false);

	dbg(4, "Emitting conflicts");
	seq = NULL;
	while (pkg_conflicts(pkg, &conflict) == EPKG_OK) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(conflict->uid));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "conflicts", 9, false);

	dbg(4, "Emitting provides");
	seq = NULL;
	vec_foreach(pkg->provides, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->provides.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "provides", 8, false);

	dbg(4, "Emitting requires");
	seq = NULL;
	vec_foreach(pkg->requires, i) {
		if (seq == NULL)
			seq = ucl_object_typed_new(UCL_ARRAY);
		ucl_array_append(seq, ucl_object_fromstring(pkg->requires.d[i]));
	}
	if (seq)
		ucl_object_insert_key(top, seq, "requires", 8, false);

	dbg(4, "Emitting options");
	map = NULL;
	vec_foreach(pkg->options, oi) {
		struct pkg_kv *opt = pkg->options.d[oi];
		dbg(4, "Emitting option: %s", opt->value);
		if (map == NULL)
			map = ucl_object_typed_new(UCL_OBJECT);
		ucl_object_insert_key(map,
		    ucl_object_fromstring(opt->value),
		    opt->key, 0, false);
	}
	if (map)
		ucl_object_insert_key(top, map, "options", 7, false);

	map = NULL;
	vec_foreach(pkg->annotations, i) {
		kv = pkg->annotations.d[i];
		if (map == NULL)
			map = ucl_object_typed_new(UCL_OBJECT);
		/* Add annotations except for internal ones. */
		if ((STREQ(kv->key, "repository") ||
		     STREQ(kv->key, "relocated")) &&
		    (flags & PKG_MANIFEST_EMIT_LOCAL_METADATA) == 0)
			continue;
		ucl_object_insert_key(map, ucl_object_fromstring(kv->value),
		    kv->key, strlen(kv->key), true);
	}
	if (map)
		ucl_object_insert_key(top, map, "annotations", 11, false);

	if ((flags & PKG_MANIFEST_EMIT_COMPACT) == 0) {
		if ((flags & PKG_MANIFEST_EMIT_NOFILES) == 0) {
			dbg(4, "Emitting files");
			map = NULL;
			while (pkg_files(pkg, &file) == EPKG_OK) {
				char dpath[MAXPATHLEN];
				const char *dp = file->path;
				ucl_object_t *file_attrs;

				if (pkg->oprefix != NULL) {
					size_t l = strlen(pkg->prefix);
					if (strncmp(file->path, pkg->prefix, l) == 0 &&
							(file->path[l] == '/' || l == 1)) {
						snprintf(dpath, sizeof(dpath), "%s%s%s",
								pkg->oprefix, l == 1 ? "/" : "", file->path + l);
						dp = dpath;
					}
				}
				if (file->sum == NULL)
					file->sum = xstrdup("-");

				file_attrs = ucl_object_typed_new(UCL_OBJECT);
				ucl_object_insert_key(file_attrs,
						      ucl_object_fromstring(file->sum),
						      "sum", 0, false);
				ucl_object_insert_key(file_attrs,
						      ucl_object_fromstring(file->uname != NULL ? file->uname : "root"),
						      "uname", 0, false);
				ucl_object_insert_key(file_attrs,
						      ucl_object_fromstring(file->gname != NULL ? file->gname : "wheel"),
						      "gname", 0, false);
				snprintf(perm_str, sizeof(perm_str), "%#4.4o", file->perm);
				ucl_object_insert_key(file_attrs,
						      ucl_object_fromstring(perm_str),
						      "perm", 0, false);
#ifdef HAVE_FFLAGSTOSTR
				if (file->fflags != 0) {
					char *fflags_str = fflagstostr(file->fflags);
					if (fflags_str != NULL && *fflags_str != '\0') {
						ucl_object_insert_key(file_attrs,
						    ucl_object_fromstring(fflags_str),
						    "fflags", 0, false);
					}
					free(fflags_str);
				}
#else
				if (file->fflags != 0) {
					ucl_object_insert_key(file_attrs,
					    ucl_object_fromint(file->fflags),
					    "fflags", 0, false);
				}
#endif
				if (file->symlink_target != NULL) {
					ucl_object_insert_key(file_attrs,
							      ucl_object_fromstring(file->symlink_target),
							      "symlink_target", 0, false);
				}
				if (file->time[1].tv_sec > 0) {
					ucl_object_insert_key(file_attrs,
							      ucl_object_fromint(file->time[1].tv_sec),
							      "mtime", 0, false);
				}

				urlencode(dp, &tmpsbuf);
				if (map == NULL)
					map = ucl_object_typed_new(UCL_OBJECT);
				ucl_object_insert_key(map, file_attrs,
						      tmpsbuf->buf, 0, true);
			}
			if (map)
				ucl_object_insert_key(top, map, "files", 5, false);

			dbg(4, "Emitting config files");
			seq = NULL;
			while (pkg_config_files(pkg, &cf) == EPKG_OK) {
				urlencode(cf->path, &tmpsbuf);
				if (seq == NULL)
					seq = ucl_object_typed_new(UCL_ARRAY);
				ucl_array_append(seq, ucl_object_fromstring(tmpsbuf->buf));
			}
			if (seq)
				ucl_object_insert_key(top, seq, "config", 6, false);

			dbg(4, "Emitting directories");
			map = NULL;
			while (pkg_dirs(pkg, &dir) == EPKG_OK) {
				ucl_object_t *dir_attrs;

				dir_attrs = ucl_object_typed_new(UCL_OBJECT);
				ucl_object_insert_key(dir_attrs,
						      ucl_object_fromstring(dir->uname ? dir->uname : "root"),
						      "uname", 0, false);
				ucl_object_insert_key(dir_attrs,
						      ucl_object_fromstring(dir->gname ? dir->gname : "wheel"),
						      "gname", 0, false);
				snprintf(perm_str, sizeof(perm_str), "%#4.4o", dir->perm);
				ucl_object_insert_key(dir_attrs,
						      ucl_object_fromstring(perm_str),
						      "perm", 0, false);
#ifdef HAVE_FFLAGSTOSTR
				if (dir->fflags != 0) {
					char *fflags_str = fflagstostr(dir->fflags);
					if (fflags_str != NULL && *fflags_str != '\0') {
						ucl_object_insert_key(dir_attrs,
						    ucl_object_fromstring(fflags_str),
						    "fflags", 0, false);
					}
					free(fflags_str);
				}
#else
				if (dir->fflags != 0) {
					ucl_object_insert_key(dir_attrs,
					    ucl_object_fromint(dir->fflags),
					    "fflags", 0, false);
				}
#endif

				urlencode(dir->path, &tmpsbuf);
				if (map == NULL)
					map = ucl_object_typed_new(UCL_OBJECT);
				ucl_object_insert_key(map, dir_attrs,
						      tmpsbuf->buf, strlen(tmpsbuf->buf), true);
			}
			if (map)
				ucl_object_insert_key(top, map, "directories", 11, false);
		}

		dbg(4, "Emitting scripts");
		map = NULL;
		for (int i = 0; i < PKG_NUM_SCRIPTS; i++) {
			if (pkg_script_get(pkg, i) == NULL)
				continue;

			switch (i) {
			case PKG_SCRIPT_PRE_INSTALL:
				script_types = "pre-install";
				break;
			case PKG_SCRIPT_INSTALL:
				script_types = "install";
				break;
			case PKG_SCRIPT_POST_INSTALL:
				script_types = "post-install";
				break;
			case PKG_SCRIPT_PRE_DEINSTALL:
				script_types = "pre-deinstall";
				break;
			case PKG_SCRIPT_DEINSTALL:
				script_types = "deinstall";
				break;
			case PKG_SCRIPT_POST_DEINSTALL:
				script_types = "post-deinstall";
				break;
			}
			urlencode(pkg_script_get(pkg, i), &tmpsbuf);
			if (map == NULL)
				map = ucl_object_typed_new(UCL_OBJECT);
			ucl_object_insert_key(map,
			    ucl_object_fromstring_common(tmpsbuf->buf,
			        strlen(tmpsbuf->buf), UCL_STRING_TRIM),
			    script_types, 0, true);
		}
		if (map)
			ucl_object_insert_key(top, map, "scripts", 7, false);

		dbg(4, "Emitting lua scripts");
		map = NULL;
		for (int i = 0; i < PKG_NUM_LUA_SCRIPTS; i++) {
			if (vec_len(&pkg->lua_scripts[i]) == 0)
				continue;
			switch(i) {
			case PKG_LUA_PRE_INSTALL:
				script_types = "pre-install";
				break;
			case PKG_LUA_POST_INSTALL:
				script_types = "post-install";
				break;
			case PKG_LUA_PRE_DEINSTALL:
				script_types = "pre-deinstall";
				break;
			case PKG_LUA_POST_DEINSTALL:
				script_types = "post-deinstall";
				break;
			}
			if (map == NULL)
				map = ucl_object_typed_new(UCL_OBJECT);
			ucl_object_insert_key(map,
			    pkg_lua_script_to_ucl(&pkg->lua_scripts[i]),
				    script_types, 0, true);
		}
		if (map)
			ucl_object_insert_key(top, map, "lua_scripts", 11, false);
	}

	dbg(4, "Emitting message");
	if (pkg_has_message(pkg))  {
		ucl_object_insert_key(top,
			pkg_message_to_ucl(pkg),
			"messages", sizeof("messages") - 1, false);
	}

	xstring_free(tmpsbuf);

	return (top);
}

int
pkg_emit_manifest_file(struct pkg *pkg, FILE *out, short flags)
{
	ucl_object_t *top;

	top = pkg_emit_object(pkg, flags);

	if ((flags & PKG_MANIFEST_EMIT_PRETTY) == PKG_MANIFEST_EMIT_PRETTY)
		ucl_object_emit_file(top, UCL_EMIT_YAML, out);
	else if ((flags & PKG_MANIFEST_EMIT_UCL) == PKG_MANIFEST_EMIT_UCL)
		ucl_object_emit_file(top, UCL_EMIT_CONFIG, out);
	else if ((flags & PKG_MANIFEST_EMIT_JSON) == PKG_MANIFEST_EMIT_JSON)
		ucl_object_emit_file(top, UCL_EMIT_JSON, out);
	else
		ucl_object_emit_file(top, UCL_EMIT_JSON_COMPACT, out);

	ucl_object_unref(top);

	return (EPKG_OK);
}