Blame src/xb-silo-export.c

Packit caecb6
/*
Packit caecb6
 * Copyright (C) 2018 Richard Hughes <richard@hughsie.com>
Packit caecb6
 *
Packit caecb6
 * SPDX-License-Identifier: LGPL-2.1+
Packit caecb6
 */
Packit caecb6
Packit caecb6
#define G_LOG_DOMAIN				"XbSilo"
Packit caecb6
Packit caecb6
#include "config.h"
Packit caecb6
Packit caecb6
#include <gio/gio.h>
Packit caecb6
Packit caecb6
#include "xb-node-private.h"
Packit caecb6
#include "xb-silo-export-private.h"
Packit caecb6
#include "xb-silo-private.h"
Packit caecb6
#include "xb-string-private.h"
Packit caecb6
Packit caecb6
typedef struct {
Packit caecb6
	GString			*xml;
Packit caecb6
	XbNodeExportFlags	 flags;
Packit caecb6
	guint32			 off;
Packit caecb6
	guint			 level;
Packit caecb6
} XbSiloExportHelper;
Packit caecb6
Packit caecb6
static gboolean
Packit caecb6
xb_silo_export_node (XbSilo *self, XbSiloExportHelper *helper, XbSiloNode *sn, GError **error)
Packit caecb6
{
Packit caecb6
	XbSiloNode *sn2;
Packit caecb6
Packit caecb6
	helper->off = xb_silo_get_offset_for_node (self, sn);
Packit caecb6
Packit caecb6
	/* add start of opening tag */
Packit caecb6
	if (helper->flags & XB_NODE_EXPORT_FLAG_FORMAT_INDENT) {
Packit caecb6
		for (guint i = 0; i < helper->level; i++)
Packit caecb6
			g_string_append (helper->xml, "  ");
Packit caecb6
	}
Packit caecb6
	g_string_append_printf (helper->xml, "<%s",
Packit caecb6
				xb_silo_from_strtab (self, sn->element_name));
Packit caecb6
Packit caecb6
	/* add any attributes */
Packit caecb6
	for (guint8 i = 0; i < sn->nr_attrs; i++) {
Packit caecb6
		XbSiloAttr *a = xb_silo_get_attr (self, helper->off, i);
Packit caecb6
		g_autofree gchar *key = xb_string_xml_escape (xb_silo_from_strtab (self, a->attr_name));
Packit caecb6
		g_autofree gchar *val = xb_string_xml_escape (xb_silo_from_strtab (self, a->attr_value));
Packit caecb6
		g_string_append_printf (helper->xml, " %s=\"%s\"", key, val);
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* finish the opening tag and add any text if it exists */
Packit caecb6
	if (sn->text != XB_SILO_UNSET) {
Packit caecb6
		g_autofree gchar *text = xb_string_xml_escape (xb_silo_from_strtab (self, sn->text));
Packit caecb6
		g_string_append (helper->xml, ">");
Packit caecb6
		g_string_append (helper->xml, text);
Packit caecb6
	} else {
Packit caecb6
		g_string_append (helper->xml, ">");
Packit caecb6
		if (helper->flags & XB_NODE_EXPORT_FLAG_FORMAT_MULTILINE)
Packit caecb6
			g_string_append (helper->xml, "\n");
Packit caecb6
	}
Packit caecb6
	helper->off += xb_silo_node_get_size (sn);
Packit caecb6
Packit caecb6
	/* recurse deeper */
Packit caecb6
	while (xb_silo_get_node(self, helper->off)->is_node) {
Packit caecb6
		XbSiloNode *child = xb_silo_get_node (self, helper->off);
Packit caecb6
		helper->level++;
Packit caecb6
		if (!xb_silo_export_node (self, helper, child, error))
Packit caecb6
			return FALSE;
Packit caecb6
		helper->level--;
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* check for the single byte sentinel */
Packit caecb6
	sn2 = xb_silo_get_node (self, helper->off);
Packit caecb6
	if (sn2->is_node) {
Packit caecb6
		g_set_error (error,
Packit caecb6
			     G_IO_ERROR,
Packit caecb6
			     G_IO_ERROR_INVALID_DATA,
Packit caecb6
			     "no seninel at %" G_GUINT32_FORMAT,
Packit caecb6
			     helper->off);
Packit caecb6
		return FALSE;
Packit caecb6
	}
Packit caecb6
	helper->off += xb_silo_node_get_size (sn2);
Packit caecb6
Packit caecb6
	/* add closing tag */
Packit caecb6
	if ((helper->flags & XB_NODE_EXPORT_FLAG_FORMAT_INDENT) > 0 &&
Packit caecb6
	    sn->text == XB_SILO_UNSET) {
Packit caecb6
		for (guint i = 0; i < helper->level; i++)
Packit caecb6
			g_string_append (helper->xml, "  ");
Packit caecb6
	}
Packit caecb6
	g_string_append_printf (helper->xml, "</%s>",
Packit caecb6
				xb_silo_from_strtab (self, sn->element_name));
Packit caecb6
Packit caecb6
	/* add any optional tail */
Packit caecb6
	if (sn->tail != XB_SILO_UNSET) {
Packit caecb6
		g_autofree gchar *tail = xb_string_xml_escape (xb_silo_from_strtab (self, sn->tail));
Packit caecb6
		g_string_append (helper->xml, tail);
Packit caecb6
	}
Packit caecb6
Packit caecb6
	if (helper->flags & XB_NODE_EXPORT_FLAG_FORMAT_MULTILINE)
Packit caecb6
		g_string_append (helper->xml, "\n");
Packit caecb6
Packit caecb6
	return TRUE;
Packit caecb6
}
Packit caecb6
Packit caecb6
/* private */
Packit caecb6
GString *
Packit caecb6
xb_silo_export_with_root (XbSilo *self, XbNode *root, XbNodeExportFlags flags, GError **error)
Packit caecb6
{
Packit caecb6
	XbSiloNode *sn;
Packit caecb6
	XbSiloExportHelper helper = {
Packit caecb6
		.flags		= flags,
Packit caecb6
		.level		= 0,
Packit caecb6
		.off		= sizeof(XbSiloHeader),
Packit caecb6
	};
Packit caecb6
Packit caecb6
	g_return_val_if_fail (XB_IS_SILO (self), NULL);
Packit caecb6
Packit caecb6
	/* this implies the other */
Packit caecb6
	if (flags & XB_NODE_EXPORT_FLAG_ONLY_CHILDREN)
Packit caecb6
		flags |= XB_NODE_EXPORT_FLAG_INCLUDE_SIBLINGS;
Packit caecb6
Packit caecb6
	/* optional subtree export */
Packit caecb6
	if (root != NULL) {
Packit caecb6
		sn = xb_node_get_sn (root);
Packit caecb6
		if (sn != NULL && flags & XB_NODE_EXPORT_FLAG_ONLY_CHILDREN)
Packit caecb6
			sn = xb_silo_node_get_child (self, sn);
Packit caecb6
	} else {
Packit caecb6
		sn = xb_silo_get_sroot (self);
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* no root */
Packit caecb6
	if (sn == NULL) {
Packit caecb6
		g_set_error_literal (error,
Packit caecb6
				     G_IO_ERROR,
Packit caecb6
				     G_IO_ERROR_NOT_FOUND,
Packit caecb6
				     "no data to export");
Packit caecb6
		return NULL;
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* root node */
Packit caecb6
	helper.xml = g_string_new (NULL);
Packit caecb6
	if ((flags & XB_NODE_EXPORT_FLAG_ADD_HEADER) > 0)
Packit caecb6
		g_string_append (helper.xml, "\n");
Packit caecb6
	do {
Packit caecb6
		if (!xb_silo_export_node (self, &helper, sn, error)) {
Packit caecb6
			g_string_free (helper.xml, TRUE);
Packit caecb6
			return NULL;
Packit caecb6
		}
Packit caecb6
		if ((flags & XB_NODE_EXPORT_FLAG_INCLUDE_SIBLINGS) == 0)
Packit caecb6
			break;
Packit caecb6
		sn = xb_silo_node_get_next (self, sn);
Packit caecb6
	} while (sn != NULL);
Packit caecb6
Packit caecb6
	/* success */
Packit caecb6
	return helper.xml;
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_silo_export:
Packit caecb6
 * @self: a #XbSilo
Packit caecb6
 * @flags: some #XbNodeExportFlags, e.g. #XB_NODE_EXPORT_FLAG_NONE
Packit caecb6
 * @error: the #GError, or %NULL
Packit caecb6
 *
Packit caecb6
 * Exports the silo back to XML.
Packit caecb6
 *
Packit caecb6
 * Returns: XML data, or %NULL for an error
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.0
Packit caecb6
 **/
Packit caecb6
gchar *
Packit caecb6
xb_silo_export (XbSilo *self, XbNodeExportFlags flags, GError **error)
Packit caecb6
{
Packit caecb6
	GString *xml;
Packit caecb6
	g_return_val_if_fail (XB_IS_SILO (self), NULL);
Packit caecb6
	g_return_val_if_fail (error == NULL || *error == NULL, NULL);
Packit caecb6
	xml = xb_silo_export_with_root (self, NULL, flags, error);
Packit caecb6
	if (xml == NULL)
Packit caecb6
		return NULL;
Packit caecb6
	return g_string_free (xml, FALSE);
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_silo_export_file:
Packit caecb6
 * @self: a #XbSilo
Packit caecb6
 * @file: a #GFile
Packit caecb6
 * @flags: some #XbNodeExportFlags, e.g. #XB_NODE_EXPORT_FLAG_NONE
Packit caecb6
 * @cancellable: a #GCancellable, or %NULL
Packit caecb6
 * @error: the #GError, or %NULL
Packit caecb6
 *
Packit caecb6
 * Exports the silo back to an XML file.
Packit caecb6
 *
Packit caecb6
 * Returns: %TRUE on success
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.2
Packit caecb6
 **/
Packit caecb6
gboolean
Packit caecb6
xb_silo_export_file (XbSilo *self,
Packit caecb6
		     GFile *file,
Packit caecb6
		     XbNodeExportFlags flags,
Packit caecb6
		     GCancellable *cancellable,
Packit caecb6
		     GError **error)
Packit caecb6
{
Packit caecb6
	g_autoptr(GString) xml = NULL;
Packit caecb6
Packit caecb6
	g_return_val_if_fail (XB_IS_SILO (self), FALSE);
Packit caecb6
	g_return_val_if_fail (G_IS_FILE (file), FALSE);
Packit caecb6
	g_return_val_if_fail (cancellable == NULL || G_IS_CANCELLABLE (cancellable), FALSE);
Packit caecb6
	g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
Packit caecb6
Packit caecb6
	xml = xb_silo_export_with_root (self, NULL, flags, error);
Packit caecb6
	if (xml == NULL)
Packit caecb6
		return FALSE;
Packit caecb6
	return g_file_replace_contents (file,
Packit caecb6
					xml->str,
Packit caecb6
					xml->len,
Packit caecb6
					NULL, /* etag */
Packit caecb6
					FALSE, /* make-backup */
Packit caecb6
					G_FILE_CREATE_NONE,
Packit caecb6
					NULL, /* new etag */
Packit caecb6
					cancellable,
Packit caecb6
					error);
Packit caecb6
}