#include <stdlib.h>
#include <glib.h>
#include <gmodule.h>
#include <check.h>
#include <string.h>
#include <stdio.h>
#include "irc.h"
#include "testsuite.h"
START_TEST(test_connect)
GIOChannel *ch1, *ch2;
irc_t *irc;
char *raw;
fail_unless(g_io_channel_pair(&ch1, &ch2));
irc = irc_new(g_io_channel_unix_get_fd(ch1));
irc_free(irc);
fail_unless(g_io_channel_read_to_end(ch2, &raw, NULL, NULL) == G_IO_STATUS_NORMAL);
fail_if(strcmp(raw, "") != 0);
g_free(raw);
END_TEST
START_TEST(test_login)
GIOChannel *ch1, *ch2;
irc_t *irc;
GError *error = NULL;
char *raw;
fail_unless(g_io_channel_pair(&ch1, &ch2));
g_io_channel_set_flags(ch1, G_IO_FLAG_NONBLOCK, NULL);
g_io_channel_set_flags(ch2, G_IO_FLAG_NONBLOCK, NULL);
irc = irc_new(g_io_channel_unix_get_fd(ch1));
fail_unless(g_io_channel_write_chars(ch2, "NICK bla\r\n"
"USER a a a a\r\n", -1, NULL, NULL) == G_IO_STATUS_NORMAL);
fail_unless(g_io_channel_flush(ch2, NULL) == G_IO_STATUS_NORMAL);
g_main_iteration(FALSE);
irc_free(irc);
fail_unless(g_io_channel_read_to_end(ch2, &pre { line-height: 125%; }
td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; }
span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; }
td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; }
span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; }
.highlight .hll { background-color: #ffffcc }
.highlight .c { color: #888888 } /* Comment */
.highlight .err { color: #a61717; background-color: #e3d2d2 } /* Error */
.highlight .k { color: #008800; font-weight: bold } /* Keyword */
.highlight .ch { color: #888888 } /* Comment.Hashbang */
.highlight .cm { color: #888888 } /* Comment.Multiline */
.highlight .cp { color: #cc0000; font-weight: bold } /* Comment.Preproc */
.highlight .cpf { color: #888888 } /* Comment.PreprocFile */
.highlight .c1 { color: #888888 } /* Comment.Single */
.highlight .cs { color: #cc0000; font-weight: bold; background-color: #fff0f0 } /* Comment.Special */
.highlight .gd { color: #000000; background-color: #ffdddd } /* Generic.Deleted */
.highlight .ge { font-style: italic } /* Generic.Emph */
.highlight .ges { font-weight: bold; font-style: italic } /* Generic.EmphStrong */
.highlight .gr { color: #aa0000 } /* Generic.Error */
.highlight .gh { color: #333333 } /* Generic.Heading */
.highlight .gi { color: #000000; background-color: #ddffdd } /* Generic.Inserted */
.highlight .go { color: #888888 } /* Generic.Output */
.highlight .gp { color: #555555 } /* Generic.Prompt */
.highlight .gs { font-weight: bold } /* Generic.Strong */
.highlight .gu { color: #666666 } /* Generic.Subheading */
.highlight .gt { color: #aa0000 } /* Generic.Traceback */
.highlight .kc { color: #008800; font-weight: bold } /* Keyword.Constant */
.highlight .kd { color: #008800; font-weight: bold } /* Keyword.Declaration */
.highlight .kn { color: #008800; font-weight: bold } /* Keyword.Namespace */
.highlight .kp { color: #008800 } /* Keyword.Pseudo */
.highlight .kr { color: #008800; font-weight: bold } /* Keyword.Reserved */
.highlight .kt { color: #888888; font-weight: bold } /* Keyword.Type */
.highlight .m { color: #0000DD; font-weight: bold } /* Literal.Number */
.highlight .s { color: #dd2200; background-color: #fff0f0 } /* Literal.String */
.highlight .na { color: #336699 } /* Name.Attribute */
.highlight .nb { color: #003388 } /* Name.Builtin */
.highlight .nc { color: #bb0066; font-weight: bold } /* Name.Class */
.highlight .no { color: #003366; font-weight: bold } /* Name.Constant */
.highlight .nd { color: #555555 } /* Name.Decorator */
.highlight .ne { color: #bb0066; font-weight: bold } /* Name.Exception */
.highlight .nf { color: #0066bb; font-weight: bold } /* Name.Function */
.highlight .nl { color: #336699; font-style: italic } /* Name.Label */
.highlight .nn { color: #bb0066; font-weight: bold } /* Name.Namespace */
.highlight .py { color: #336699; font-weight: bold } /* Name.Property */
.highlight .nt { color: #bb0066; font-weight: bold } /* Name.Tag */
.highlight .nv { color: #336699 } /* Name.Variable */
.highlight .ow { color: #008800 } /* Operator.Word */
.highlight .w { color: #bbbbbb } /* Text.Whitespace */
.highlight .mb { color: #0000DD; font-weight: bold } /* Literal.Number.Bin */
.highlight .mf { color: #0000DD; font-weight: bold } /* Literal.Number.Float */
.highlight .mh { color: #0000DD; font-weight: bold } /* Literal.Number.Hex */
.highlight .mi { color: #0000DD; font-weight: bold } /* Literal.Number.Integer */
.highlight .mo { color: #0000DD; font-weight: bold } /* Literal.Number.Oct */
.highlight .sa { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Affix */
.highlight .sb { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Backtick */
.highlight .sc { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Char */
.highlight .dl { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Delimiter */
.highlight .sd { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Doc */
.highlight .s2 { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Double */
.highlight .se { color: #0044dd; background-color: #fff0f0 } /* Literal.String.Escape */
.highlight .sh { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Heredoc */
.highlight .si { color: #3333bb; background-color: #fff0f0 } /* Literal.String.Interpol */
.highlight .sx { color: #22bb22; background-color: #f0fff0 } /* Literal.String.Other */
.highlight .sr { color: #008800; background-color: #fff0ff } /* Literal.String.Regex */
.highlight .s1 { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Single */
.highlight .ss { color: #aa6600; background-color: #fff0f0 } /* Literal.String.Symbol */
.highlight .bp { color: #003388 } /* Name.Builtin.Pseudo */
.highlight .fm { color: #0066bb; font-weight: bold } /* Name.Function.Magic */
.highlight .vc { color: #336699 } /* Name.Variable.Class */
.highlight .vg { color: #dd7700 } /* Name.Variable.Global */
.highlight .vi { color: #3333bb } /* Name.Variable.Instance */
.highlight .vm { color: #336699 } /* Name.Variable.Magic */
.highlight .il { color: #0000DD; font-weight: bold } /* Literal.Number.Integer.Long *//*
* libyahoo2: yahoo_httplib.c
*
* Copyright (C) 2002-2004, Philip S Tellis <philip.tellis AT gmx.net>
*
* 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; 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 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
*
*/
#include <stdio.h>
#include <stdlib.h>
#if STDC_HEADERS
# include <string.h>
#else
# if !HAVE_STRCHR
# define strchr index
# define strrchr rindex
# endif
char *strchr (), *strrchr ();
# if !HAVE_MEMCPY
# define memcpy(d, s, n) bcopy ((s), (d), (n))
# define memmove(d, s, n) bcopy ((s), (d), (n))
# endif
#endif
#include <errno.h>
#ifndef _WIN32
#include <unistd.h>
#endif
#include <ctype.h>
#include "yahoo2.h"
#include "yahoo2_callbacks.h"
#include "yahoo_httplib.h"
#include "yahoo_util.h"
#include "yahoo_debug.h"
#ifdef __MINGW32__
# include <winsock2.h>
# define write(a,b,c) send(a,b,c,0)
# define read(a,b,c) recv(a,b,c,0)
# define snprintf _snprintf
#endif
#ifdef USE_STRUCT_CALLBACKS
extern struct yahoo_callbacks *yc;
#define YAHOO_CALLBACK(x) yc->x
#else
#define YAHOO_CALLBACK(x) x
#endif
extern enum yahoo_log_level log_level;
int yahoo_tcp_readline(char *ptr, int maxlen, int fd)
{
int n, rc;
char c;
for (n = 1; n < maxlen; n++) {
do {
rc = read(fd, &c, 1);
} while(rc == -1 && (errno == EINTR || errno == EAGAIN)); /* this is bad - it should be done asynchronously */
if (rc == 1) {
if(c == '\r') /* get rid of \r */
continue;
*ptr = c;
if (c == '\n')
break;
ptr++;
} else if (rc == 0) {
if (n == 1)
return (0); /* EOF, no data */
else
break; /* EOF, w/ data */
} else {
return -1;
}
}
*ptr = 0;
return (n);
}
static int url_to_host_port_path(const char *url,
char *host, int *port, char *path)
{
char *urlcopy=NULL;
char *slash=NULL;
char *colon=NULL;
/*
* http://hostname
* http://hostname/
* http://hostname/path
* http://hostname/path:foo
* http://hostname:port
* http://hostname:port/
* http://hostname:port/path
* http://hostname:port/path:foo
*/
if(strstr(url, "http://") == url) {
urlcopy = strdup(url+7);
} else {
WARNING(("Weird url - unknown protocol: %s", url));
return 0;
}
slash = strchr(urlcopy, '/');
colon = strchr(urlcopy, ':');
if(!colon || (slash && slash < colon)) {
*port = 80;
} else {
*colon = 0;
*port = atoi(colon+1);
}
if(!slash) {
strcpy(path, "/");
} else {
strcpy(path, slash);
*slash = 0;
}
strcpy(host, urlcopy);
FREE(urlcopy);
return 1;
}
static int isurlchar(unsigned char c)
{
return (isalnum(c) || '-' == c || '_' == c);
}
char *yahoo_urlencode(const char *instr)
{
int ipos=0, bpos=0;
char *str = NULL;
int len = strlen(instr);
if(!(str = y_new(char, 3*len + 1) ))
return "";
while(instr[ipos]) {
while(isurlchar(instr[ipos]))
str[bpos++] = instr[ipos++];
if(!instr[ipos])
break;
snprintf(&str[bpos], 4, "%%%.2x", instr[ipos]);
bpos+=3;
ipos++;
}
str[bpos]='\0';
/* free extra alloc'ed mem. */
len = strlen(str);
str = y_renew(char, str, len+1);
return (str);
}
char *yahoo_urldecode(const char *instr)
{
int ipos=0, bpos=0;
char *str = NULL;
char entity[3]={0,0,0};
unsigned dec;
int len = strlen(instr);
if(!(str = y_new(char, len+1) ))
return "";
while(instr[ipos]) {
while(instr[ipos] && instr[ipos]!='%')
if(instr[ipos]=='+') {
str[bpos++]=' ';
ipos++;
} else
str[bpos++] = instr[ipos++];
if(!instr[ipos])
break;
if(instr[ipos+1] && instr[ipos+2]) {
ipos++;
entity[0]=instr[ipos++];
entity[1]=instr[ipos++];
sscanf(entity, "%2x", &dec);
str[bpos++] = (char)dec;
} else {
str[bpos++] = instr[ipos++];
}
}
str[bpos]='\0';
/* free extra alloc'ed mem. */
len = strlen(str);
str = y_renew(char, str, len+1);
return (str);
}
char *yahoo_xmldecode(const char *instr)
{
int ipos=0, bpos=0, epos=0;
char *str = NULL;
char entity[4]={0,0,0,0};
char *entitymap[5][2]={
{"amp;", "&"},
{"quot;", "\""},
{"lt;", "<"},
{"gt;", "<"},
{"nbsp;", " "}
};
unsigned dec;
int len = strlen(instr);
if(!(str = y_new(char, len+1) ))
return "";
while(instr[ipos]) {
while(instr[ipos] && instr[ipos]!='&')
if(instr[ipos]=='+') {
str[bpos++]=' ';
ipos++;
} else
str[bpos++] = instr[ipos++];
if(!instr[ipos] || !instr[ipos+1])
break;
ipos++;
if(instr[ipos] == '#') {
ipos++;
epos=0;
while(instr[ipos] != ';')
entity[epos++]=instr[ipos++];
sscanf(entity, "%u", &dec);
str[bpos++] = (char)dec;
ipos++;
} else {
int i;
for (i=0; i<5; i++)
if(!strncmp(instr+ipos, entitymap[i][0],
strlen(entitymap[i][0]))) {
str[bpos++] = entitymap[i][1][0];
ipos += strlen(entitymap[i][0]);
break;
}
}
}
str[bpos]='\0';
/* free extra alloc'ed mem. */
len = strlen(str);
str = y_renew(char, str, len+1);
return (str);
}
typedef void (*http_connected)(int id, int fd, int error);
struct callback_data {
int id;
yahoo_get_fd_callback callback;
char *request;
void *user_data;
};
static void connect_complete(int fd, int error, void *data)
{
struct callback_data *ccd = data;
if(error == 0 && fd > 0)
write(fd, ccd->request, strlen(ccd->request));
FREE(ccd->request);
ccd->callback(ccd->id, fd, error, ccd->user_data);
FREE(ccd);
}
static void yahoo_send_http_request(int id, char *host, int port, char *request,
yahoo_get_fd_callback callback, void *data)
{
struct callback_data *ccd=y_new0(struct callback_data, 1);
ccd->callback = callback;
ccd->id = id;
ccd->request = strdup(request);
ccd->user_data = data;
YAHOO_CALLBACK(ext_yahoo_connect_async)(id, host, port, connect_complete, ccd);
}
void yahoo_http_post(int id, const char *url, const char *cookies, long content_length,
yahoo_get_fd_callback callback, void *data)
{
char host[255];
int port = 80;
char path[255];
char buff[1024];
if(!url_to_host_port_path(url, host, &port, path))
return;
snprintf(buff, sizeof(buff),
"POST %s HTTP/1.0\r\n"
"Content-length: %ld\r\n"
"User-Agent: Mozilla/4.5 [en] (" PACKAGE "/" VERSION ")\r\n"
"Host: %s:%d\r\n"
"Cookie: %s\r\n"
"\r\n",
path, content_length,
host, port,
cookies);
yahoo_send_http_request(id, host, port, buff, callback, data);
}
void yahoo_http_get(int id, const char *url, const char *cookies,
yahoo_get_fd_callback callback, void *data)
{
char host[255];
int port = 80;
char path[255];
char buff[1024];
if(!url_to_host_port_path(url, host, &port, path))
return;
snprintf(buff, sizeof(buff),
"GET %s HTTP/1.0\r\n"
"Host: %s:%d\r\n"
"User-Agent: Mozilla/4.5 [en] (" PACKAGE "/" VERSION ")\r\n"
"Cookie: %s\r\n"
"\r\n",
path, host, port, cookies);
yahoo_send_http_request(id, host, port, buff, callback, data);
}
struct url_data {
yahoo_get_url_handle_callback callback;
void *user_data;
};
static void yahoo_got_url_fd(int id, int fd, int error, void *data)
{
char *tmp=NULL;
char buff[1024];
unsigned long filesize=0;
char *filename=NULL;
int n;
struct url_data *ud = data;
if(error || fd < 0) {
ud->callback(id, fd, error, filename, filesize, ud->user_data);
FREE(ud);
return;
}
while((n=yahoo_tcp_readline(buff, sizeof(buff), fd)) > 0) {
LOG(("Read:%s:\n", buff));
if(!strcmp(buff, ""))
break;
if( !strncasecmp(buff, "Content-length:",
strlen("Content-length:")) ) {
tmp = strrchr(buff, ' ');
if(tmp)
filesize = atol(tmp);
}
if( !strncasecmp(buff, "Content-disposition:",
strlen("Content-disposition:")) ) {
tmp = strstr(buff, "name=");
if(tmp) {
tmp+=strlen("name=");
if(tmp[0] == '"') {
char *tmp2;
tmp++;
tmp2 = strchr(tmp, '"');
if(tmp2)
*tmp2 = '\0';
} else {
char *tmp2;
tmp2 = strchr(tmp, ';');
if(!tmp2)
tmp2 = strchr(tmp, '\r');
if(!tmp2)
tmp2 = strchr(tmp, '\n');
if(tmp2)
*tmp2 = '\0';
}
filename = strdup(tmp);
}
}
}
LOG(("n == %d\n", n));
LOG(("Calling callback, filename:%s, size: %ld\n", filename, filesize));
ud->callback(id, fd, error, filename, filesize, ud->user_data);
FREE(ud);
FREE(filename);
}
void yahoo_get_url_fd(int id, const char *url, const struct yahoo_data *yd,
yahoo_get_url_handle_callback callback, void *data)
{
char buff[1024];
struct url_data *ud = y_new0(struct url_data, 1);
snprintf(buff, sizeof(buff), "Y=%s; T=%s", yd->cookie_y, yd->cookie_t);
ud->callback = callback;
ud->user_data = data;
yahoo_http_get(id, url, buff, yahoo_got_url_fd, ud);
}