diff plugins/yay/rxhandlers.c @ 1546:92b3dd1e4129

[gaim-migrate @ 1556] libyay. no more libyahoo. committer: Tailor Script <tailor@pidgin.im>
author Eric Warmenhoven <eric@warmenhoven.org>
date Thu, 08 Mar 2001 09:18:58 +0000
parents
children b669f48f0990
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/plugins/yay/rxhandlers.c	Thu Mar 08 09:18:58 2001 +0000
@@ -0,0 +1,305 @@
+/*
+ * libyay
+ *
+ * Copyright (C) 2001 Eric Warmenhoven <warmenhoven@yahoo.com>
+ * 
+ * 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 "internal.h"
+#include <string.h>
+#include <stdlib.h>
+#include <ctype.h>
+
+static int yahoo_parse_config(struct yahoo_session *session, struct yahoo_conn *conn, char *buf)
+{
+	char **str_array = g_strsplit(buf, "\n", 1024);
+	char **it;
+	int state = 0;
+
+	for (it = str_array; *it; it++) {
+		if (!strncmp(*it, "ERROR", strlen("ERROR"))) {
+			yahoo_close(session, conn);
+			CALLBACK(session, YAHOO_HANDLE_BADPASSWORD);
+			return 1;
+		} else if (!strncmp(*it, "Set-Cookie: ", strlen("Set-Cookie: "))) {
+			char **sa;
+			char **m;
+
+			char *end = strchr(*it, ';');
+			if (session->cookie)
+				g_free(session->cookie);
+			session->cookie = g_strndup(*it + strlen("Set-Cookie: "),
+					end - *it - strlen("Set-Cookie: "));
+			YAHOO_PRINT(session, YAHOO_LOG_DEBUG, session->cookie);
+			if (!session->cookie) {
+				yahoo_close(session, conn);
+				CALLBACK(session, YAHOO_HANDLE_DISCONNECT);
+				return 1;
+			}
+
+			sa = g_strsplit(session->cookie, "&", 8);
+			for (m = sa; *m; m++) {
+				if (!strncmp(*m, "n=", 2)) {
+					if (session->login_cookie)
+						g_free(session->login_cookie);
+					session->login_cookie = g_strdup(*m + 2);
+					YAHOO_PRINT(session, YAHOO_LOG_DEBUG, session->login_cookie);
+				}
+			}
+			g_strfreev(sa);
+		} else if (!strncmp(*it, "BEGIN BUDDYLIST", strlen("BEGIN BUDDYLIST"))) {
+			state = 1;
+		} else if (!strncmp(*it, "END BUDDYLIST", strlen("END BUDDYLIST"))) {
+			state = 0;
+		} else if (!strncmp(*it, "BEGIN IGNORELIST", strlen("BEGIN IGNORELIST"))) {
+			state = 2;
+		} else if (!strncmp(*it, "END IGNORELIST", strlen("END IGNORELIST"))) {
+			state = 0;
+		} else if (!strncmp(*it, "BEGIN IDENTITIES", strlen("BEGIN IDENTITIES"))) {
+			state = 3;
+		} else if (!strncmp(*it, "END IDENTITIES", strlen("END IDENTITIES"))) {
+			state = 0;
+		} else if (!strncmp(*it, "Mail=", strlen("Mail="))) {
+			session->mail = atoi(*it + strlen("Mail="));
+		} else if (!strncmp(*it, "Login=", strlen("Login="))) {
+			if (session->login)
+				g_free(session->login);
+			session->login = g_strdup(*it + strlen("Login="));
+		} else {
+			if (state == 1) {
+				struct yahoo_group *grp = g_new0(struct yahoo_group, 1);
+				char *end = strchr(*it, ':');
+				grp->name = g_strndup(*it, end - *it);
+				end++;
+				grp->buddies = g_strsplit(end, ",", 1024);
+				session->groups = g_list_append(session->groups, grp);
+			} else if (state == 2) {
+				session->ignored = g_list_append(session->ignored, g_strdup(*it));
+			} else if (state == 3) {
+				session->identities = g_strsplit(*it, ",", 6);
+			}
+		}
+	}
+
+	g_strfreev(str_array);
+	yahoo_close(session, conn);
+	CALLBACK(session, YAHOO_HANDLE_LOGINCOOKIE);
+	return 0;
+}
+
+static void yahoo_parse_status(struct yahoo_session *sess, struct yahoo_packet *pkt)
+{
+	char *tmp = pkt->content;
+	int count = 0;
+	char **strs;
+	int i;
+
+	YAHOO_PRINT(sess, YAHOO_LOG_DEBUG, pkt->content);
+
+	if (strstr(pkt->content, "was not AWAY"))
+		return;
+
+	while (*tmp && isdigit((int)*tmp))
+		count = count * 10 + *tmp++ - '0';
+	if (*tmp == ',')
+		tmp++;
+	count = count ? count : 1;
+
+	if (count > 1)
+		strs = g_strsplit(tmp, "),", count);
+	else
+		strs = g_strsplit(tmp, ")", count);
+
+	for (i = 0; i < count && strs[i]; i++) {
+		char **vals;
+		char *end, *who;
+		char **it;
+		int c, j;
+
+		who = strs[i];
+		end = strchr(who, '(');
+		*end++ = '\0';
+
+		vals = g_strsplit(end, ",", 1024);
+
+		for (it = vals, c = 0; *it; it++, c++);
+		if (c > 6)
+			end = g_strdup(vals[1]);
+		for (j = 2; j < c - 5; j++) {
+			char *x = end;
+			end = g_strconcat(end, ",", vals[j], NULL);
+			g_free(x);
+		}
+
+		CALLBACK(sess, YAHOO_HANDLE_STATUS, who, atoi(vals[0]), end,
+				atoi(vals[c - 3]), atoi(vals[c - 2]), atoi(vals[c - 1]));
+
+		if (c > 6)
+			g_free(end);
+		g_strfreev(vals);
+	}
+
+	g_strfreev(strs);
+}
+
+static void yahoo_parse_message(struct yahoo_session *sess, struct yahoo_packet *pkt)
+{
+	char buf[256];
+	int type = yahoo_makeint(pkt->msgtype);
+	char **str_array;
+	switch(type) {
+	case YAHOO_MESSAGE_NORMAL:
+		str_array = g_strsplit(pkt->content, ",,", 2);
+		CALLBACK(sess, YAHOO_HANDLE_MESSAGE, pkt->nick2, str_array[0], str_array[1]);
+		g_strfreev(str_array);
+		break;
+	default:
+		g_snprintf(buf, sizeof(buf), "unhandled message type %d: %s", type, pkt->content);
+		YAHOO_PRINT(sess, YAHOO_LOG_WARNING, buf);
+		break;
+	}
+}
+
+static void yahoo_parse_packet(struct yahoo_session *sess,
+		struct yahoo_conn *conn, struct yahoo_packet *pkt)
+{
+	char buf[256];
+	int service = yahoo_makeint(pkt->service);
+	conn->magic_id = yahoo_makeint(pkt->magic_id);
+	g_snprintf(buf, sizeof(buf), "Service %d (msgtype %d)", service, yahoo_makeint(pkt->msgtype));
+	YAHOO_PRINT(sess, YAHOO_LOG_DEBUG, buf);
+	switch(service) {
+	case YAHOO_SERVICE_LOGON:
+		if (yahoo_makeint(pkt->msgtype) == 0)
+			CALLBACK(sess, YAHOO_HANDLE_ONLINE);
+	case YAHOO_SERVICE_LOGOFF:
+	case YAHOO_SERVICE_ISAWAY:
+	case YAHOO_SERVICE_ISBACK:
+		yahoo_parse_status(sess, pkt);
+		break;
+	case YAHOO_SERVICE_IDACT:
+		CALLBACK(sess, YAHOO_HANDLE_ACTIVATE, pkt->content);
+		break;
+	case YAHOO_SERVICE_MESSAGE:
+		yahoo_parse_message(sess, pkt);
+		break;
+	case YAHOO_SERVICE_NEWMAIL:
+		CALLBACK(sess, YAHOO_HANDLE_NEWMAIL, strlen(pkt->content) ? atoi(pkt->content) : 0);
+		break;
+	default:
+		g_snprintf(buf, sizeof(buf), "unhandled service type %d: %s", service, pkt->content);
+		YAHOO_PRINT(sess, YAHOO_LOG_WARNING, buf);
+		break;
+	}
+}
+
+void yahoo_socket_handler(struct yahoo_session *session, int socket, int type)
+{
+	int pos = 0;
+	struct yahoo_conn *conn;
+
+	if (!session)
+		return;
+
+	if (!(conn = yahoo_find_conn(session, socket)))
+		return;
+
+	if (type == YAHOO_SOCKET_WRITE) {
+		int error = ETIMEDOUT, len = sizeof(error);
+
+		if (getsockopt(socket, SOL_SOCKET, SO_ERROR, &error, &len) < 0)
+			error = errno;
+		if (error) {
+			yahoo_close(session, conn);
+			YAHOO_PRINT(session, YAHOO_LOG_ERROR, "unable to connect");
+			CALLBACK(session, YAHOO_HANDLE_DISCONNECT);
+			return;
+		}
+
+		fcntl(socket, F_SETFL, 0);
+
+		YAHOO_PRINT(session, YAHOO_LOG_NOTICE, "connected");
+
+		if (yahoo_socket_notify)
+			(*yahoo_socket_notify)(session, socket, YAHOO_SOCKET_WRITE, FALSE);
+		if (yahoo_socket_notify)
+			(*yahoo_socket_notify)(session, socket, YAHOO_SOCKET_READ, TRUE);
+
+		if (conn->type == YAHOO_CONN_TYPE_AUTH) {
+			CALLBACK(session, YAHOO_HANDLE_AUTHCONNECT);
+		} else if (conn->type == YAHOO_CONN_TYPE_MAIN) {
+			CALLBACK(session, YAHOO_HANDLE_MAINCONNECT);
+		} else if (conn->type == YAHOO_CONN_TYPE_DUMB) {
+			YAHOO_PRINT(session, YAHOO_LOG_DEBUG, "sending to buddy list host");
+			yahoo_write(session, conn, conn->txqueue, strlen(conn->txqueue));
+			g_free(conn->txqueue);
+			conn->txqueue = NULL;
+		}
+
+		return;
+	}
+
+	if (conn->type == YAHOO_CONN_TYPE_AUTH) {
+		char *buf = g_malloc0(5000);
+		while (read(socket, &buf[pos++], 1) == 1);
+		if (pos == 1) {
+			g_free(buf);
+			yahoo_close(session, conn);
+			CALLBACK(session, YAHOO_HANDLE_DISCONNECT);
+			return;
+		}
+		YAHOO_PRINT(session, YAHOO_LOG_DEBUG, buf);
+		if (yahoo_parse_config(session, conn, buf))
+			CALLBACK(session, YAHOO_HANDLE_DISCONNECT);
+		g_free(buf);
+	} else if (conn->type == YAHOO_CONN_TYPE_MAIN) {
+		struct yahoo_packet pkt;
+		int len;
+
+		if ((read(socket, &pkt, 8) != 8) || strcmp(pkt.version, "YHOO1.0")) {
+			yahoo_close(session, conn);
+			CALLBACK(session, YAHOO_HANDLE_DISCONNECT);
+			return;
+		}
+
+		if (read(socket, &pkt.len, 4) != 4) {
+			yahoo_close(session, conn);
+			CALLBACK(session, YAHOO_HANDLE_DISCONNECT);
+			return;
+		}
+		len = yahoo_makeint(pkt.len);
+		len = ntohs(len) >> 8;
+
+		if (read(socket, &pkt.service, len - 12) != len - 12) {
+			yahoo_close(session, conn);
+			CALLBACK(session, YAHOO_HANDLE_DISCONNECT);
+			return;
+		}
+		yahoo_parse_packet(session, conn, &pkt);
+	} else if (conn->type == YAHOO_CONN_TYPE_DUMB) {
+		YAHOO_PRINT(session, YAHOO_LOG_DEBUG, "closing buddy list host connnection");
+		yahoo_close(session, conn);
+	}
+}
+
+void yahoo_add_handler(struct yahoo_session *session, int type, yahoo_callback function)
+{
+	if (!session)
+		return;
+
+	session->callbacks[type].function = function;
+}