Blame gdata/services/freebase/gdata-freebase-query.c

Packit 4b6dd7
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
Packit 4b6dd7
/*
Packit 4b6dd7
 * GData Client
Packit 4b6dd7
 * Copyright (C) 2014 Carlos Garnacho <carlosg@gnome.org>
Packit 4b6dd7
 *
Packit 4b6dd7
 * GData Client is free software; you can redistribute it and/or
Packit 4b6dd7
 * modify it under the terms of the GNU Lesser General Public
Packit 4b6dd7
 * License as published by the Free Software Foundation; either
Packit 4b6dd7
 * version 2.1 of the License, or (at your option) any later version.
Packit 4b6dd7
 *
Packit 4b6dd7
 * GData Client is distributed in the hope that it will be useful,
Packit 4b6dd7
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 4b6dd7
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 4b6dd7
 * Lesser General Public License for more details.
Packit 4b6dd7
 *
Packit 4b6dd7
 * You should have received a copy of the GNU Lesser General Public
Packit 4b6dd7
 * License along with GData Client.  If not, see <http://www.gnu.org/licenses/>.
Packit 4b6dd7
 */
Packit 4b6dd7
Packit 4b6dd7
/**
Packit 4b6dd7
 * SECTION:gdata-freebase-query
Packit 4b6dd7
 * @short_description: GData Freebase query object
Packit 4b6dd7
 * @stability: Stable
Packit 4b6dd7
 * @include: gdata/services/freebase/gdata-freebase-query.h
Packit 4b6dd7
 *
Packit 4b6dd7
 * #GDataFreebaseQuery represents a MQL query specific to the Google Freebase service.
Packit 4b6dd7
 *
Packit 4b6dd7
 * This implementation of #GDataQuery respects the gdata_query_set_max_results() call.
Packit 4b6dd7
 *
Packit 4b6dd7
 * For more details of Google Freebase API, see the <ulink type="http" url="https://developers.google.com/freebase/v1/">
Packit 4b6dd7
 * online documentation</ulink>.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Since: 0.15.1
Packit 4b6dd7
 * Deprecated: 0.17.7: Google Freebase has been permanently shut down.
Packit 4b6dd7
 */
Packit 4b6dd7
Packit 4b6dd7
#include <config.h>
Packit 4b6dd7
#include <glib.h>
Packit 4b6dd7
#include <glib/gi18n-lib.h>
Packit 4b6dd7
#include <string.h>
Packit 4b6dd7
#include <json-glib/json-glib.h>
Packit 4b6dd7
Packit 4b6dd7
#include "gdata-freebase-query.h"
Packit 4b6dd7
#include "gdata-query.h"
Packit 4b6dd7
#include "gdata-parser.h"
Packit 4b6dd7
#include "gdata-private.h"
Packit 4b6dd7
Packit 4b6dd7
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
Packit 4b6dd7
Packit 4b6dd7
static void gdata_freebase_query_finalize (GObject *self);
Packit 4b6dd7
static void gdata_freebase_query_set_property (GObject *self, guint prop_id, const GValue *value, GParamSpec *pspec);
Packit 4b6dd7
static void gdata_freebase_query_get_property (GObject *self, guint prop_id, GValue *value, GParamSpec *pspec);
Packit 4b6dd7
static void get_query_uri (GDataQuery *self, const gchar *feed_uri, GString *query_uri, gboolean *params_started);
Packit 4b6dd7
Packit 4b6dd7
struct _GDataFreebaseQueryPrivate {
Packit 4b6dd7
	/* These params are here not in GDataQuery due of differently named query params for JSON protocols therefore need for different parse_uri */
Packit 4b6dd7
	GVariant *variant;
Packit 4b6dd7
	JsonNode *query_node;
Packit 4b6dd7
};
Packit 4b6dd7
Packit 4b6dd7
enum {
Packit 4b6dd7
	PROP_VARIANT = 1,
Packit 4b6dd7
};
Packit 4b6dd7
Packit 4b6dd7
G_DEFINE_TYPE (GDataFreebaseQuery, gdata_freebase_query, GDATA_TYPE_QUERY)
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
gdata_freebase_query_class_init (GDataFreebaseQueryClass *klass)
Packit 4b6dd7
{
Packit 4b6dd7
	GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Packit 4b6dd7
	GDataQueryClass *query_class = GDATA_QUERY_CLASS (klass);
Packit 4b6dd7
Packit 4b6dd7
	g_type_class_add_private (klass, sizeof (GDataFreebaseQueryPrivate));
Packit 4b6dd7
Packit 4b6dd7
	gobject_class->finalize = gdata_freebase_query_finalize;
Packit 4b6dd7
	gobject_class->set_property = gdata_freebase_query_set_property;
Packit 4b6dd7
	gobject_class->get_property = gdata_freebase_query_get_property;
Packit 4b6dd7
Packit 4b6dd7
	query_class->get_query_uri = get_query_uri;
Packit 4b6dd7
Packit 4b6dd7
	/**
Packit 4b6dd7
	 * GDataFreebaseQuery:variant:
Packit 4b6dd7
	 *
Packit 4b6dd7
	 * Variant containing the MQL query. The variant is a very generic container of type "a{smv}",
Packit 4b6dd7
	 * containing (possibly nested) Freebase schema types and values.
Packit 4b6dd7
	 *
Packit 4b6dd7
	 * Since: 0.15.1
Packit 4b6dd7
	 * Deprecated: 0.17.7: Google Freebase has been permanently shut down.
Packit 4b6dd7
	 */
Packit 4b6dd7
	g_object_class_install_property (gobject_class, PROP_VARIANT,
Packit 4b6dd7
	                                 g_param_spec_variant ("variant",
Packit 4b6dd7
							       "Variant",
Packit 4b6dd7
							       "Variant to construct the query from.",
Packit 4b6dd7
							       G_VARIANT_TYPE ("a{smv}"), NULL,
Packit 4b6dd7
							       G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS |
Packit 4b6dd7
							       G_PARAM_DEPRECATED));
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
gdata_freebase_query_init (GDataFreebaseQuery *self)
Packit 4b6dd7
{
Packit 4b6dd7
	self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, GDATA_TYPE_FREEBASE_QUERY, GDataFreebaseQueryPrivate);
Packit 4b6dd7
Packit 4b6dd7
	/* https://developers.google.com/freebase/v1/search#cursor */
Packit 4b6dd7
	_gdata_query_set_pagination_type (GDATA_QUERY (self),
Packit 4b6dd7
	                                  GDATA_QUERY_PAGINATION_INDEXED);
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
gdata_freebase_query_finalize (GObject *self)
Packit 4b6dd7
{
Packit 4b6dd7
	GDataFreebaseQueryPrivate *priv = GDATA_FREEBASE_QUERY (self)->priv;
Packit 4b6dd7
Packit 4b6dd7
	if (priv->variant != NULL)
Packit 4b6dd7
		g_variant_unref (priv->variant);
Packit 4b6dd7
	if (priv->query_node != NULL)
Packit 4b6dd7
		json_node_free (priv->query_node);
Packit 4b6dd7
	/* Chain up to the parent class */
Packit 4b6dd7
	G_OBJECT_CLASS (gdata_freebase_query_parent_class)->finalize (self);
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
gdata_freebase_query_set_property (GObject *self, guint prop_id, const GValue *value, GParamSpec *pspec)
Packit 4b6dd7
{
Packit 4b6dd7
	GDataFreebaseQueryPrivate *priv = GDATA_FREEBASE_QUERY (self)->priv;
Packit 4b6dd7
Packit 4b6dd7
	switch (prop_id) {
Packit 4b6dd7
	case PROP_VARIANT:
Packit 4b6dd7
		priv->variant = g_value_get_variant (value);
Packit 4b6dd7
		if (priv->variant)
Packit 4b6dd7
			priv->query_node = json_gvariant_serialize (priv->variant);
Packit 4b6dd7
		break;
Packit 4b6dd7
	default:
Packit 4b6dd7
		G_OBJECT_WARN_INVALID_PROPERTY_ID (self, prop_id, pspec);
Packit 4b6dd7
		break;
Packit 4b6dd7
	}
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
gdata_freebase_query_get_property (GObject *self, guint prop_id, GValue *value, GParamSpec *pspec)
Packit 4b6dd7
{
Packit 4b6dd7
	GDataFreebaseQueryPrivate *priv = GDATA_FREEBASE_QUERY (self)->priv;
Packit 4b6dd7
Packit 4b6dd7
	switch (prop_id) {
Packit 4b6dd7
	case PROP_VARIANT:
Packit 4b6dd7
		g_value_set_variant (value, priv->variant);
Packit 4b6dd7
		break;
Packit 4b6dd7
	default:
Packit 4b6dd7
		G_OBJECT_WARN_INVALID_PROPERTY_ID (self, prop_id, pspec);
Packit 4b6dd7
		break;
Packit 4b6dd7
	}
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
get_query_uri (GDataQuery *self, const gchar *feed_uri, GString *query_uri, gboolean *params_started)
Packit 4b6dd7
{
Packit 4b6dd7
	GDataFreebaseQueryPrivate *priv = GDATA_FREEBASE_QUERY (self)->priv;
Packit 4b6dd7
	const gchar *query;
Packit 4b6dd7
Packit 4b6dd7
#define APPEND_SEP g_string_append_c (query_uri, (*params_started == FALSE) ? '?' : '&';; *params_started = TRUE;
Packit 4b6dd7
Packit 4b6dd7
	query = gdata_query_get_q (self);
Packit 4b6dd7
Packit 4b6dd7
	if (query != NULL) {
Packit 4b6dd7
		APPEND_SEP;
Packit 4b6dd7
		g_string_append (query_uri, "query=");
Packit 4b6dd7
		g_string_append (query_uri, query);
Packit 4b6dd7
	} else if (priv->query_node != NULL) {
Packit 4b6dd7
		JsonGenerator *generator;
Packit 4b6dd7
		JsonNode *copy;
Packit 4b6dd7
		gchar *json;
Packit 4b6dd7
		guint limit;
Packit 4b6dd7
Packit 4b6dd7
		copy = json_node_copy (priv->query_node);
Packit 4b6dd7
Packit 4b6dd7
		limit = gdata_query_get_max_results (self);
Packit 4b6dd7
Packit 4b6dd7
		if (limit > 0) {
Packit 4b6dd7
			JsonNode *limit_node;
Packit 4b6dd7
			JsonObject *object;
Packit 4b6dd7
Packit 4b6dd7
			limit_node = json_node_new (JSON_NODE_VALUE);
Packit 4b6dd7
			json_node_set_int (limit_node, limit);
Packit 4b6dd7
Packit 4b6dd7
			object = json_node_get_object (copy);
Packit 4b6dd7
			json_object_set_member (object, "limit", limit_node);
Packit 4b6dd7
		}
Packit 4b6dd7
Packit 4b6dd7
		generator = json_generator_new ();
Packit 4b6dd7
		json_generator_set_root (generator, copy);
Packit 4b6dd7
		json = json_generator_to_data (generator, NULL);
Packit 4b6dd7
		g_object_unref (generator);
Packit 4b6dd7
Packit 4b6dd7
		APPEND_SEP;
Packit 4b6dd7
		g_string_append (query_uri, "query=");
Packit 4b6dd7
		g_string_append (query_uri, json);
Packit 4b6dd7
		g_free (json);
Packit 4b6dd7
	}
Packit 4b6dd7
Packit 4b6dd7
	/* We don't chain up with parent class get_query_uri because it uses
Packit 4b6dd7
	 *  GData protocol parameters and they aren't compatible with newest API family
Packit 4b6dd7
	 */
Packit 4b6dd7
#undef APPEND_SEP
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
/**
Packit 4b6dd7
 * gdata_freebase_query_new:
Packit 4b6dd7
 * @mql: a MQL query string
Packit 4b6dd7
 *
Packit 4b6dd7
 * Creates a new #GDataFreebaseQuery with the MQL query provided in @mql. MQL
Packit 4b6dd7
 * is a JSON-based query language, analogous to SPARQL. To learn more about MQL,
Packit 4b6dd7
 * see the <ulink type="http" url="https://developers.google.com/freebase/v1/mql-overview">
Packit 4b6dd7
 * MQL overview</ulink> and <ulink type="http" url="https://developers.google.com/freebase/v1/mql-cookbook">
Packit 4b6dd7
 * cookbook</ulink>.
Packit 4b6dd7
 *
Packit 4b6dd7
 * For detailed information on Freebase schemas, The <ulink type="http" url="http://www.freebase.com/schema">"Schema"
Packit 4b6dd7
 * section</ulink> on the main site allows for natural search and navigation through the multiple data properties and domains.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Return value: (transfer full): a new #GDataFreebaseQuery
Packit 4b6dd7
 *
Packit 4b6dd7
 * Since: 0.15.1
Packit 4b6dd7
 * Deprecated: 0.17.7: Google Freebase has been permanently shut down.
Packit 4b6dd7
 */
Packit 4b6dd7
GDataFreebaseQuery *
Packit 4b6dd7
gdata_freebase_query_new (const gchar *mql)
Packit 4b6dd7
{
Packit 4b6dd7
	g_return_val_if_fail (mql != NULL, NULL);
Packit 4b6dd7
Packit 4b6dd7
	return g_object_new (GDATA_TYPE_FREEBASE_QUERY, "q", mql, NULL);
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
/**
Packit 4b6dd7
 * gdata_freebase_query_new_from_variant:
Packit 4b6dd7
 * @variant: a variant containing the MQL query structure
Packit 4b6dd7
 *
Packit 4b6dd7
 * Creates a new #GDataFreebaseQuery with the MQL query provided in a serialized form as @variant
Packit 4b6dd7
 * of type "a{smv}" containing the JSON data tree of a MQL query. One convenient way
Packit 4b6dd7
 * to build the variant is using json_gvariant_serialize() from a #JsonNode. For more information
Packit 4b6dd7
 * about MQL, see gdata_freebase_query_new().
Packit 4b6dd7
 *
Packit 4b6dd7
 * #GDataFreebaseQuery takes ownership on @variant, if it has a floating reference, it will be sunk.
Packit 4b6dd7
 * Otherwise an extra reference will be added.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Return value: (transfer full): a new #GDataFreebaseQuery
Packit 4b6dd7
 *
Packit 4b6dd7
 * Since: 0.15.1
Packit 4b6dd7
 * Deprecated: 0.17.7: Google Freebase has been permanently shut down.
Packit 4b6dd7
 */
Packit 4b6dd7
GDataFreebaseQuery *
Packit 4b6dd7
gdata_freebase_query_new_from_variant (GVariant *variant)
Packit 4b6dd7
{
Packit 4b6dd7
	g_return_val_if_fail (variant != NULL, NULL);
Packit 4b6dd7
Packit 4b6dd7
	return g_object_new (GDATA_TYPE_FREEBASE_QUERY,
Packit 4b6dd7
			     "variant", g_variant_ref_sink (variant),
Packit 4b6dd7
			     NULL);
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
G_GNUC_END_IGNORE_DEPRECATIONS