diff options
| author | Marius Halden <marius.h@lden.org> | 2018-04-19 10:23:08 +0200 | 
|---|---|---|
| committer | Marius Halden <marius.h@lden.org> | 2018-04-19 10:23:08 +0200 | 
| commit | db02ac8971379f9fee2f3a618bb08b8076d1a83d (patch) | |
| tree | 0e7d46bad6162509300aa73ae6d77e571a5b26d0 /protocols/skype/skype.c | |
| parent | 6dc54671e368d03ed447d0fbd2d662af6c36b7fd (diff) | |
| parent | 246b98bbdf221448fd7a638fea04373ed1612de5 (diff) | |
Merge branch 'master' into patched-master
Diffstat (limited to 'protocols/skype/skype.c')
| -rw-r--r-- | protocols/skype/skype.c | 1778 | 
1 files changed, 0 insertions, 1778 deletions
| diff --git a/protocols/skype/skype.c b/protocols/skype/skype.c deleted file mode 100644 index bd8a1850..00000000 --- a/protocols/skype/skype.c +++ /dev/null @@ -1,1778 +0,0 @@ -/* - *  skype.c - Skype plugin for BitlBee - * - *  Copyright (c) 2007-2013 by Miklos Vajna <vmiklos@vmiklos.hu> - * - *  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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, - *  USA. - */ - -#define _XOPEN_SOURCE -#include <poll.h> -#include <stdio.h> -#include <bitlbee.h> -#include <ssl_client.h> - -#define SKYPE_DEFAULT_SERVER "localhost" -#define SKYPE_DEFAULT_PORT "2727" -#define IRC_LINE_SIZE 16384 -#define ARRAY_SIZE(x) (sizeof(x) / sizeof(x[0])) - -/* - * Enumerations - */ - -enum { -	SKYPE_CALL_RINGING = 1, -	SKYPE_CALL_MISSED, -	SKYPE_CALL_CANCELLED, -	SKYPE_CALL_FINISHED, -	SKYPE_CALL_REFUSED -}; - -enum { -	SKYPE_FILETRANSFER_NEW = 1, -	SKYPE_FILETRANSFER_TRANSFERRING, -	SKYPE_FILETRANSFER_COMPLETED, -	SKYPE_FILETRANSFER_FAILED -}; - -/* - * Structures - */ - -struct skype_data { -	struct im_connection *ic; -	char *username; -	/* The effective file descriptor. We store it here so any function can -	 * write() to it. */ -	int fd; -	/* File descriptor returned by bitlbee. we store it so we know when -	 * we're connected and when we aren't. */ -	int bfd; -	/* ssl_getfd() uses this to get the file desciptor. */ -	void *ssl; -	/* When we receive a new message id, we query the properties, finally -	 * the chatname. Store the properties here so that we can use -	 * imcb_buddy_msg() when we got the chatname. */ -	char *handle; -	/* List, because of multiline messages. */ -	GList *body; -	char *type; -	/* This is necessary because we send a notification when we get the -	 * handle. So we store the state here and then we can send a -	 * notification about the handle is in a given status. */ -	int call_status; -	char *call_id; -	char *call_duration; -	/* If the call is outgoing or not */ -	int call_out; -	/* Same for file transfers. */ -	int filetransfer_status; -	/* Path of the file being transferred. */ -	char *filetransfer_path; -	/* Using /j #nick we want to have a groupchat with two people. Usually -	 * not (default). */ -	char *groupchat_with; -	/* The user who invited us to the chat. */ -	char *adder; -	/* If we are waiting for a confirmation about we changed the topic. */ -	int topic_wait; -	/* These are used by the info command. */ -	char *info_fullname; -	char *info_phonehome; -	char *info_phoneoffice; -	char *info_phonemobile; -	char *info_nrbuddies; -	char *info_tz; -	char *info_seen; -	char *info_birthday; -	char *info_sex; -	char *info_language; -	char *info_country; -	char *info_province; -	char *info_city; -	char *info_homepage; -	char *info_about; -	/* When a call fails, we get the reason and later we get the failure -	 * event, so store the failure code here till then */ -	int failurereason; -	/* If this is just an update of an already received message. */ -	int is_edit; -	/* List of struct skype_group* */ -	GList *groups; -	/* Pending user which has to be added to the next group which is -	 * created. */ -	char *pending_user; -	/* If the info command was used, to determine what to do with FULLNAME result. */ -	int is_info; -}; - -struct skype_away_state { -	char *code; -	char *full_name; -}; - -struct skype_buddy_ask_data { -	struct im_connection *ic; -	/* This is also used for call IDs for simplicity */ -	char *handle; -}; - -struct skype_group { -	int id; -	char *name; -	GList *users; -}; - -/* - * Tables - */ - -const struct skype_away_state skype_away_state_list[] = { -	{ "AWAY", "Away" }, -	{ "NA", "Not available" }, -	{ "DND", "Do Not Disturb" }, -	{ "INVISIBLE", "Invisible" }, -	{ "OFFLINE", "Offline" }, -	{ "SKYPEME", "Skype Me" }, -	{ "ONLINE", "Online" }, -	{ NULL, NULL } -}; - -/* - * Functions - */ - -int skype_write(struct im_connection *ic, char *buf, int len) -{ -	struct skype_data *sd = ic->proto_data; -	struct pollfd pfd[1]; - -	if (!sd->ssl) { -		return FALSE; -	} - -	pfd[0].fd = sd->fd; -	pfd[0].events = POLLOUT; - -	/* This poll is necessary or we'll get a SIGPIPE when we write() to -	 * sd->fd. */ -	poll(pfd, 1, 1000); -	if (pfd[0].revents & POLLHUP) { -		imc_logout(ic, TRUE); -		return FALSE; -	} -	ssl_write(sd->ssl, buf, len); - -	return TRUE; -} - -int skype_printf(struct im_connection *ic, char *fmt, ...) -{ -	va_list args; -	char str[IRC_LINE_SIZE]; - -	va_start(args, fmt); -	g_vsnprintf(str, IRC_LINE_SIZE, fmt, args); -	va_end(args); - -	return skype_write(ic, str, strlen(str)); -} - -static void skype_buddy_ask_yes(void *data) -{ -	struct skype_buddy_ask_data *bla = data; - -	skype_printf(bla->ic, "SET USER %s ISAUTHORIZED TRUE\n", -	             bla->handle); -	g_free(bla->handle); -	g_free(bla); -} - -static void skype_buddy_ask_no(void *data) -{ -	struct skype_buddy_ask_data *bla = data; - -	skype_printf(bla->ic, "SET USER %s ISAUTHORIZED FALSE\n", -	             bla->handle); -	g_free(bla->handle); -	g_free(bla); -} - -void skype_buddy_ask(struct im_connection *ic, char *handle, char *message) -{ -	struct skype_buddy_ask_data *bla = g_new0(struct skype_buddy_ask_data, -	                                          1); -	char *buf; - -	bla->ic = ic; -	bla->handle = g_strdup(handle); - -	buf = g_strdup_printf("The user %s wants to add you to his/her buddy list, saying: '%s'.", handle, message); -	imcb_ask(ic, buf, bla, skype_buddy_ask_yes, skype_buddy_ask_no); -	g_free(buf); -} - -static void skype_call_ask_yes(void *data) -{ -	struct skype_buddy_ask_data *bla = data; - -	skype_printf(bla->ic, "SET CALL %s STATUS INPROGRESS\n", -	             bla->handle); -	g_free(bla->handle); -	g_free(bla); -} - -static void skype_call_ask_no(void *data) -{ -	struct skype_buddy_ask_data *bla = data; - -	skype_printf(bla->ic, "SET CALL %s STATUS FINISHED\n", -	             bla->handle); -	g_free(bla->handle); -	g_free(bla); -} - -void skype_call_ask(struct im_connection *ic, char *call_id, char *message) -{ -	struct skype_buddy_ask_data *bla = g_new0(struct skype_buddy_ask_data, -	                                          1); - -	bla->ic = ic; -	bla->handle = g_strdup(call_id); - -	imcb_ask(ic, message, bla, skype_call_ask_yes, skype_call_ask_no); -} - -static char *skype_call_strerror(int err) -{ -	switch (err) { -	case 1: -		return "Miscellaneous error"; -	case 2: -		return "User or phone number does not exist."; -	case 3: -		return "User is offline"; -	case 4: -		return "No proxy found"; -	case 5: -		return "Session terminated."; -	case 6: -		return "No common codec found."; -	case 7: -		return "Sound I/O error."; -	case 8: -		return "Problem with remote sound device."; -	case 9: -		return "Call blocked by recipient."; -	case 10: -		return "Recipient not a friend."; -	case 11: -		return "Current user not authorized by recipient."; -	case 12: -		return "Sound recording error."; -	default: -		return "Unknown error"; -	} -} - -static char *skype_group_by_username(struct im_connection *ic, char *username) -{ -	struct skype_data *sd = ic->proto_data; -	int i, j; - -	/* NEEDSWORK: we just search for the first group of the user, multiple -	 * groups / user is not yet supported by BitlBee. */ - -	for (i = 0; i < g_list_length(sd->groups); i++) { -		struct skype_group *sg = g_list_nth_data(sd->groups, i); -		for (j = 0; j < g_list_length(sg->users); j++) { -			if (!strcmp(g_list_nth_data(sg->users, j), username)) { -				return sg->name; -			} -		} -	} -	return NULL; -} - -static struct skype_group *skype_group_by_name(struct im_connection *ic, char *name) -{ -	struct skype_data *sd = ic->proto_data; -	int i; - -	for (i = 0; i < g_list_length(sd->groups); i++) { -		struct skype_group *sg = g_list_nth_data(sd->groups, i); -		if (!strcmp(sg->name, name)) { -			return sg; -		} -	} -	return NULL; -} - -static struct groupchat *skype_chat_get_or_create(struct im_connection *ic, char *id) -{ -	struct skype_data *sd = ic->proto_data; -	struct groupchat *gc = bee_chat_by_title(ic->bee, ic, id); - -	if (!gc) { -		gc = imcb_chat_new(ic, id); -		imcb_chat_name_hint(gc, id); -		imcb_chat_add_buddy(gc, sd->username); - -		skype_printf(ic, "GET CHAT %s ADDER\n", id); -		skype_printf(ic, "GET CHAT %s TOPIC\n", id); -		skype_printf(ic, "GET CHAT %s ACTIVEMEMBERS\n", id); -	} - -	return gc; -} - -static void skype_parse_users(struct im_connection *ic, char *line) -{ -	char **i, **nicks; - -	nicks = g_strsplit(line + 6, ", ", 0); -	for (i = nicks; *i; i++) { -		skype_printf(ic, "GET USER %s ONLINESTATUS\n", *i); -		skype_printf(ic, "GET USER %s FULLNAME\n", *i); -	} -	g_strfreev(nicks); -} - -static void skype_parse_user(struct im_connection *ic, char *line) -{ -	int flags = 0; -	char *ptr; -	struct skype_data *sd = ic->proto_data; -	char *user = strchr(line, ' '); -	char *status = strrchr(line, ' '); - -	status++; -	ptr = strchr(++user, ' '); -	if (!ptr) { -		return; -	} -	*ptr = '\0'; -	ptr++; -	if (!strncmp(ptr, "ONLINESTATUS ", 13)) { -		if (!strlen(user) || !strcmp(user, sd->username)) { -			return; -		} -		if (!set_getbool(&ic->acc->set, "test_join") -		    && !strcmp(user, "echo123")) { -			return; -		} -		ptr = g_strdup_printf("%s@skype.com", user); -		imcb_add_buddy(ic, ptr, skype_group_by_username(ic, user)); -		if (strcmp(status, "OFFLINE") && (strcmp(status, "SKYPEOUT") || -		                                  !set_getbool(&ic->acc->set, "skypeout_offline"))) { -			flags |= OPT_LOGGED_IN; -		} -		if (strcmp(status, "ONLINE") && strcmp(status, "SKYPEME")) { -			flags |= OPT_AWAY; -		} -		imcb_buddy_status(ic, ptr, flags, NULL, NULL); -		g_free(ptr); -	} else if (!strncmp(ptr, "RECEIVEDAUTHREQUEST ", 20)) { -		char *message = ptr + 20; -		if (strlen(message)) { -			skype_buddy_ask(ic, user, message); -		} -	} else if (!strncmp(ptr, "BUDDYSTATUS ", 12)) { -		char *st = ptr + 12; -		if (!strcmp(st, "3")) { -			char *buf = g_strdup_printf("%s@skype.com", user); -			imcb_add_buddy(ic, buf, skype_group_by_username(ic, user)); -			g_free(buf); -		} -	} else if (!strncmp(ptr, "MOOD_TEXT ", 10)) { -		char *buf = g_strdup_printf("%s@skype.com", user); -		bee_user_t *bu = bee_user_by_handle(ic->bee, ic, buf); -		g_free(buf); -		buf = ptr + 10; -		if (bu) { -			imcb_buddy_status(ic, bu->handle, bu->flags, NULL, -			                  *buf ? buf : NULL); -		} -		if (set_getbool(&ic->acc->set, "show_moods")) { -			imcb_log(ic, "User `%s' changed mood text to `%s'", user, buf); -		} -	} else if (!strncmp(ptr, "FULLNAME ", 9)) { -		char *name = ptr + 9; -		if (sd->is_info) { -			sd->is_info = FALSE; -			sd->info_fullname = g_strdup(name); -		} else { -			char *buf = g_strdup_printf("%s@skype.com", user); -			imcb_rename_buddy(ic, buf, name); -			g_free(buf); -		} -	} else if (!strncmp(ptr, "PHONE_HOME ", 11)) { -		sd->info_phonehome = g_strdup(ptr + 11); -	} else if (!strncmp(ptr, "PHONE_OFFICE ", 13)) { -		sd->info_phoneoffice = g_strdup(ptr + 13); -	} else if (!strncmp(ptr, "PHONE_MOBILE ", 13)) { -		sd->info_phonemobile = g_strdup(ptr + 13); -	} else if (!strncmp(ptr, "NROF_AUTHED_BUDDIES ", 20)) { -		sd->info_nrbuddies = g_strdup(ptr + 20); -	} else if (!strncmp(ptr, "TIMEZONE ", 9)) { -		sd->info_tz = g_strdup(ptr + 9); -	} else if (!strncmp(ptr, "LASTONLINETIMESTAMP ", 20)) { -		sd->info_seen = g_strdup(ptr + 20); -	} else if (!strncmp(ptr, "SEX ", 4)) { -		sd->info_sex = g_strdup(ptr + 4); -	} else if (!strncmp(ptr, "LANGUAGE ", 9)) { -		sd->info_language = g_strdup(ptr + 9); -	} else if (!strncmp(ptr, "COUNTRY ", 8)) { -		sd->info_country = g_strdup(ptr + 8); -	} else if (!strncmp(ptr, "PROVINCE ", 9)) { -		sd->info_province = g_strdup(ptr + 9); -	} else if (!strncmp(ptr, "CITY ", 5)) { -		sd->info_city = g_strdup(ptr + 5); -	} else if (!strncmp(ptr, "HOMEPAGE ", 9)) { -		sd->info_homepage = g_strdup(ptr + 9); -	} else if (!strncmp(ptr, "ABOUT ", 6)) { -		/* Support multiple about lines. */ -		if (!sd->info_about) { -			sd->info_about = g_strdup(ptr + 6); -		} else { -			GString *st = g_string_new(sd->info_about); -			g_string_append_printf(st, "\n%s", ptr + 6); -			g_free(sd->info_about); -			sd->info_about = g_strdup(st->str); -			g_string_free(st, TRUE); -		} -	} else if (!strncmp(ptr, "BIRTHDAY ", 9)) { -		sd->info_birthday = g_strdup(ptr + 9); - -		GString *st = g_string_new("Contact Information\n"); -		g_string_append_printf(st, "Skype Name: %s\n", user); -		if (sd->info_fullname) { -			if (strlen(sd->info_fullname)) { -				g_string_append_printf(st, "Full Name: %s\n", -				                       sd->info_fullname); -			} -			g_free(sd->info_fullname); -			sd->info_fullname = NULL; -		} -		if (sd->info_phonehome) { -			if (strlen(sd->info_phonehome)) { -				g_string_append_printf(st, "Home Phone: %s\n", -				                       sd->info_phonehome); -			} -			g_free(sd->info_phonehome); -			sd->info_phonehome = NULL; -		} -		if (sd->info_phoneoffice) { -			if (strlen(sd->info_phoneoffice)) { -				g_string_append_printf(st, "Office Phone: %s\n", -				                       sd->info_phoneoffice); -			} -			g_free(sd->info_phoneoffice); -			sd->info_phoneoffice = NULL; -		} -		if (sd->info_phonemobile) { -			if (strlen(sd->info_phonemobile)) { -				g_string_append_printf(st, "Mobile Phone: %s\n", -				                       sd->info_phonemobile); -			} -			g_free(sd->info_phonemobile); -			sd->info_phonemobile = NULL; -		} -		g_string_append_printf(st, "Personal Information\n"); -		if (sd->info_nrbuddies) { -			if (strlen(sd->info_nrbuddies)) { -				g_string_append_printf(st, -				                       "Contacts: %s\n", sd->info_nrbuddies); -			} -			g_free(sd->info_nrbuddies); -			sd->info_nrbuddies = NULL; -		} -		if (sd->info_tz) { -			if (strlen(sd->info_tz)) { -				char ib[256]; -				time_t t = time(NULL); -				t += atoi(sd->info_tz) - (60 * 60 * 24); -				struct tm *gt = gmtime(&t); -				strftime(ib, 256, "%H:%M:%S", gt); -				g_string_append_printf(st, -				                       "Local Time: %s\n", ib); -			} -			g_free(sd->info_tz); -			sd->info_tz = NULL; -		} -		if (sd->info_seen) { -			if (strlen(sd->info_seen)) { -				char ib[256]; -				time_t it = atoi(sd->info_seen); -				struct tm *tm = localtime(&it); -				strftime(ib, 256, ("%Y. %m. %d. %H:%M"), tm); -				g_string_append_printf(st, -				                       "Last Seen: %s\n", ib); -			} -			g_free(sd->info_seen); -			sd->info_seen = NULL; -		} -		if (sd->info_birthday) { -			if (strlen(sd->info_birthday) && -			    strcmp(sd->info_birthday, "0")) { -				char ib[256]; -				struct tm tm; -				strptime(sd->info_birthday, "%Y%m%d", &tm); -				strftime(ib, 256, "%B %d, %Y", &tm); -				g_string_append_printf(st, -				                       "Birthday: %s\n", ib); - -				strftime(ib, 256, "%Y", &tm); -				int year = atoi(ib); -				time_t t = time(NULL); -				struct tm *lt = localtime(&t); -				g_string_append_printf(st, -				                       "Age: %d\n", lt->tm_year + 1900 - year); -			} -			g_free(sd->info_birthday); -			sd->info_birthday = NULL; -		} -		if (sd->info_sex) { -			if (strlen(sd->info_sex)) { -				char *iptr = sd->info_sex; -				while (*iptr++) { -					*iptr = g_ascii_tolower(*iptr); -				} -				g_string_append_printf(st, -				                       "Gender: %s\n", sd->info_sex); -			} -			g_free(sd->info_sex); -			sd->info_sex = NULL; -		} -		if (sd->info_language) { -			if (strlen(sd->info_language)) { -				char *iptr = strchr(sd->info_language, ' '); -				if (iptr) { -					iptr++; -				} else { -					iptr = sd->info_language; -				} -				g_string_append_printf(st, -				                       "Language: %s\n", iptr); -			} -			g_free(sd->info_language); -			sd->info_language = NULL; -		} -		if (sd->info_country) { -			if (strlen(sd->info_country)) { -				char *iptr = strchr(sd->info_country, ' '); -				if (iptr) { -					iptr++; -				} else { -					iptr = sd->info_country; -				} -				g_string_append_printf(st, -				                       "Country: %s\n", iptr); -			} -			g_free(sd->info_country); -			sd->info_country = NULL; -		} -		if (sd->info_province) { -			if (strlen(sd->info_province)) { -				g_string_append_printf(st, -				                       "Region: %s\n", sd->info_province); -			} -			g_free(sd->info_province); -			sd->info_province = NULL; -		} -		if (sd->info_city) { -			if (strlen(sd->info_city)) { -				g_string_append_printf(st, -				                       "City: %s\n", sd->info_city); -			} -			g_free(sd->info_city); -			sd->info_city = NULL; -		} -		if (sd->info_homepage) { -			if (strlen(sd->info_homepage)) { -				g_string_append_printf(st, -				                       "Homepage: %s\n", sd->info_homepage); -			} -			g_free(sd->info_homepage); -			sd->info_homepage = NULL; -		} -		if (sd->info_about) { -			if (strlen(sd->info_about)) { -				g_string_append_printf(st, "%s\n", -				                       sd->info_about); -			} -			g_free(sd->info_about); -			sd->info_about = NULL; -		} -		imcb_log(ic, "%s", st->str); -		g_string_free(st, TRUE); -	} -} - -static void skype_parse_chatmessage_said_emoted(struct im_connection *ic, struct groupchat *gc, char *body) -{ -	struct skype_data *sd = ic->proto_data; -	char buf[IRC_LINE_SIZE]; - -	if (!strcmp(sd->type, "SAID")) { -		if (!sd->is_edit) { -			g_snprintf(buf, IRC_LINE_SIZE, "%s", body); -		} else { -			g_snprintf(buf, IRC_LINE_SIZE, "%s %s", set_getstr(&ic->acc->set, "edit_prefix"), body); -			sd->is_edit = 0; -		} -	} else { -		g_snprintf(buf, IRC_LINE_SIZE, "/me %s", body); -	} -	if (!gc) { -		/* Private message */ -		imcb_buddy_msg(ic, sd->handle, buf, 0, 0); -	} else { -		/* Groupchat message */ -		imcb_chat_msg(gc, sd->handle, buf, 0, 0); -	} -} - -static void skype_parse_chatmessage(struct im_connection *ic, char *line) -{ -	struct skype_data *sd = ic->proto_data; -	char *id = strchr(line, ' '); - -	if (!++id) { -		return; -	} -	char *info = strchr(id, ' '); - -	if (!info) { -		return; -	} -	*info = '\0'; -	info++; -	if (!strcmp(info, "STATUS RECEIVED") || !strncmp(info, "EDITED_TIMESTAMP", 16)) { -		/* New message ID: -		 * (1) Request its from field -		 * (2) Request its body -		 * (3) Request its type -		 * (4) Query chatname -		 */ -		skype_printf(ic, "GET CHATMESSAGE %s FROM_HANDLE\n", id); -		if (!strcmp(info, "STATUS RECEIVED")) { -			skype_printf(ic, "GET CHATMESSAGE %s BODY\n", id); -		} else { -			sd->is_edit = 1; -		} -		skype_printf(ic, "GET CHATMESSAGE %s TYPE\n", id); -		skype_printf(ic, "GET CHATMESSAGE %s CHATNAME\n", id); -	} else if (!strncmp(info, "FROM_HANDLE ", 12)) { -		info += 12; -		/* New from field value. Store -		 * it, then we can later use it -		 * when we got the message's -		 * body. */ -		g_free(sd->handle); -		sd->handle = g_strdup_printf("%s@skype.com", info); -	} else if (!strncmp(info, "EDITED_BY ", 10)) { -		info += 10; -		/* This is the same as -		 * FROM_HANDLE, except that we -		 * never request these lines -		 * from Skype, we just get -		 * them. */ -		g_free(sd->handle); -		sd->handle = g_strdup_printf("%s@skype.com", info); -	} else if (!strncmp(info, "BODY ", 5)) { -		info += 5; -		sd->body = g_list_append(sd->body, g_strdup(info)); -	} else if (!strncmp(info, "TYPE ", 5)) { -		info += 5; -		g_free(sd->type); -		sd->type = g_strdup(info); -	} else if (!strncmp(info, "CHATNAME ", 9)) { -		info += 9; -		if (sd->handle && sd->body && sd->type) { -			struct groupchat *gc = skype_chat_get_or_create(ic, info); -			int i; -			for (i = 0; i < g_list_length(sd->body); i++) { -				char *body = g_list_nth_data(sd->body, i); -				if (!strcmp(sd->type, "SAID") || -				    !strcmp(sd->type, "EMOTED")) { -					skype_parse_chatmessage_said_emoted(ic, gc, body); -				} else if (!strcmp(sd->type, "SETTOPIC") && gc) { -					imcb_chat_topic(gc, -					                sd->handle, body, 0); -				} else if (!strcmp(sd->type, "LEFT") && gc) { -					imcb_chat_remove_buddy(gc, -					                       sd->handle, NULL); -				} -			} -			g_list_free(sd->body); -			sd->body = NULL; -		} -	} -} - -static void skype_parse_call(struct im_connection *ic, char *line) -{ -	struct skype_data *sd = ic->proto_data; -	char *id = strchr(line, ' '); -	char buf[IRC_LINE_SIZE]; - -	if (!++id) { -		return; -	} -	char *info = strchr(id, ' '); - -	if (!info) { -		return; -	} -	*info = '\0'; -	info++; -	if (!strncmp(info, "FAILUREREASON ", 14)) { -		sd->failurereason = atoi(strchr(info, ' ')); -	} else if (!strcmp(info, "STATUS RINGING")) { -		if (sd->call_id) { -			g_free(sd->call_id); -		} -		sd->call_id = g_strdup(id); -		skype_printf(ic, "GET CALL %s PARTNER_HANDLE\n", id); -		sd->call_status = SKYPE_CALL_RINGING; -	} else if (!strcmp(info, "STATUS MISSED")) { -		skype_printf(ic, "GET CALL %s PARTNER_HANDLE\n", id); -		sd->call_status = SKYPE_CALL_MISSED; -	} else if (!strcmp(info, "STATUS CANCELLED")) { -		skype_printf(ic, "GET CALL %s PARTNER_HANDLE\n", id); -		sd->call_status = SKYPE_CALL_CANCELLED; -	} else if (!strcmp(info, "STATUS FINISHED")) { -		skype_printf(ic, "GET CALL %s PARTNER_HANDLE\n", id); -		sd->call_status = SKYPE_CALL_FINISHED; -	} else if (!strcmp(info, "STATUS REFUSED")) { -		skype_printf(ic, "GET CALL %s PARTNER_HANDLE\n", id); -		sd->call_status = SKYPE_CALL_REFUSED; -	} else if (!strcmp(info, "STATUS UNPLACED")) { -		if (sd->call_id) { -			g_free(sd->call_id); -		} -		/* Save the ID for later usage (Cancel/Finish). */ -		sd->call_id = g_strdup(id); -		sd->call_out = TRUE; -	} else if (!strcmp(info, "STATUS FAILED")) { -		imcb_error(ic, "Call failed: %s", -		           skype_call_strerror(sd->failurereason)); -		sd->call_id = NULL; -	} else if (!strncmp(info, "DURATION ", 9)) { -		if (sd->call_duration) { -			g_free(sd->call_duration); -		} -		sd->call_duration = g_strdup(info + 9); -	} else if (!strncmp(info, "PARTNER_HANDLE ", 15)) { -		info += 15; -		if (!sd->call_status) { -			return; -		} -		switch (sd->call_status) { -		case SKYPE_CALL_RINGING: -			if (sd->call_out) { -				imcb_log(ic, "You are currently ringing the user %s.", info); -			} else { -				g_snprintf(buf, IRC_LINE_SIZE, -				           "The user %s is currently ringing you.", -				           info); -				skype_call_ask(ic, sd->call_id, buf); -			} -			break; -		case SKYPE_CALL_MISSED: -			imcb_log(ic, "You have missed a call from user %s.", -			         info); -			break; -		case SKYPE_CALL_CANCELLED: -			imcb_log(ic, "You cancelled the call to the user %s.", -			         info); -			sd->call_status = 0; -			sd->call_out = FALSE; -			break; -		case SKYPE_CALL_REFUSED: -			if (sd->call_out) { -				imcb_log(ic, "The user %s refused the call.", -				         info); -			} else { -				imcb_log(ic, -				         "You refused the call from user %s.", -				         info); -			} -			sd->call_out = FALSE; -			break; -		case SKYPE_CALL_FINISHED: -			if (sd->call_duration) { -				imcb_log(ic, -				         "You finished the call to the user %s " -				         "(duration: %s seconds).", -				         info, sd->call_duration); -			} else { -				imcb_log(ic, -				         "You finished the call to the user %s.", -				         info); -			} -			sd->call_out = FALSE; -			break; -		default: -			/* Don't be noisy, ignore other statuses for now. */ -			break; -		} -		sd->call_status = 0; -	} -} - -static void skype_parse_filetransfer(struct im_connection *ic, char *line) -{ -	struct skype_data *sd = ic->proto_data; -	char *id = strchr(line, ' '); - -	if (!++id) { -		return; -	} -	char *info = strchr(id, ' '); - -	if (!info) { -		return; -	} -	*info = '\0'; -	info++; -	if (!strcmp(info, "STATUS NEW")) { -		skype_printf(ic, "GET FILETRANSFER %s PARTNER_HANDLE\n", -		             id); -		sd->filetransfer_status = SKYPE_FILETRANSFER_NEW; -	} else if (!strcmp(info, "STATUS FAILED")) { -		skype_printf(ic, "GET FILETRANSFER %s PARTNER_HANDLE\n", -		             id); -		sd->filetransfer_status = SKYPE_FILETRANSFER_FAILED; -	} else if (!strcmp(info, "STATUS COMPLETED")) { -		skype_printf(ic, "GET FILETRANSFER %s PARTNER_HANDLE\n", id); -		sd->filetransfer_status = SKYPE_FILETRANSFER_COMPLETED; -	} else if (!strcmp(info, "STATUS TRANSFERRING")) { -		skype_printf(ic, "GET FILETRANSFER %s PARTNER_HANDLE\n", id); -		sd->filetransfer_status = SKYPE_FILETRANSFER_TRANSFERRING; -	} else if (!strncmp(info, "FILEPATH ", 9)) { -		info += 9; -		sd->filetransfer_path = g_strdup(info); -	} else if (!strncmp(info, "PARTNER_HANDLE ", 15)) { -		info += 15; -		if (!sd->filetransfer_status) { -			return; -		} -		switch (sd->filetransfer_status) { -		case SKYPE_FILETRANSFER_NEW: -			imcb_log(ic, "The user %s offered a new file for you.", -			         info); -			break; -		case SKYPE_FILETRANSFER_FAILED: -			imcb_log(ic, "Failed to transfer file from user %s.", -			         info); -			break; -		case SKYPE_FILETRANSFER_COMPLETED: -			imcb_log(ic, "File transfer from user %s completed.", info); -			break; -		case SKYPE_FILETRANSFER_TRANSFERRING: -			if (sd->filetransfer_path) { -				imcb_log(ic, "File transfer from user %s started, saving to %s.", info, -				         sd->filetransfer_path); -				g_free(sd->filetransfer_path); -				sd->filetransfer_path = NULL; -			} -			break; -		} -		sd->filetransfer_status = 0; -	} -} - -static struct skype_group *skype_group_by_id(struct im_connection *ic, int id) -{ -	struct skype_data *sd = ic->proto_data; -	int i; - -	for (i = 0; i < g_list_length(sd->groups); i++) { -		struct skype_group *sg = (struct skype_group *) g_list_nth_data(sd->groups, i); - -		if (sg->id == id) { -			return sg; -		} -	} -	return NULL; -} - -static void skype_group_free(struct skype_group *sg, gboolean usersonly) -{ -	int i; - -	for (i = 0; i < g_list_length(sg->users); i++) { -		char *user = g_list_nth_data(sg->users, i); -		g_free(user); -	} -	sg->users = NULL; -	if (usersonly) { -		return; -	} -	g_free(sg->name); -	g_free(sg); -} - -/* Update the group of each user in this group */ -static void skype_group_users(struct im_connection *ic, struct skype_group *sg) -{ -	int i; - -	for (i = 0; i < g_list_length(sg->users); i++) { -		char *user = g_list_nth_data(sg->users, i); -		char *buf = g_strdup_printf("%s@skype.com", user); -		imcb_add_buddy(ic, buf, sg->name); -		g_free(buf); -	} -} - -static void skype_parse_group(struct im_connection *ic, char *line) -{ -	struct skype_data *sd = ic->proto_data; -	char *id = strchr(line, ' '); - -	if (!++id) { -		return; -	} - -	char *info = strchr(id, ' '); - -	if (!info) { -		return; -	} -	*info = '\0'; -	info++; - -	if (!strncmp(info, "DISPLAYNAME ", 12)) { -		info += 12; - -		/* Name given for a group ID: try to update it or insert a new -		 * one if not found */ -		struct skype_group *sg = skype_group_by_id(ic, atoi(id)); -		if (sg) { -			g_free(sg->name); -			sg->name = g_strdup(info); -		} else { -			sg = g_new0(struct skype_group, 1); -			sg->id = atoi(id); -			sg->name = g_strdup(info); -			sd->groups = g_list_append(sd->groups, sg); -		} -	} else if (!strncmp(info, "USERS ", 6)) { -		struct skype_group *sg = skype_group_by_id(ic, atoi(id)); - -		if (sg) { -			char **i; -			char **users = g_strsplit(info + 6, ", ", 0); - -			skype_group_free(sg, TRUE); -			i = users; -			while (*i) { -				sg->users = g_list_append(sg->users, g_strdup(*i)); -				i++; -			} -			g_strfreev(users); -			skype_group_users(ic, sg); -		} else { -			log_message(LOGLVL_ERROR, -			            "No skype group with id %s. That's probably a bug.", id); -		} -	} else if (!strncmp(info, "NROFUSERS ", 10)) { -		if (!sd->pending_user) { -			/* Number of users changed in this group, query its type to see -			 * if it's a custom one we should care about. */ -			skype_printf(ic, "GET GROUP %s TYPE\n", id); -			return; -		} - -		/* This is a newly created group, we have a single user -		 * to add. */ -		struct skype_group *sg = skype_group_by_id(ic, atoi(id)); - -		if (sg) { -			skype_printf(ic, "ALTER GROUP %d ADDUSER %s\n", sg->id, sd->pending_user); -			g_free(sd->pending_user); -			sd->pending_user = NULL; -		} else { -			log_message(LOGLVL_ERROR, -			            "No skype group with id %s. That's probably a bug.", id); -		} -	} else if (!strcmp(info, "TYPE CUSTOM_GROUP")) { -		/* This one is interesting, query its users. */ -		skype_printf(ic, "GET GROUP %s USERS\n", id); -	} -} - -static void skype_parse_chat(struct im_connection *ic, char *line) -{ -	struct skype_data *sd = ic->proto_data; -	char buf[IRC_LINE_SIZE]; -	char *id = strchr(line, ' '); - -	if (!++id) { -		return; -	} -	struct groupchat *gc; -	char *info = strchr(id, ' '); - -	if (!info) { -		return; -	} -	*info = '\0'; -	info++; -	/* Remove fake chat if we created one in skype_chat_with() */ -	gc = bee_chat_by_title(ic->bee, ic, ""); -	if (gc) { -		imcb_chat_free(gc); -	} -	if (!strcmp(info, "STATUS MULTI_SUBSCRIBED")) { -		skype_chat_get_or_create(ic, id); -	} else if (!strcmp(info, "STATUS DIALOG") && sd->groupchat_with) { -		gc = skype_chat_get_or_create(ic, id); -		/* According to the docs this -		 * is necessary. However it -		 * does not seem the situation -		 * and it would open an extra -		 * window on our client, so -		 * just leave it out. */ -		/*skype_printf(ic, "OPEN CHAT %s\n", id);*/ -		g_snprintf(buf, IRC_LINE_SIZE, "%s@skype.com", -		           sd->groupchat_with); -		imcb_chat_add_buddy(gc, buf); -		g_free(sd->groupchat_with); -		sd->groupchat_with = NULL; -	} else if (!strcmp(info, "STATUS UNSUBSCRIBED")) { -		gc = bee_chat_by_title(ic->bee, ic, id); -		if (gc) { -			gc->data = (void *) FALSE; -		} -	} else if (!strncmp(info, "ADDER ", 6)) { -		info += 6; -		g_free(sd->adder); -		sd->adder = g_strdup_printf("%s@skype.com", info); -	} else if (!strncmp(info, "TOPIC ", 6)) { -		info += 6; -		gc = bee_chat_by_title(ic->bee, ic, id); -		if (gc && (sd->adder || sd->topic_wait)) { -			if (sd->topic_wait) { -				sd->adder = g_strdup(sd->username); -				sd->topic_wait = 0; -			} -			imcb_chat_topic(gc, sd->adder, info, 0); -			g_free(sd->adder); -			sd->adder = NULL; -		} -	} else if (!strncmp(info, "MEMBERS ", 8) || !strncmp(info, "ACTIVEMEMBERS ", 14)) { -		if (!strncmp(info, "MEMBERS ", 8)) { -			info += 8; -		} else { -			info += 14; -		} -		gc = bee_chat_by_title(ic->bee, ic, id); -		/* Hack! We set ->data to TRUE -		 * while we're on the channel -		 * so that we won't rejoin -		 * after a /part. */ -		if (!gc || gc->data) { -			return; -		} -		char **members = g_strsplit(info, " ", 0); -		int i; -		for (i = 0; members[i]; i++) { -			if (!strcmp(members[i], sd->username)) { -				continue; -			} -			g_snprintf(buf, IRC_LINE_SIZE, "%s@skype.com", -			           members[i]); -			if (!g_list_find_custom(gc->in_room, buf, -			                        (GCompareFunc) strcmp)) { -				imcb_chat_add_buddy(gc, buf); -			} -		} -		imcb_chat_add_buddy(gc, sd->username); -		g_strfreev(members); -	} -} - -static void skype_parse_password(struct im_connection *ic, char *line) -{ -	if (!strncmp(line + 9, "OK", 2)) { -		imcb_connected(ic); -	} else { -		imcb_error(ic, "Authentication Failed"); -		imc_logout(ic, TRUE); -	} -} - -static void skype_parse_profile(struct im_connection *ic, char *line) -{ -	imcb_log(ic, "SkypeOut balance value is '%s'.", line + 21); -} - -static void skype_parse_ping(struct im_connection *ic, char *line) -{ -	/* Unused parameter */ -	line = line; -	skype_printf(ic, "PONG\n"); -} - -static void skype_parse_chats(struct im_connection *ic, char *line) -{ -	char **i; -	char **chats = g_strsplit(line + 6, ", ", 0); - -	i = chats; -	while (*i) { -		skype_printf(ic, "GET CHAT %s STATUS\n", *i); -		skype_printf(ic, "GET CHAT %s ACTIVEMEMBERS\n", *i); -		i++; -	} -	g_strfreev(chats); -} - -static void skype_parse_groups(struct im_connection *ic, char *line) -{ -	if (!set_getbool(&ic->acc->set, "read_groups")) { -		return; -	} - -	char **i; -	char **groups = g_strsplit(line + 7, ", ", 0); - -	i = groups; -	while (*i) { -		skype_printf(ic, "GET GROUP %s DISPLAYNAME\n", *i); -		skype_printf(ic, "GET GROUP %s USERS\n", *i); -		i++; -	} -	g_strfreev(groups); -} - -static void skype_parse_alter_group(struct im_connection *ic, char *line) -{ -	char *id = line + strlen("ALTER GROUP"); - -	if (!++id) { -		return; -	} - -	char *info = strchr(id, ' '); - -	if (!info) { -		return; -	} -	*info = '\0'; -	info++; - -	if (!strncmp(info, "ADDUSER ", 8)) { -		struct skype_group *sg = skype_group_by_id(ic, atoi(id)); - -		info += 8; -		if (sg) { -			char *buf = g_strdup_printf("%s@skype.com", info); -			sg->users = g_list_append(sg->users, g_strdup(info)); -			imcb_add_buddy(ic, buf, sg->name); -			g_free(buf); -		} else { -			log_message(LOGLVL_ERROR, -			            "No skype group with id %s. That's probably a bug.", id); -		} -	} -} - -typedef void (*skype_parser)(struct im_connection *ic, char *line); - -static gboolean skype_read_callback(gpointer data, gint fd, -                                    b_input_condition cond) -{ -	struct im_connection *ic = data; -	struct skype_data *sd = ic->proto_data; -	char buf[IRC_LINE_SIZE]; -	int st, i; -	char **lines, **lineptr, *line; -	static struct parse_map { -		char *k; -		skype_parser v; -	} parsers[] = { -		{ "USERS ", skype_parse_users }, -		{ "USER ", skype_parse_user }, -		{ "CHATMESSAGE ", skype_parse_chatmessage }, -		{ "CALL ", skype_parse_call }, -		{ "FILETRANSFER ", skype_parse_filetransfer }, -		{ "CHAT ", skype_parse_chat }, -		{ "GROUP ", skype_parse_group }, -		{ "PASSWORD ", skype_parse_password }, -		{ "PROFILE PSTN_BALANCE ", skype_parse_profile }, -		{ "PING", skype_parse_ping }, -		{ "CHATS ", skype_parse_chats }, -		{ "GROUPS ", skype_parse_groups }, -		{ "ALTER GROUP ", skype_parse_alter_group }, -	}; - -	/* Unused parameters */ -	fd = fd; -	cond = cond; - -	if (!sd || sd->fd == -1) { -		return FALSE; -	} -	/* Read the whole data. */ -	st = ssl_read(sd->ssl, buf, sizeof(buf)); -	if (st >= IRC_LINE_SIZE - 1) { -		/* As we don't buffer incoming data, if IRC_LINE_SIZE amount of bytes -		 * were received, there's a good chance last message was truncated -		 * and the next recv() will yield garbage. */ -		imcb_error(ic, "Unable to handle incoming data from skyped"); -		st = 0; -	} -	if (st > 0) { -		buf[st] = '\0'; -		/* Then split it up to lines. */ -		lines = g_strsplit(buf, "\n", 0); -		lineptr = lines; -		while ((line = *lineptr)) { -			if (!strlen(line)) { -				break; -			} -			if (set_getbool(&ic->acc->set, "skypeconsole_receive")) { -				imcb_buddy_msg(ic, "skypeconsole", line, 0, 0); -			} -			for (i = 0; i < ARRAY_SIZE(parsers); i++) { -				if (!strncmp(line, parsers[i].k, -				             strlen(parsers[i].k))) { -					parsers[i].v(ic, line); -					break; -				} -			} -			lineptr++; -		} -		g_strfreev(lines); -	} else if (st == 0 || (st < 0 && !ssl_sockerr_again(sd->ssl))) { -		ssl_disconnect(sd->ssl); -		sd->fd = -1; -		sd->ssl = NULL; - -		imcb_error(ic, "Error while reading from server"); -		imc_logout(ic, TRUE); -		return FALSE; -	} -	return TRUE; -} - -gboolean skype_start_stream(struct im_connection *ic) -{ -	struct skype_data *sd = ic->proto_data; -	int st; - -	if (!sd) { -		return FALSE; -	} - -	if (sd->bfd <= 0) { -		sd->bfd = b_input_add(sd->fd, B_EV_IO_READ, -		                      skype_read_callback, ic); -	} - -	/* Log in */ -	skype_printf(ic, "USERNAME %s\n", ic->acc->user); -	skype_printf(ic, "PASSWORD %s\n", ic->acc->pass); - -	/* This will download all buddies and groups. */ -	st = skype_printf(ic, "SEARCH GROUPS CUSTOM\n"); -	skype_printf(ic, "SEARCH FRIENDS\n"); - -	skype_printf(ic, "SET USERSTATUS ONLINE\n"); - -	/* Auto join to bookmarked chats if requested.*/ -	if (set_getbool(&ic->acc->set, "auto_join")) { -		skype_printf(ic, "SEARCH BOOKMARKEDCHATS\n"); -		skype_printf(ic, "SEARCH ACTIVECHATS\n"); -		skype_printf(ic, "SEARCH MISSEDCHATS\n"); -		skype_printf(ic, "SEARCH RECENTCHATS\n"); -	} -	return st; -} - -gboolean skype_connected(gpointer data, int returncode, void *source, b_input_condition cond) -{ -	struct im_connection *ic = data; -	struct skype_data *sd = ic->proto_data; - -	/* Unused parameter */ -	cond = cond; - -	if (!source) { -		sd->ssl = NULL; -		imcb_error(ic, "Could not connect to server"); -		imc_logout(ic, TRUE); -		return FALSE; -	} -	imcb_log(ic, "Connected to server, logging in"); - -	return skype_start_stream(ic); -} - -static void skype_login(account_t *acc) -{ -	struct im_connection *ic = imcb_new(acc); -	struct skype_data *sd = g_new0(struct skype_data, 1); - -	ic->proto_data = sd; - -	imcb_log(ic, "Connecting"); -	sd->ssl = ssl_connect(set_getstr(&acc->set, "server"), -	                      set_getint(&acc->set, "port"), FALSE, skype_connected, ic); -	sd->fd = sd->ssl ? ssl_getfd(sd->ssl) : -1; -	sd->username = g_strdup(acc->user); - -	sd->ic = ic; - -	if (set_getbool(&acc->set, "skypeconsole")) { -		imcb_add_buddy(ic, "skypeconsole", NULL); -	} -} - -static void skype_logout(struct im_connection *ic) -{ -	struct skype_data *sd = ic->proto_data; -	int i; - -	skype_printf(ic, "SET USERSTATUS OFFLINE\n"); - -	while (ic->groupchats) { -		imcb_chat_free(ic->groupchats->data); -	} - -	for (i = 0; i < g_list_length(sd->groups); i++) { -		struct skype_group *sg = (struct skype_group *) g_list_nth_data(sd->groups, i); -		skype_group_free(sg, FALSE); -	} - -	if (sd->ssl) { -		ssl_disconnect(sd->ssl); -	} - -	g_free(sd->username); -	g_free(sd->handle); -	g_free(sd); -	ic->proto_data = NULL; -} - -static int skype_buddy_msg(struct im_connection *ic, char *who, char *message, -                           int flags) -{ -	char *ptr, *nick; -	int st; - -	/* Unused parameter */ -	flags = flags; - -	nick = g_strdup(who); -	ptr = strchr(nick, '@'); -	if (ptr) { -		*ptr = '\0'; -	} - -	if (!strncmp(who, "skypeconsole", 12)) { -		st = skype_printf(ic, "%s\n", message); -	} else { -		st = skype_printf(ic, "MESSAGE %s %s\n", nick, message); -	} -	g_free(nick); - -	return st; -} - -const struct skype_away_state *skype_away_state_by_name(char *name) -{ -	int i; - -	for (i = 0; skype_away_state_list[i].full_name; i++) { -		if (g_strcasecmp(skype_away_state_list[i].full_name, name) == 0) { -			return skype_away_state_list + i; -		} -	} - -	return NULL; -} - -static void skype_set_away(struct im_connection *ic, char *state_txt, -                           char *message) -{ -	const struct skype_away_state *state; - -	/* Unused parameter */ -	message = message; - -	if (state_txt == NULL) { -		state = skype_away_state_by_name("Online"); -	} else { -		state = skype_away_state_by_name(state_txt); -	} -	skype_printf(ic, "SET USERSTATUS %s\n", state->code); -} - -static GList *skype_away_states(struct im_connection *ic) -{ -	static GList *l; -	int i; - -	/* Unused parameter */ -	ic = ic; - -	if (l == NULL) { -		for (i = 0; skype_away_state_list[i].full_name; i++) { -			l = g_list_append(l, -			                  (void *) skype_away_state_list[i].full_name); -		} -	} - -	return l; -} - -static char *skype_set_display_name(set_t *set, char *value) -{ -	account_t *acc = set->data; -	struct im_connection *ic = acc->ic; - -	skype_printf(ic, "SET PROFILE FULLNAME %s\n", value); -	return value; -} - -static char *skype_set_mood_text(set_t *set, char *value) -{ -	account_t *acc = set->data; -	struct im_connection *ic = acc->ic; - -	skype_printf(ic, "SET PROFILE MOOD_TEXT %s\n", value); -	return value; -} - -static char *skype_set_balance(set_t *set, char *value) -{ -	account_t *acc = set->data; -	struct im_connection *ic = acc->ic; - -	skype_printf(ic, "GET PROFILE PSTN_BALANCE\n"); -	return value; -} - -static void skype_call(struct im_connection *ic, char *value) -{ -	char *nick = g_strdup(value); -	char *ptr = strchr(nick, '@'); - -	if (ptr) { -		*ptr = '\0'; -	} -	skype_printf(ic, "CALL %s\n", nick); -	g_free(nick); -} - -static void skype_hangup(struct im_connection *ic) -{ -	struct skype_data *sd = ic->proto_data; - -	if (sd->call_id) { -		skype_printf(ic, "SET CALL %s STATUS FINISHED\n", -		             sd->call_id); -		g_free(sd->call_id); -		sd->call_id = 0; -	} else { -		imcb_error(ic, "There are no active calls currently."); -	} -} - -static char *skype_set_call(set_t *set, char *value) -{ -	account_t *acc = set->data; -	struct im_connection *ic = acc->ic; - -	if (value) { -		skype_call(ic, value); -	} else { -		skype_hangup(ic); -	} -	return value; -} - -static void skype_add_buddy(struct im_connection *ic, char *who, char *group) -{ -	struct skype_data *sd = ic->proto_data; -	char *nick, *ptr; - -	nick = g_strdup(who); -	ptr = strchr(nick, '@'); -	if (ptr) { -		*ptr = '\0'; -	} - -	if (!group) { -		skype_printf(ic, "SET USER %s BUDDYSTATUS 2 Please authorize me\n", -		             nick); -		g_free(nick); -	} else { -		struct skype_group *sg = skype_group_by_name(ic, group); - -		if (!sg) { -			/* No such group, we need to create it, then have to -			 * add the user once it's created. */ -			skype_printf(ic, "CREATE GROUP %s\n", group); -			sd->pending_user = g_strdup(nick); -		} else { -			skype_printf(ic, "ALTER GROUP %d ADDUSER %s\n", sg->id, nick); -		} -	} -} - -static void skype_remove_buddy(struct im_connection *ic, char *who, char *group) -{ -	char *nick, *ptr; - -	/* Unused parameter */ -	group = group; - -	nick = g_strdup(who); -	ptr = strchr(nick, '@'); -	if (ptr) { -		*ptr = '\0'; -	} -	skype_printf(ic, "SET USER %s BUDDYSTATUS 1\n", nick); -	g_free(nick); -} - -void skype_chat_msg(struct groupchat *gc, char *message, int flags) -{ -	struct im_connection *ic = gc->ic; - -	/* Unused parameter */ -	flags = flags; - -	skype_printf(ic, "CHATMESSAGE %s %s\n", gc->title, message); -} - -void skype_chat_leave(struct groupchat *gc) -{ -	struct im_connection *ic = gc->ic; - -	skype_printf(ic, "ALTER CHAT %s LEAVE\n", gc->title); -	gc->data = (void *) TRUE; -} - -void skype_chat_invite(struct groupchat *gc, char *who, char *message) -{ -	struct im_connection *ic = gc->ic; -	char *ptr, *nick; - -	nick = g_strdup(who); -	ptr = strchr(nick, '@'); -	if (ptr) { -		*ptr = '\0'; -	} -	skype_printf(ic, "ALTER CHAT %s ADDMEMBERS %s\n", gc->title, nick); -	g_free(nick); -} - -void skype_chat_topic(struct groupchat *gc, char *message) -{ -	struct im_connection *ic = gc->ic; -	struct skype_data *sd = ic->proto_data; - -	skype_printf(ic, "ALTER CHAT %s SETTOPIC %s\n", -	             gc->title, message); -	sd->topic_wait = 1; -} - -struct groupchat *skype_chat_with(struct im_connection *ic, char *who) -{ -	struct skype_data *sd = ic->proto_data; -	char *ptr, *nick; - -	nick = g_strdup(who); -	ptr = strchr(nick, '@'); -	if (ptr) { -		*ptr = '\0'; -	} -	skype_printf(ic, "CHAT CREATE %s\n", nick); -	sd->groupchat_with = g_strdup(nick); -	g_free(nick); -	/* We create a fake chat for now. We will replace it with a real one in -	 * the real callback. */ -	return imcb_chat_new(ic, ""); -} - -static void skype_get_info(struct im_connection *ic, char *who) -{ -	struct skype_data *sd = ic->proto_data; -	char *ptr, *nick; - -	nick = g_strdup(who); -	ptr = strchr(nick, '@'); -	if (ptr) { -		*ptr = '\0'; -	} -	sd->is_info = TRUE; -	skype_printf(ic, "GET USER %s FULLNAME\n", nick); -	skype_printf(ic, "GET USER %s PHONE_HOME\n", nick); -	skype_printf(ic, "GET USER %s PHONE_OFFICE\n", nick); -	skype_printf(ic, "GET USER %s PHONE_MOBILE\n", nick); -	skype_printf(ic, "GET USER %s NROF_AUTHED_BUDDIES\n", nick); -	skype_printf(ic, "GET USER %s TIMEZONE\n", nick); -	skype_printf(ic, "GET USER %s LASTONLINETIMESTAMP\n", nick); -	skype_printf(ic, "GET USER %s SEX\n", nick); -	skype_printf(ic, "GET USER %s LANGUAGE\n", nick); -	skype_printf(ic, "GET USER %s COUNTRY\n", nick); -	skype_printf(ic, "GET USER %s PROVINCE\n", nick); -	skype_printf(ic, "GET USER %s CITY\n", nick); -	skype_printf(ic, "GET USER %s HOMEPAGE\n", nick); -	skype_printf(ic, "GET USER %s ABOUT\n", nick); -	/* -	 * Hack: we query the bithday property which is always a single line, -	 * so we can send the collected properties to the user when we have -	 * this one. -	 */ -	skype_printf(ic, "GET USER %s BIRTHDAY\n", nick); -} - -static void skype_init(account_t *acc) -{ -	set_t *s; - -	s = set_add(&acc->set, "server", SKYPE_DEFAULT_SERVER, set_eval_account, -	            acc); -	s->flags |= ACC_SET_OFFLINE_ONLY; - -	s = set_add(&acc->set, "port", SKYPE_DEFAULT_PORT, set_eval_int, acc); -	s->flags |= ACC_SET_OFFLINE_ONLY; - -	s = set_add(&acc->set, "display_name", NULL, skype_set_display_name, -	            acc); -	s->flags |= SET_NOSAVE | ACC_SET_ONLINE_ONLY; - -	s = set_add(&acc->set, "mood_text", NULL, skype_set_mood_text, acc); -	s->flags |= SET_NOSAVE | ACC_SET_ONLINE_ONLY; - -	s = set_add(&acc->set, "call", NULL, skype_set_call, acc); -	s->flags |= SET_NOSAVE | ACC_SET_ONLINE_ONLY; - -	s = set_add(&acc->set, "balance", NULL, skype_set_balance, acc); -	s->flags |= SET_NOSAVE | ACC_SET_ONLINE_ONLY; - -	s = set_add(&acc->set, "skypeout_offline", "true", set_eval_bool, acc); - -	s = set_add(&acc->set, "skypeconsole", "false", set_eval_bool, acc); -	s->flags |= ACC_SET_OFFLINE_ONLY; - -	s = set_add(&acc->set, "skypeconsole_receive", "false", set_eval_bool, -	            acc); -	s->flags |= ACC_SET_OFFLINE_ONLY; - -	s = set_add(&acc->set, "auto_join", "false", set_eval_bool, acc); -	s->flags |= ACC_SET_OFFLINE_ONLY; - -	s = set_add(&acc->set, "test_join", "false", set_eval_bool, acc); -	s->flags |= ACC_SET_OFFLINE_ONLY; - -	s = set_add(&acc->set, "show_moods", "false", set_eval_bool, acc); - -	s = set_add(&acc->set, "edit_prefix", "EDIT:", -	            NULL, acc); - -	s = set_add(&acc->set, "read_groups", "false", set_eval_bool, acc); -} - -#if BITLBEE_VERSION_CODE > BITLBEE_VER(3, 0, 1) -GList *skype_buddy_action_list(bee_user_t *bu) -{ -	static GList *ret; - -	/* Unused parameter */ -	bu = bu; - -	if (ret == NULL) { -		static const struct buddy_action ba[2] = { -			{ "CALL", "Initiate a call" }, -			{ "HANGUP", "Hang up a call" }, -		}; -		int i; - -		for (i = 0; i < ARRAY_SIZE(ba); i++) { -			ret = g_list_prepend(ret, (void *) (ba + i)); -		} -	} - -	return ret; -} - -void *skype_buddy_action(struct bee_user *bu, const char *action, char * const args[], void *data) -{ -	/* Unused parameters */ -	args = args; -	data = data; - -	if (!g_strcasecmp(action, "CALL")) { -		skype_call(bu->ic, bu->handle); -	} else if (!g_strcasecmp(action, "HANGUP")) { -		skype_hangup(bu->ic); -	} - -	return NULL; -} -#endif - -void init_plugin(void) -{ -	struct prpl *ret = g_new0(struct prpl, 1); - -	ret->name = "skype"; -	ret->login = skype_login; -	ret->init = skype_init; -	ret->logout = skype_logout; -	ret->buddy_msg = skype_buddy_msg; -	ret->get_info = skype_get_info; -	ret->away_states = skype_away_states; -	ret->set_away = skype_set_away; -	ret->add_buddy = skype_add_buddy; -	ret->remove_buddy = skype_remove_buddy; -	ret->chat_msg = skype_chat_msg; -	ret->chat_leave = skype_chat_leave; -	ret->chat_invite = skype_chat_invite; -	ret->chat_with = skype_chat_with; -	ret->handle_cmp = g_strcasecmp; -	ret->chat_topic = skype_chat_topic; -#if BITLBEE_VERSION_CODE > BITLBEE_VER(3, 0, 1) -	ret->buddy_action_list = skype_buddy_action_list; -	ret->buddy_action = skype_buddy_action; -#endif -	register_protocol(ret); -} - -struct plugin_info *init_plugin_info(void) -{ -	static struct plugin_info info = { -		BITLBEE_ABI_VERSION_CODE, -		"skype", -		BITLBEE_VERSION, -		"Skype protocol plugin", -		NULL, -		NULL -	}; - -	return &info; -} | 
