Blame src/xb-silo-query.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 <string.h>
Packit caecb6
#include <gio/gio.h>
Packit caecb6
Packit caecb6
#include "xb-node-private.h"
Packit caecb6
#include "xb-opcode.h"
Packit caecb6
#include "xb-silo-private.h"
Packit caecb6
#include "xb-silo-query-private.h"
Packit caecb6
#include "xb-stack-private.h"
Packit caecb6
#include "xb-query-private.h"
Packit caecb6
Packit caecb6
static gboolean
Packit caecb6
xb_silo_query_node_matches (XbSilo *self,
Packit caecb6
			    XbMachine *machine,
Packit caecb6
			    XbSiloNode *sn,
Packit caecb6
			    XbQuerySection *section,
Packit caecb6
			    XbSiloQueryData *query_data,
Packit caecb6
			    gboolean *result,
Packit caecb6
			    GError **error)
Packit caecb6
{
Packit caecb6
	/* we have an index into the string table */
Packit caecb6
	if (section->element_idx != sn->element_name &&
Packit caecb6
	    section->kind != XB_SILO_QUERY_KIND_WILDCARD) {
Packit caecb6
		*result = FALSE;
Packit caecb6
		return TRUE;
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* for section */
Packit caecb6
	query_data->position += 1;
Packit caecb6
Packit caecb6
	/* check predicates */
Packit caecb6
	if (section->predicates != NULL) {
Packit caecb6
		for (guint i = 0; i < section->predicates->len; i++) {
Packit caecb6
			XbStack *opcodes = g_ptr_array_index (section->predicates, i);
Packit caecb6
			if (!xb_machine_run (machine, opcodes, result, query_data, error))
Packit caecb6
				return FALSE;
Packit caecb6
		}
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* success */
Packit caecb6
	return TRUE;
Packit caecb6
}
Packit caecb6
Packit caecb6
typedef struct {
Packit caecb6
	GPtrArray	*sections;	/* of XbQuerySection */
Packit caecb6
	GPtrArray	*results;	/* of XbNode */
Packit caecb6
	GHashTable	*results_hash;	/* of sn:1 */
Packit caecb6
	guint		 limit;
Packit caecb6
	XbSiloQueryData	*query_data;
Packit caecb6
} XbSiloQueryHelper;
Packit caecb6
Packit caecb6
static gboolean
Packit caecb6
xb_silo_query_section_add_result (XbSilo *self, XbSiloQueryHelper *helper, XbSiloNode *sn)
Packit caecb6
{
Packit caecb6
	if (g_hash_table_lookup (helper->results_hash, sn) != NULL)
Packit caecb6
		return FALSE;
Packit caecb6
	g_ptr_array_add (helper->results, xb_silo_node_create (self, sn));
Packit caecb6
	g_hash_table_add (helper->results_hash, sn);
Packit caecb6
	return helper->results->len == helper->limit;
Packit caecb6
}
Packit caecb6
Packit caecb6
/*
Packit caecb6
 * @parent: (allow-none)
Packit caecb6
 */
Packit caecb6
static gboolean
Packit caecb6
xb_silo_query_section_root (XbSilo *self,
Packit caecb6
			    XbSiloNode *sn,
Packit caecb6
			    guint i,
Packit caecb6
			    XbSiloQueryHelper *helper,
Packit caecb6
			    GError **error)
Packit caecb6
{
Packit caecb6
	XbMachine *machine = xb_silo_get_machine (self);
Packit caecb6
	XbSiloQueryData *query_data = helper->query_data;
Packit caecb6
	XbQuerySection *section = g_ptr_array_index (helper->sections, i);
Packit caecb6
Packit caecb6
	/* handle parent */
Packit caecb6
	if (section->kind == XB_SILO_QUERY_KIND_PARENT) {
Packit caecb6
		XbSiloNode *parent;
Packit caecb6
		if (sn == NULL) {
Packit caecb6
			g_set_error_literal (error,
Packit caecb6
					     G_IO_ERROR,
Packit caecb6
					     G_IO_ERROR_INVALID_ARGUMENT,
Packit caecb6
					     "cannot obtain parent for root");
Packit caecb6
			return FALSE;
Packit caecb6
		}
Packit caecb6
		parent = xb_silo_node_get_parent (self, sn);
Packit caecb6
		if (parent == NULL) {
Packit caecb6
			g_set_error (error,
Packit caecb6
				     G_IO_ERROR,
Packit caecb6
				     G_IO_ERROR_INVALID_ARGUMENT,
Packit caecb6
				     "no parent set for %s",
Packit caecb6
				     xb_silo_node_get_element (self, sn));
Packit caecb6
			return FALSE;
Packit caecb6
		}
Packit caecb6
		if (i == helper->sections->len - 1) {
Packit caecb6
			xb_silo_query_section_add_result (self, helper, parent);
Packit caecb6
			return TRUE;
Packit caecb6
		}
Packit caecb6
//		g_debug ("PARENT @%u",
Packit caecb6
//			 xb_silo_get_offset_for_node (self, parent));
Packit caecb6
		return xb_silo_query_section_root (self, parent, i + 1, helper, error);
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* no node means root */
Packit caecb6
	if (sn == NULL) {
Packit caecb6
		sn = xb_silo_get_sroot (self);
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
					     "silo root not found");
Packit caecb6
			return FALSE;
Packit caecb6
		}
Packit caecb6
	} else {
Packit caecb6
		sn = xb_silo_node_get_child (self, sn);
Packit caecb6
		if (sn == NULL)
Packit caecb6
			return TRUE;
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* set up level pointer */
Packit caecb6
	query_data->position = 0;
Packit caecb6
Packit caecb6
	/* continue matching children ".." */
Packit caecb6
	do {
Packit caecb6
		gboolean result = TRUE;
Packit caecb6
		query_data->sn = sn;
Packit caecb6
		if (!xb_silo_query_node_matches (self, machine, sn, section,
Packit caecb6
						 query_data, &result, error))
Packit caecb6
			return FALSE;
Packit caecb6
		if (result) {
Packit caecb6
			if (i == helper->sections->len - 1) {
Packit caecb6
//				g_debug ("add result %u",
Packit caecb6
//					 xb_silo_get_offset_for_node (self, sn));
Packit caecb6
				if (xb_silo_query_section_add_result (self, helper, sn))
Packit caecb6
					break;
Packit caecb6
			} else {
Packit caecb6
//				g_debug ("MATCH %s at @%u, deeper",
Packit caecb6
//					 xb_silo_node_get_element (self, sn),
Packit caecb6
//					 xb_silo_get_offset_for_node (self, sn));
Packit caecb6
				if (!xb_silo_query_section_root (self, sn, i + 1, helper, error))
Packit caecb6
					return FALSE;
Packit caecb6
				if (helper->results->len > 0 &&
Packit caecb6
				    helper->results->len == helper->limit)
Packit caecb6
					break;
Packit caecb6
			}
Packit caecb6
		}
Packit caecb6
		if (sn->next == 0x0)
Packit caecb6
			break;
Packit caecb6
		sn = xb_silo_get_node (self, sn->next);
Packit caecb6
	} while (TRUE);
Packit caecb6
	return TRUE;
Packit caecb6
}
Packit caecb6
Packit caecb6
static gboolean
Packit caecb6
xb_silo_query_part (XbSilo *self,
Packit caecb6
		    XbSiloNode *sroot,
Packit caecb6
		    GPtrArray *results,
Packit caecb6
		    GHashTable *results_hash,
Packit caecb6
		    XbQuery *query,
Packit caecb6
		    XbSiloQueryData *query_data,
Packit caecb6
		    GError **error)
Packit caecb6
{
Packit caecb6
	XbSiloQueryHelper helper = {
Packit caecb6
		.results = results,
Packit caecb6
		.limit = xb_query_get_limit (query),
Packit caecb6
		.results_hash = results_hash,
Packit caecb6
		.query_data = query_data,
Packit caecb6
	};
Packit caecb6
Packit caecb6
	/* find each section */
Packit caecb6
	helper.sections = xb_query_get_sections (query);
Packit caecb6
	return xb_silo_query_section_root (self, sroot, 0, &helper, error);
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_silo_query_with_root: (skip)
Packit caecb6
 * @self: a #XbSilo
Packit caecb6
 * @n: (allow-none): a #XbNode
Packit caecb6
 * @xpath: an XPath, e.g. `/components/component[@type=desktop]/id[abe.desktop]`
Packit caecb6
 * @limit: maximum number of results to return, or 0 for "all"
Packit caecb6
 * @error: the #GError, or %NULL
Packit caecb6
 *
Packit caecb6
 * Searches the silo using an XPath query, returning up to @limit results.
Packit caecb6
 *
Packit caecb6
 * It is safe to call this function from a different thread to the one that
Packit caecb6
 * created the #XbSilo.
Packit caecb6
 *
Packit caecb6
 * Please note: Only a subset of XPath is supported.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer container) (element-type XbNode): results, or %NULL if unfound
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.0
Packit caecb6
 **/
Packit caecb6
GPtrArray *
Packit caecb6
xb_silo_query_with_root (XbSilo *self, XbNode *n, const gchar *xpath, guint limit, GError **error)
Packit caecb6
{
Packit caecb6
	XbSiloNode *sn = NULL;
Packit caecb6
	g_auto(GStrv) split = NULL;
Packit caecb6
	g_autoptr(GHashTable) results_hash = g_hash_table_new (g_direct_hash, g_direct_equal);
Packit caecb6
	g_autoptr(GPtrArray) results = g_ptr_array_new_with_free_func ((GDestroyNotify) g_object_unref);
Packit caecb6
	g_autoptr(GTimer) timer = g_timer_new ();
Packit caecb6
	XbSiloQueryData query_data = {
Packit caecb6
		.sn = NULL,
Packit caecb6
		.position = 0,
Packit caecb6
	};
Packit caecb6
Packit caecb6
	g_return_val_if_fail (XB_IS_SILO (self), NULL);
Packit caecb6
	g_return_val_if_fail (xpath != NULL, NULL);
Packit caecb6
	g_return_val_if_fail (error == NULL || *error == NULL, NULL);
Packit caecb6
Packit caecb6
	/* empty silo */
Packit caecb6
	if (xb_silo_is_empty (self)) {
Packit caecb6
		g_set_error_literal (error,
Packit caecb6
				     G_IO_ERROR,
Packit caecb6
				     G_IO_ERROR_NOT_FOUND,
Packit caecb6
				     "silo has no data");
Packit caecb6
		return NULL;
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* subtree query */
Packit caecb6
	if (n != NULL) {
Packit caecb6
		sn = xb_node_get_sn (n);
Packit caecb6
		if (xpath[0] == '/') {
Packit caecb6
			g_set_error_literal (error,
Packit caecb6
					     G_IO_ERROR,
Packit caecb6
					     G_IO_ERROR_NOT_SUPPORTED,
Packit caecb6
					     "XPath node query not supported");
Packit caecb6
			return NULL;
Packit caecb6
		}
Packit caecb6
	} else {
Packit caecb6
		/* assume it's just a root query */
Packit caecb6
		if (xpath[0] == '/')
Packit caecb6
			xpath++;
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* do 'or' searches */
Packit caecb6
	split = g_strsplit (xpath, "|", -1);
Packit caecb6
	for (guint i = 0; split[i] != NULL; i++) {
Packit caecb6
		g_autoptr(GError) error_local = NULL;
Packit caecb6
		g_autoptr(XbQuery) query = xb_query_new (self, split[i], &error_local);
Packit caecb6
		if (query == NULL) {
Packit caecb6
			if (g_error_matches (error_local, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT) &&
Packit caecb6
			    (split[i + 1] != NULL || results->len > 0)) {
Packit caecb6
				g_debug ("ignoring for OR statement: %s",
Packit caecb6
					 error_local->message);
Packit caecb6
				continue;
Packit caecb6
			}
Packit caecb6
			g_propagate_prefixed_error (error,
Packit caecb6
						    g_steal_pointer (&error_local),
Packit caecb6
						    "failed to process %s: ",
Packit caecb6
						    xpath);
Packit caecb6
			return NULL;
Packit caecb6
		}
Packit caecb6
		xb_query_set_limit (query, limit);
Packit caecb6
		if (!xb_silo_query_part (self, sn,
Packit caecb6
					 results, results_hash,
Packit caecb6
					 query, &query_data,
Packit caecb6
					 error)) {
Packit caecb6
			return NULL;
Packit caecb6
		}
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* profile */
Packit caecb6
	if (xb_silo_get_profile_flags (self) & XB_SILO_PROFILE_FLAG_XPATH) {
Packit caecb6
		xb_silo_add_profile (self, timer,
Packit caecb6
				     "query on %s with `%s` limit=%u -> %u results",
Packit caecb6
				     n != NULL ? xb_node_get_element (n) : "/",
Packit caecb6
				     xpath, limit, results->len);
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* nothing found */
Packit caecb6
	if (results->len == 0) {
Packit caecb6
		g_set_error (error,
Packit caecb6
			     G_IO_ERROR,
Packit caecb6
			     G_IO_ERROR_NOT_FOUND,
Packit caecb6
			     "no results for XPath query '%s'",
Packit caecb6
			     xpath);
Packit caecb6
		return NULL;
Packit caecb6
	}
Packit caecb6
	return g_steal_pointer (&results);
Packit caecb6
}
Packit caecb6
Packit caecb6
static void
Packit caecb6
_g_ptr_array_reverse (GPtrArray *array)
Packit caecb6
{
Packit caecb6
	guint last_idx = array->len - 1;
Packit caecb6
	for (guint i = 0; i < array->len / 2; i++) {
Packit caecb6
		gpointer tmp = array->pdata[i];
Packit caecb6
		array->pdata[i] = array->pdata[last_idx - i];
Packit caecb6
		array->pdata[last_idx - i] = tmp;
Packit caecb6
	}
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_silo_query_with_root_full: (skip)
Packit caecb6
 * @self: a #XbSilo
Packit caecb6
 * @n: (allow-none): a #XbNode
Packit caecb6
 * @query: an #XbQuery
Packit caecb6
 * @error: the #GError, or %NULL
Packit caecb6
 *
Packit caecb6
 * Searches the silo using an XPath query, returning up to @limit results.
Packit caecb6
 *
Packit caecb6
 * It is safe to call this function from a different thread to the one that
Packit caecb6
 * created the #XbSilo.
Packit caecb6
 *
Packit caecb6
 * Please note: Only a subset of XPath is supported.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer container) (element-type XbNode): results, or %NULL if unfound
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.4
Packit caecb6
 **/
Packit caecb6
GPtrArray *
Packit caecb6
xb_silo_query_with_root_full (XbSilo *self, XbNode *n, XbQuery *query, GError **error)
Packit caecb6
{
Packit caecb6
	XbSiloNode *sn = NULL;
Packit caecb6
	g_autoptr(GHashTable) results_hash = g_hash_table_new (g_direct_hash, g_direct_equal);
Packit caecb6
	g_autoptr(GPtrArray) results = g_ptr_array_new_with_free_func ((GDestroyNotify) g_object_unref);
Packit caecb6
	g_autoptr(GTimer) timer = g_timer_new ();
Packit caecb6
	XbSiloQueryData query_data = {
Packit caecb6
		.sn = NULL,
Packit caecb6
		.position = 0,
Packit caecb6
	};
Packit caecb6
Packit caecb6
	/* empty silo */
Packit caecb6
	if (xb_silo_is_empty (self)) {
Packit caecb6
		g_set_error_literal (error,
Packit caecb6
				     G_IO_ERROR,
Packit caecb6
				     G_IO_ERROR_NOT_FOUND,
Packit caecb6
				     "silo has no data");
Packit caecb6
		return NULL;
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* subtree query */
Packit caecb6
	if (n != NULL)
Packit caecb6
		sn = xb_node_get_sn (n);
Packit caecb6
Packit caecb6
	/* only one query allowed */
Packit caecb6
	if (!xb_silo_query_part (self, sn, results, results_hash,
Packit caecb6
				 query, &query_data, error))
Packit caecb6
		return NULL;
Packit caecb6
Packit caecb6
	/* profile */
Packit caecb6
	if (xb_silo_get_profile_flags (self) & XB_SILO_PROFILE_FLAG_XPATH) {
Packit caecb6
		g_autofree gchar *tmp = xb_query_to_string (query);
Packit caecb6
		xb_silo_add_profile (self, timer,
Packit caecb6
				     "query on %s with `%s` limit=%u -> %u results",
Packit caecb6
				     n != NULL ? xb_node_get_element (n) : "/",
Packit caecb6
				     tmp,
Packit caecb6
				     xb_query_get_limit (query),
Packit caecb6
				     results->len);
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* nothing found */
Packit caecb6
	if (results->len == 0) {
Packit caecb6
		g_autofree gchar *tmp = xb_query_to_string (query);
Packit caecb6
		g_set_error (error,
Packit caecb6
			     G_IO_ERROR,
Packit caecb6
			     G_IO_ERROR_NOT_FOUND,
Packit caecb6
			     "no results for XPath query '%s'",
Packit caecb6
			     tmp);
Packit caecb6
		return NULL;
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* reverse order */
Packit caecb6
	if (xb_query_get_flags (query) & XB_QUERY_FLAG_REVERSE)
Packit caecb6
		_g_ptr_array_reverse (results);
Packit caecb6
Packit caecb6
	return g_steal_pointer (&results);
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_silo_query_full:
Packit caecb6
 * @self: a #XbSilo
Packit caecb6
 * @query: an #XbQuery
Packit caecb6
 * @error: the #GError, or %NULL
Packit caecb6
 *
Packit caecb6
 * Searches the silo using an XPath query.
Packit caecb6
 *
Packit caecb6
 * It is safe to call this function from a different thread to the one that
Packit caecb6
 * created the #XbSilo.
Packit caecb6
 *
Packit caecb6
 * Please note: Only a subset of XPath is supported.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer container) (element-type XbNode): results, or %NULL if unfound
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.13
Packit caecb6
 **/
Packit caecb6
GPtrArray *
Packit caecb6
xb_silo_query_full (XbSilo *self, XbQuery *query, GError **error)
Packit caecb6
{
Packit caecb6
	g_return_val_if_fail (XB_IS_SILO (self), NULL);
Packit caecb6
	g_return_val_if_fail (XB_IS_QUERY (query), NULL);
Packit caecb6
	g_return_val_if_fail (error == NULL || *error == NULL, NULL);
Packit caecb6
	return xb_silo_query_with_root_full (self, NULL, query, error);
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_silo_query_first_full:
Packit caecb6
 * @self: a #XbSilo
Packit caecb6
 * @query: an #XbQuery
Packit caecb6
 * @error: the #GError, or %NULL
Packit caecb6
 *
Packit caecb6
 * Searches the silo using an XPath query, returning up to one result.
Packit caecb6
 *
Packit caecb6
 * It is safe to call this function from a different thread to the one that
Packit caecb6
 * created the #XbSilo.
Packit caecb6
 *
Packit caecb6
 * Please note: Only a tiny subset of XPath 1.0 is supported.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer none): a #XbNode, or %NULL if unfound
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.13
Packit caecb6
 **/
Packit caecb6
XbNode *
Packit caecb6
xb_silo_query_first_full (XbSilo *self, XbQuery *query, GError **error)
Packit caecb6
{
Packit caecb6
	g_autoptr(GPtrArray) results = NULL;
Packit caecb6
	results = xb_silo_query_full (self, query, error);
Packit caecb6
	if (results == NULL)
Packit caecb6
		return NULL;
Packit caecb6
	return g_object_ref (g_ptr_array_index (results, 0));
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_silo_query:
Packit caecb6
 * @self: a #XbSilo
Packit caecb6
 * @xpath: an XPath, e.g. `/components/component[@type=desktop]/id[abe.desktop]`
Packit caecb6
 * @limit: maximum number of results to return, or 0 for "all"
Packit caecb6
 * @error: the #GError, or %NULL
Packit caecb6
 *
Packit caecb6
 * Searches the silo using an XPath query, returning up to @limit results.
Packit caecb6
 *
Packit caecb6
 * It is safe to call this function from a different thread to the one that
Packit caecb6
 * created the #XbSilo.
Packit caecb6
 *
Packit caecb6
 * Please note: Only a subset of XPath is supported.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer container) (element-type XbNode): results, or %NULL if unfound
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.0
Packit caecb6
 **/
Packit caecb6
GPtrArray *
Packit caecb6
xb_silo_query (XbSilo *self, const gchar *xpath, guint limit, GError **error)
Packit caecb6
{
Packit caecb6
	g_return_val_if_fail (XB_IS_SILO (self), NULL);
Packit caecb6
	g_return_val_if_fail (xpath != NULL, NULL);
Packit caecb6
	g_return_val_if_fail (error == NULL || *error == NULL, NULL);
Packit caecb6
	return xb_silo_query_with_root (self, NULL, xpath, limit, error);
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_silo_query_first:
Packit caecb6
 * @self: a #XbSilo
Packit caecb6
 * @xpath: An XPath, e.g. `/components/component[@type=desktop]/id[abe.desktop]`
Packit caecb6
 * @error: the #GError, or %NULL
Packit caecb6
 *
Packit caecb6
 * Searches the silo using an XPath query, returning up to one result.
Packit caecb6
 *
Packit caecb6
 * It is safe to call this function from a different thread to the one that
Packit caecb6
 * created the #XbSilo.
Packit caecb6
 *
Packit caecb6
 * Please note: Only a tiny subset of XPath 1.0 is supported.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer none): a #XbNode, or %NULL if unfound
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.0
Packit caecb6
 **/
Packit caecb6
XbNode *
Packit caecb6
xb_silo_query_first (XbSilo *self, const gchar *xpath, GError **error)
Packit caecb6
{
Packit caecb6
	g_autoptr(GPtrArray) results = NULL;
Packit caecb6
	results = xb_silo_query_with_root (self, NULL, xpath, 1, error);
Packit caecb6
	if (results == NULL)
Packit caecb6
		return NULL;
Packit caecb6
	return g_object_ref (g_ptr_array_index (results, 0));
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_silo_query_build_index:
Packit caecb6
 * @self: a #XbSilo
Packit caecb6
 * @xpath: An XPath, e.g. `/components/component[@type=desktop]/id[abe.desktop]`
Packit caecb6
 * @attr: (nullable): Attribute name, e.g. `type`, or NULL
Packit caecb6
 * @error: the #GError, or %NULL
Packit caecb6
 *
Packit caecb6
 * Adds the `attr()` or `text()` results of a query to the index.
Packit caecb6
 *
Packit caecb6
 * Returns: %TRUE for success
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.4
Packit caecb6
 **/
Packit caecb6
gboolean
Packit caecb6
xb_silo_query_build_index (XbSilo *self,
Packit caecb6
			   const gchar *xpath,
Packit caecb6
			   const gchar *attr,
Packit caecb6
			   GError **error)
Packit caecb6
{
Packit caecb6
	g_autoptr(GError) error_local = NULL;
Packit caecb6
	g_autoptr(GPtrArray) array = NULL;
Packit caecb6
Packit caecb6
	g_return_val_if_fail (XB_IS_SILO (self), FALSE);
Packit caecb6
	g_return_val_if_fail (xpath != NULL, FALSE);
Packit caecb6
	g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
Packit caecb6
Packit caecb6
	/* do the query */
Packit caecb6
	array = xb_silo_query_with_root (self, NULL, xpath, 0, &error_local);
Packit caecb6
	if (array == NULL) {
Packit caecb6
		if (g_error_matches (error_local,
Packit caecb6
				     G_IO_ERROR,
Packit caecb6
				     G_IO_ERROR_INVALID_ARGUMENT) ||
Packit caecb6
		    g_error_matches (error_local,
Packit caecb6
				     G_IO_ERROR,
Packit caecb6
				     G_IO_ERROR_NOT_FOUND)) {
Packit caecb6
			g_debug ("ignoring index: %s", error_local->message);
Packit caecb6
			return TRUE;
Packit caecb6
		}
Packit caecb6
		g_propagate_error (error, g_steal_pointer (&error_local));
Packit caecb6
		return FALSE;
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* add each attribute name AND value */
Packit caecb6
	for (guint i = 0; i < array->len; i++) {
Packit caecb6
		XbNode *n = g_ptr_array_index (array, i);
Packit caecb6
		XbSiloNode *sn = xb_node_get_sn (n);
Packit caecb6
		if (attr != NULL) {
Packit caecb6
			guint32 off = xb_silo_get_offset_for_node (self, sn);
Packit caecb6
			for (guint8 j = 0; j < sn->nr_attrs; j++) {
Packit caecb6
				XbSiloAttr *a = xb_silo_get_attr (self, off, j);
Packit caecb6
				xb_silo_strtab_index_insert (self, a->attr_name);
Packit caecb6
				xb_silo_strtab_index_insert (self, a->attr_value);
Packit caecb6
			}
Packit caecb6
		} else {
Packit caecb6
			xb_silo_strtab_index_insert (self, sn->text);
Packit caecb6
		}
Packit caecb6
	}
Packit caecb6
Packit caecb6
	/* success */
Packit caecb6
	return TRUE;
Packit caecb6
}