diff -r 000000000000 -r 28c358053693 branches/gmyth-0.1b/src/gmyth_http.c --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/branches/gmyth-0.1b/src/gmyth_http.c Wed Feb 14 23:06:17 2007 +0000 @@ -0,0 +1,343 @@ +/** + * GMyth Library + * + * @file gmyth/gmyth_http.c + * + * @brief

GMythHttp class provides a wrapper to access + * data from the database using http+xml + * + * Copyright (C) 2006 INdT - Instituto Nokia de Tecnologia. + * @author Artur Duque de Souza <@indt.org.br> + * + */ +/* + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include +#include +#include +#include + +#include "gmyth_http.h" +#include "gmyth_debug.h" +#include "gmyth_socket.h" + + +xmlXPathObjectPtr +getnodeset(xmlDocPtr doc, xmlChar *xpath) +{ + + xmlXPathContextPtr context; + xmlXPathObjectPtr result; + + context = xmlXPathNewContext(doc); + result = xmlXPathEvalExpression(xpath, context); + + if(xmlXPathNodeSetIsEmpty(result->nodesetval)) + { + g_fprintf(stderr, "Error: No result at XPath\n"); + return NULL; + } + + xmlXPathFreeContext(context); + return result; +} + + +xmlDocPtr XMLParse (const char *content, int length) +{ + xmlDocPtr doc; /* the resulting document tree */ + + doc = xmlReadMemory(content, length, NULL, NULL, 0); + if (doc == NULL) + { + g_fprintf(stderr, "Error: Failed to parse XML document\n"); + return NULL; + } + + return doc; +} + +xmlXPathObjectPtr getXPath (xmlChar *xpath, xmlDocPtr doc) +{ + xmlXPathObjectPtr result; + result = getnodeset(doc, xpath); + return result; +} + +/** Retrieves the Progam List from the Channel + * + * @param nodeTab A pointer to a node inside the XML + * @return A GSList containing a list of all the programs + */ +GSList* get_Program_List(xmlNodePtr node) +{ + GSList* program_list = NULL; + + while (node != NULL) + { + if (g_ascii_strcasecmp((char *)node->name, "text") != 0) + { + GMythProgram* program = (GMythProgram*)g_malloc(sizeof(struct _GMythProgram)); + + program->title = g_string_new((char *)xmlGetProp(node, (xmlChar *)"title")); + program->subtitle = g_string_new((char *)xmlGetProp(node, (xmlChar *)"subtitle")); + program->catType = g_string_new((char *)xmlGetProp(node, (xmlChar *)"catType")); + program->category = g_string_new((char *)xmlGetProp(node, (xmlChar *)"category")); + + sscanf ((char *)xmlGetProp(node, (xmlChar *)"repeat"), "%d", &(program->repeat)); + + program->startTime = gmyth_util_string_to_time_val ((char *)xmlGetProp(node, \ + (xmlChar *)"startTime")); + program->endTime = gmyth_util_string_to_time_val ((char *)xmlGetProp(node, \ + (xmlChar *)"endTime")); + + program_list = g_slist_append(program_list, program); + } + + node = node->next; + } + + return program_list; +} + +/** Retrieves the Channel List from the ProgramGuide + * + * @param node A pointer to a node inside the XML + * @param epg The struct where is the current epg + * @return The epg from "param" updated + */ +void get_Channel_List (xmlNodePtr node, GMythEpg* epg) +{ + int i; + epg->channelList = NULL; + + for (i=1; i <= epg->numOfChannels; i++) + { + GMythChannel* channel = (GMythChannel*)g_malloc(sizeof(struct _GMythChannel)); + + channel->channelName = g_string_new((char *)xmlGetProp(node, (xmlChar *)"channelName")); + channel->chanNum = g_string_new((char *)xmlGetProp(node, (xmlChar *)"chanNum")); + + sscanf ((char *)xmlGetProp(node, (xmlChar *)"chanId"), "%d", &(channel->chanId)); + sscanf ((char *)xmlGetProp(node, (xmlChar *)"callSign"), "%d", &(channel->callSign)); + + channel->programList = get_Program_List(node->children); + + epg->channelList = g_slist_append(epg->channelList, channel); + } +} + +/** Retrieves the properties from the ProgramGuide + * + * @param nodeTab A pointer to a node inside the XML + * @param epg The struct where is the current epg + * @return The epg from "param" updated + */ +void get_ProgramGuide_Properties (xmlNodePtr nodeTab, GMythEpg* epg) +{ + sscanf ((char *)xmlGetProp(nodeTab, (xmlChar *)"startChanId"), "%d", &(epg->startChanId)); + sscanf ((char *)xmlGetProp(nodeTab, (xmlChar *)"endChanId"), "%d", &(epg->endChanId)); + + epg->version = g_string_new((char *)xmlGetProp(nodeTab, (xmlChar *)"version")); + + sscanf ((char *)xmlGetProp(nodeTab, (xmlChar *)"protoVer"), "%d", &(epg->protoVer)); + sscanf ((char *)xmlGetProp(nodeTab, (xmlChar *)"totalCount"), "%d", &(epg->totalCount)); + sscanf ((char *)xmlGetProp(nodeTab, (xmlChar *)"numOfChannels"), "%d", &(epg->numOfChannels)); + + epg->asOf = gmyth_util_string_to_time_val ((char *)xmlGetProp(nodeTab, (xmlChar *)"asOf")); + epg->startTime = gmyth_util_string_to_time_val ((char *)xmlGetProp(nodeTab, (xmlChar *)"startTime")); + epg->endTime = gmyth_util_string_to_time_val ((char *)xmlGetProp(nodeTab, (xmlChar *)"endTime")); + + sscanf ((char *)xmlGetProp(nodeTab, (xmlChar *)"details"), "%d", &(epg->details)); + + // go to Channel section and retrieve Channels and Programs + get_Channel_List(nodeTab->children->next->children->next, epg); +} + +/** Aux function to retrieve the Eletronic Program Guide + * + * @param doc An XML document (xmlDocPtr) + * @return The epg + */ +void getEpg (xmlDocPtr doc, GMythEpg* epg) +{ + xmlXPathObjectPtr result; + xmlNodeSetPtr nodeset; + xmlChar *keyword; + + int i; + result = getXPath((xmlChar *)"/*", doc); + + if (result) + { + nodeset = result->nodesetval; + for (i=0; i < nodeset->nodeNr; i++) + { + keyword = (xmlChar*)nodeset->nodeTab[i]->name; + if (g_ascii_strcasecmp((char *)keyword, "ProgramGuide") == 0) + get_ProgramGuide_Properties(nodeset->nodeTab[i], epg); + } + xmlXPathFreeObject (result); + } + +} + +/** Retrieves the Eletronic Program Guide from the backend + * + * @param doc An XML document (xmlDocPtr) + * @return The epg + */ +GMythEpg retrieve_epg (GMythBackendInfo *backend_info, int port, \ + GTimeVal* StartTime, GTimeVal* EndTime, \ + gint StartChanId, gint NumOfChannels, \ + gchar* Details) +{ + GMythEpg epg; + MemoryStruct chunk; + + chunk.memory=NULL; /* we expect realloc(NULL, size) to work */ + chunk.size = 0; /* no data at this point */ + + gchar* starttime = (gchar*)g_malloc(sizeof(gchar)*20); + starttime = gmyth_util_time_to_mythformat_from_time_val(StartTime); + + gchar* endtime = (gchar*)g_malloc(sizeof(gchar)*20); + endtime = gmyth_util_time_to_mythformat_from_time_val(EndTime); + + GString* command = g_string_new(""); + g_string_printf(command, "getProgramGuide?StartTime=%s&EndTime=%s&StartChanId=%d" + "&NumOfChannels=%d&Details=%s", starttime, endtime, \ + StartChanId, NumOfChannels, Details); + + chunk = gmyth_http_request(backend_info, command); + xmlDocPtr doc = XMLParse(chunk.memory, strlen(chunk.memory)); + getEpg(doc, &epg); + free(chunk.memory); + + return epg; +} + +/* Aux functions got from libcurl */ +void *myrealloc (void *ptr, size_t size) +{ + /* There might be a realloc() out there that doesn't like reallocing + NULL pointers, so we take care of it here */ + if(ptr) + return realloc(ptr, size); + else + return malloc(size); +} + +size_t +WriteMemoryCallback (void *ptr, size_t size, size_t nmemb, void *data) +{ + size_t realsize = size * nmemb; + MemoryStruct *mem = (struct _MemoryStruct *)data; + + mem->memory = (char *)myrealloc(mem->memory, mem->size + realsize + 1); + if (mem->memory) + { + memcpy(&(mem->memory[mem->size]), ptr, realsize); + mem->size += realsize; + mem->memory[mem->size] = 0; + } + + return realsize; +} + +/** Create a String containing the URL with the command + * + * @param command A string with the command + * @param variables Vars to use with their values \ + * MUST FINISH WITH NULL!!! + * @return The response + */ +GString* gmyth_http_create_command (GString *command, ...) +{ + va_list args; + va_start(args, command); + gchar* s = NULL; + + g_string_append(command, "?"); + + /* Sintax: var, value */ + while ( (s = va_arg(args, gchar *)) != NULL ) + { + g_string_append(command, s); + g_string_append(command, "="); + s = va_arg(args, gchar *); + g_string_append(command, s); + g_string_append(command, "&"); + } + + free(s); + va_end(args); + + return command; +} + +/** Send HTTP Command and receives the result of it + * + * @return A string with the response from the server + * NULL if there is no response. + */ +MemoryStruct gmyth_http_request (GMythBackendInfo *backend_info, GString *command) +{ + LIBXML_TEST_VERSION + + size_t size = strlen(backend_info->hostname) + strlen(command->str) + 13; + gchar *URL = (gchar *)g_malloc(sizeof(gchar)*size); + g_snprintf(URL, size+1, "http://%s:6544/%s", backend_info->hostname, command->str); + + CURL *curl_handle; + + MemoryStruct chunk; + + chunk.memory=NULL; /* we expect realloc(NULL, size) to work */ + chunk.size = 0; /* no data at this point */ + + curl_global_init(CURL_GLOBAL_ALL); + + /* init the curl session */ + curl_handle = curl_easy_init(); + + /* specify URL to get */ + curl_easy_setopt(curl_handle, CURLOPT_URL, URL); + + /* send all data to this function */ + curl_easy_setopt(curl_handle, CURLOPT_WRITEFUNCTION, WriteMemoryCallback); + + /* we pass our 'chunk' struct to the callback function */ + curl_easy_setopt(curl_handle, CURLOPT_WRITEDATA, (void *)&chunk); + + /* some servers don't like requests that are made without a user-agent + field, so we provide one */ + curl_easy_setopt(curl_handle, CURLOPT_USERAGENT, "libcurl-agent/1.0"); + + /* get it! */ + curl_easy_perform(curl_handle); + + /* cleanup curl stuff */ + curl_easy_cleanup(curl_handle); + + return chunk; +}