1
0
mirror of https://github.com/haiwen/libsearpc.git synced 2025-08-08 07:47:18 +00:00
libsearpc/searpc-client.c
2011-04-08 20:58:15 +08:00

218 lines
5.4 KiB
C

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <json-glib/json-glib.h>
#include "searpc-client.h"
#include "searpc-utils.h"
SearpcClient *
searpc_client_new ()
{
return g_new0 (SearpcClient, 1);
}
void
searpc_client_free (SearpcClient *client)
{
if (!client)
return;
g_free (client);
}
char *
searpc_client_transport_send (SearpcClient *client,
const gchar *fcall_str,
size_t fcall_len,
size_t *ret_len)
{
return client->transport(client->arg, fcall_str,
fcall_len, ret_len);
}
/*
* serialize function call from array to string
*/
static char *
fcall_common (JsonArray *array, gsize *len)
{
gchar *data;
JsonGenerator *gen = json_generator_new ();
JsonNode *root;
root = json_node_new (JSON_NODE_ARRAY);
json_node_take_array (root, array);
json_generator_set_root (gen, root);
g_object_set (gen, "pretty", FALSE, NULL);
data = json_generator_to_data (gen, len);
json_node_free (root);
g_object_unref (gen);
return data;
}
#include "fcall-impr.h"
/*
* Returns -1 if error happens in parsing data or data contains error
* message. In this case, the calling function should simply return
* to up layer.
*
* Returns 0 otherwise, and root is set to the root node, object set
* to the root node's containing object.
*/
static int
handle_ret_common (char *data, size_t len, JsonParser **parser,
JsonNode **root,
JsonObject **object, GError **error)
{
gint err_code;
const gchar *err_msg;
g_return_val_if_fail (root != 0 || object != 0, -1);
*parser = json_parser_new ();
if (!json_parser_load_from_data (*parser, data, strlen(data), error)) {
g_object_unref (*parser);
*parser = NULL;
return -1;
}
*root = json_parser_get_root (*parser);
*object = json_node_get_object (*root);
if (*object == NULL) {
g_set_error (error, 0, 502, "Invalid data: not a object");
g_object_unref (*parser);
*parser = NULL;
*root = NULL;
return -1;
}
if (json_object_has_member (*object, "err_code")) {
err_code = json_object_get_int_member (*object, "err_code");
err_msg = json_object_get_string_or_null_member (*object, "err_msg");
g_set_error (error, 0, err_code, "%s", err_msg);
g_object_unref (*parser);
*parser = NULL;
*object = NULL;
*root = NULL;
return -1;
}
return 0;
}
char *
searpc_client_fret__string (char *data, size_t len, GError **error)
{
JsonParser *parser = NULL;
JsonObject *object = NULL;
JsonNode *root = NULL;
gchar *ret_str = NULL;
if (handle_ret_common(data, len, &parser, &root, &object, error) == 0) {
ret_str = g_strdup (
json_object_get_string_or_null_member (object, "ret"));
g_object_unref (parser);
return ret_str;
}
return NULL;
}
int
searpc_client_fret__int (char *data, size_t len, GError **error)
{
JsonParser *parser = NULL;
JsonNode *root = NULL;
JsonObject *object = NULL;
int ret;
if (handle_ret_common(data, len, &parser, &root, &object, error) == 0) {
ret = (int) json_object_get_int_member(object, "ret");
g_object_unref (parser);
return ret;
}
return -1;
}
GObject*
searpc_client_fret__object (GType gtype, char *data, size_t len, GError **error)
{
JsonParser *parser = NULL;
JsonNode *root = NULL;
JsonObject *object = NULL;
GObject *ret = NULL;
JsonNode *member;
if (handle_ret_common(data, len, &parser, &root, &object, error) == 0) {
member = json_object_get_member (object, "ret");
if (json_node_get_node_type(member) == JSON_NODE_NULL) {
g_object_unref (parser);
return NULL;
}
ret = json_gobject_deserialize(gtype, member);
g_object_unref (parser);
return ret;
}
return NULL;
}
static void clean_objlist(GList *list)
{
GList *ptr;
for (ptr = list; ptr; ptr = ptr->next)
g_object_unref(ptr->data);
g_list_free (list);
}
GList*
searpc_client_fret__objlist (GType gtype, char *data, size_t len, GError **error)
{
JsonParser *parser = NULL;
JsonNode *root = NULL;
JsonObject *object = NULL;
JsonArray *array;
JsonNode *member;
GList *ret = NULL;
if (handle_ret_common(data, len, &parser, &root, &object, error) == 0) {
member = json_object_get_member (object, "ret");
if (json_node_get_node_type(member) == JSON_NODE_NULL) {
g_object_unref (parser);
return NULL;
}
array = json_node_get_array (member);
g_assert (array);
int i;
for (i = 0; i < json_array_get_length(array); i++) {
JsonNode *member = json_array_get_element (array, i);
GObject *obj = json_gobject_deserialize(gtype, member);
if (obj == NULL) {
g_set_error (error, 0, 503,
"Invalid data: object list contains null");
clean_objlist(ret);
g_object_unref (parser);
return NULL;
}
ret = g_list_prepend (ret, obj);
}
g_object_unref (parser);
return g_list_reverse(ret);
}
return NULL;
}