Surfer
/* $Id$ */
/* Copyright (c) 2010 Pierre Pronchery <khorben@defora.org> */
/* This file is part of DeforaOS Desktop Surfer */
/* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, version 3 of the License.
*
* 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 General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>. */
#include <string.h>
#include <libintl.h>
#define GNET_EXPERIMENTAL
#include <gnet.h>
#ifndef _
# define _(string) gettext(string)
#endif
/* Conn */
/* private */
/* types */
typedef struct _Conn Conn;
struct _Conn
{
Surfer * surfer;
char * url;
char const * anchor;
guint64 content_length;
guint64 data_received;
gdouble progress;
char * status;
int direct;
int image;
/* callback */
ssize_t (*callback_write)(Conn *, char const *, size_t, gpointer);
gpointer callback_write_data;
/* http */
GConnHttp * http;
GConnHttpMethod http_method;
gchar * http_post;
};
/* prototypes */
static Conn * _conn_new(Surfer * surfer, char const * url,
char const * post);
static void _conn_delete(Conn * conn);
/* accessors */
static gdouble _conn_get_progress(Conn * conn);
static char const * _conn_get_status(Conn * conn);
static void _conn_set_progress(Conn * conn, gdouble progress);
static void _conn_set_status(Conn * conn, char const * status);
static void _conn_set_callback_write(Conn * conn,
ssize_t (*callback)(Conn *, char const *, size_t, gpointer),
gpointer data);
/* useful */
/* conn_load */
static int _conn_load(Conn * conn);
/* functions */
/* conn_new */
static ssize_t _new_callback_write(Conn * conn, char const * buf, size_t size,
gpointer data);
static Conn * _conn_new(Surfer * surfer, char const * url, char const * post)
{
Conn * conn;
char * p;
if((conn = malloc(sizeof(*conn))) == NULL)
return NULL;
conn->surfer = surfer;
conn->url = strdup(url);
conn->status = NULL;
conn->http_post = (post != NULL) ? strdup(post) : NULL;
if(conn->url == NULL || (post != NULL && conn->http_post == NULL))
{
_conn_delete(conn);
return NULL;
}
if((p = strrchr(conn->url, '#')) != NULL)
{
*(p++) = '\0';
conn->anchor = p;
}
conn->content_length = 0;
conn->data_received = 0;
conn->progress = -1.0;
conn->direct = 0;
conn->image = 0;
conn->callback_write = _new_callback_write;
conn->callback_write_data = stdout;
conn->http = NULL;
conn->http_method = (post == NULL) ? GNET_CONN_HTTP_METHOD_GET
: GNET_CONN_HTTP_METHOD_POST;
return conn;
}
static ssize_t _new_callback_write(Conn * conn, char const * buf, size_t size,
gpointer data)
{
ssize_t ret;
if((ret = fwrite(buf, sizeof(*buf), size, data)) != size)
return -1;
return ret;
}
/* conn_delete */
static void _conn_delete(Conn * conn)
{
free(conn->url);
free(conn->status);
free(conn->http_post);
free(conn);
}
/* accessors */
/* conn_get_progress */
static gdouble _conn_get_progress(Conn * conn)
{
return conn->progress;
}
/* conn_get_status */
static char const * _conn_get_status(Conn * conn)
{
return conn->status;
}
/* conn_set_progress */
static void _conn_set_progress(Conn * conn, gdouble progress)
{
conn->progress = progress;
surfer_set_progress(conn->surfer, progress);
}
/* conn_set_status */
static void _conn_set_status(Conn * conn, char const * status)
{
free(conn->status);
/* XXX may fail */
conn->status = (status != NULL) ? strdup(status) : NULL;
surfer_set_status(conn->surfer, status);
}
/* conn_set_callback_write */
static void _conn_set_callback_write(Conn * conn,
ssize_t (*callback)(Conn *, char const *, size_t, gpointer),
gpointer data)
{
conn->callback_write = callback;
conn->callback_write_data = data;
}
/* useful */
/* conn_load */
static void _load_watch_http(GConnHttp * connhttp, GConnHttpEvent * event,
gpointer data);
static void _http_connected(Conn * conn);
static void _http_data_complete(GConnHttpEventData * event, Conn * conn);
static void _http_data_partial(GConnHttpEventData * event, Conn * conn);
static void _http_data_progress(GConnHttpEventData * event, Conn * conn);
static void _http_error(GConnHttpEventError * event, Conn * conn);
static void _http_redirect(GConnHttpEventRedirect * event, Conn * conn);
static void _http_resolved(GConnHttpEventResolved * event, Conn * conn);
static void _http_response(GConnHttpEventResponse * event, Conn * conn);
static void _http_timeout(Conn * conn);
static int _conn_load(Conn * conn)
{
static const char http[] = "http:";
if(strncmp(conn->url, http, sizeof(http) - 1) != 0)
{
/* FIXME support "file:", report error */
return 1;
}
_conn_set_progress(conn, -1.0);
_conn_set_status(conn, _("Resolving..."));
conn->http = gnet_conn_http_new();
gnet_conn_http_set_uri(conn->http, conn->url);
gnet_conn_http_set_user_agent(conn->http, "DeforaOS " PACKAGE);
gnet_conn_http_set_method(conn->http, conn->http_method,
conn->http_post, (conn->http_post != NULL)
? strlen(conn->http_post) : 0);
gnet_conn_http_run_async(conn->http, _load_watch_http, conn);
return 0;
}
static void _load_watch_http(GConnHttp * connhttp, GConnHttpEvent * event,
gpointer data)
{
Conn * conn = data;
if(conn->http != connhttp)
return; /* FIXME shouldn't happen, but report error */
switch(event->type)
{
case GNET_CONN_HTTP_CONNECTED:
_http_connected(conn);
break;
case GNET_CONN_HTTP_DATA_COMPLETE:
_http_data_complete((GConnHttpEventData*)event, conn);
return;
case GNET_CONN_HTTP_DATA_PARTIAL:
_http_data_partial((GConnHttpEventData*)event, conn);
return;
case GNET_CONN_HTTP_ERROR:
_http_error((GConnHttpEventError*)event, conn);
return;
case GNET_CONN_HTTP_REDIRECT:
_http_redirect((GConnHttpEventRedirect*)event, conn);
return;
case GNET_CONN_HTTP_RESOLVED:
_http_resolved((GConnHttpEventResolved*)event, conn);
return;
case GNET_CONN_HTTP_RESPONSE:
_http_response((GConnHttpEventResponse*)event, conn);
return;
case GNET_CONN_HTTP_TIMEOUT:
_http_timeout(conn);
return;
}
}
static void _http_connected(Conn * conn)
{
_conn_set_status(conn, _("Connected"));
}
static void _http_data_complete(GConnHttpEventData * event, Conn * conn)
{
gchar * buf;
gsize size;
if(gnet_conn_http_steal_buffer(conn->http, &buf, &size) != TRUE)
{
/* FIXME report error */
_conn_set_progress(conn, -1.0);
_conn_set_status(conn, NULL);
return;
}
conn->callback_write(conn, buf, size, conn->callback_write_data);
/* FIXME find a more elegant way? */
conn->callback_write(conn, NULL, 0, conn->callback_write_data);
_http_data_progress(event, conn);
_conn_set_status(conn, NULL);
}
static void _http_data_partial(GConnHttpEventData * event, Conn * conn)
{
gchar * buf;
gsize size;
_conn_set_status(conn, _("Downloading..."));
if(gnet_conn_http_steal_buffer(conn->http, &buf, &size) != TRUE)
/* FIXME report error */
return;
conn->callback_write(conn, buf, size, conn->callback_write_data);
_http_data_progress(event, conn);
}
static void _http_data_progress(GConnHttpEventData * event, Conn * conn)
{
gdouble fraction;
conn->data_received = event->data_received;
conn->content_length = event->content_length;
fraction = conn->data_received;
_conn_set_progress(conn, fraction / conn->content_length);
}
static void _http_error(GConnHttpEventError * event, Conn * conn)
{
char * msg;
switch(event->code)
{
case GNET_CONN_HTTP_ERROR_PROTOCOL_UNSUPPORTED:
msg = _("Unsupported protocol");
break;
#if GNET_CHECK_VERSION(2, 0, 8) /* XXX unsure about the exact version */
case GNET_CONN_HTTP_ERROR_HOSTNAME_RESOLUTION:
msg = _("Unknown host");
break;
#endif
case GNET_CONN_HTTP_ERROR_UNSPECIFIED:
default:
msg = _("Unspecified error");
break;
}
_conn_set_progress(conn, -1.0);
_conn_set_status(conn, NULL);
surfer_error(conn->surfer, msg, 0);
}
static void _http_redirect(GConnHttpEventRedirect * event, Conn * conn)
{
char buf[256] = "Redirecting...";
char * url = event->new_location;
if(url == NULL)
{
_conn_set_status(conn, buf);
return;
}
/* FIXME implement */
_conn_set_status(conn, buf);
}
static void _http_resolved(GConnHttpEventResolved * event, Conn * conn)
{
/* FIXME implement */
}
static void _http_response(GConnHttpEventResponse * event, Conn * conn)
{
/* FIXME implement */
}
static void _http_timeout(Conn * conn)
{
surfer_error(conn->surfer, _("Timeout"), 0);
_conn_set_progress(conn, -1.0);
_conn_set_status(conn, NULL);
}