Marco Trevisan (Treviño) wrote:
>  - Using the "prpl-msn" in standard mode (so, like this client generally
>    does) with my account (that is a very "heavy" one since I've to load
>    something like 450 contacts) I can connect, load contacts and so on
>    but I really I can sign-in in the server (I mean that I'm visible to
>    others and I get notification/messages) only if I run the client in
>    valgrind and with libpurple debug enabled! Yes, this is so weird as
>    it seems!
>    I only once was able to connect to it running it in the normal way...
>    Nevertheless, the "valgrind+debug" way seems to work always! :|

Ok, It seems that this was a my fault and now I've fixed it too :P

On the first days of the development I found that the only way to
complete the connection was not to create a new fd handler if the client
was asked for one already in use, but using the previous one (also Adium
seems to do this). And I the "connection-fix" patch .
After implementing this I was able to connect, but... Not with my
account using the "pure" MSN protocol (I mean, without using the
"http_method").

Now, after some debugging, I've reverted the patch I did, and do you
know what? The client connects correctly to all the protocols I've
tested and with all the accounts I have. I neither get any error or
segmentation fault.

You can find as attachment what I think can be hopefully considered the
"final version" with the patch against the previous version.

Bye ;)

-- 
Treviño's World - Life and Linux
http://www.3v1n0.net/
/*
 * nullclient-ecore.c - an ecore libpurple client test.
 * Copyright (C) 2009, Marco Trevisan (Treviño) <m...@3v1n0.net>
 * --
 *
 * Pidgin is the legal property of its developers, whose names are too numerous
 * to list here.  Please refer to the COPYRIGHT file distributed with this
 * source distribution.
 *
 * 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  02111-1301  USA
 *
 */

#include "purple.h"

#include <Ecore.h>
#include <eina_list.h>

#include <signal.h>
#include <string.h>
#include <unistd.h>

// Client related settings
#define CUSTOM_USER_DIRECTORY  "/dev/null"
#define CUSTOM_PLUGIN_PATH     ""
#define PLUGIN_SAVE_PREF       "/purple/nullclient-ecore/plugins/saved"
#define UI_ID                  "nullclient-ecore"


/*** Event uiops ***/
static guint loopID = 0;
Eina_List *timerList = NULL;
Eina_List *inputList = NULL;

typedef struct _timerRef {
	Ecore_Timer *timer;
	guint id;
} timerRef;

typedef struct _fdRef {
	PurpleInputFunction function;
	Ecore_Fd_Handler_Flags condition;
	Ecore_Fd_Handler *fdh;
	guint id;
	void *data;
} fdRef;

static guint ecore_purple_timeout_add(guint interval, GSourceFunc function, gpointer data) {
	timerRef *ref = calloc(1, sizeof(timerRef));
	double timer_interval = ((double) interval)/1000;
	
	ref->timer = ecore_timer_add(timer_interval, function, data);
	ref->id = ++loopID;
	timerList = eina_list_append(timerList, ref);
	
	return ref->id;
}

static gboolean ecore_purple_timeout_remove(guint handle) {
	timerRef *ref;
	Eina_List *tlist;
	
	EINA_LIST_FOREACH(timerList, tlist, ref) {
		if (ref->id == handle) {
			if (ref->timer)
				ecore_timer_del(ref->timer);
			timerList = eina_list_remove(timerList, ref);
			free(ref);
			return TRUE;
		}
	}
	
	return FALSE;
}

static int ecore_purple_input_cb(void *data, Ecore_Fd_Handler *fd_handler) {
	fdRef *ref = data;
	PurpleInputCondition purple_cond = 0;

	if (ref->condition & ECORE_FD_READ)
		purple_cond |= PURPLE_INPUT_READ;

	if (ref->condition & ECORE_FD_WRITE)
		purple_cond |= PURPLE_INPUT_WRITE;

	if (ref->function && fd_handler) {
		ref->function(ref->data, ecore_main_fd_handler_fd_get(fd_handler), purple_cond);
		return 1;
	} else
		return 0;
}

static guint ecore_purple_input_add(int fd, PurpleInputCondition condition,
							 PurpleInputFunction func, gpointer user_data) {
	Ecore_Fd_Handler_Flags fdflags = 0;
	fdRef *ref;
	
	if (fd < 0)
		return ++loopID;

	if (condition & PURPLE_INPUT_READ)
		fdflags |= ECORE_FD_READ;

	if (condition & PURPLE_INPUT_WRITE)
		fdflags |= ECORE_FD_WRITE;

	ref = calloc(1, sizeof(fdRef));
	ref->fdh = ecore_main_fd_handler_add(fd, fdflags, ecore_purple_input_cb, ref, NULL, NULL);
	ref->id = ++loopID;
	ref->data = user_data;
	ref->function = func;
	inputList = eina_list_append(inputList, ref);

	return ref->id;
}

static gboolean ecore_purple_input_remove(guint handle) {
	fdRef *ref;
	Eina_List *tlist;
	
	EINA_LIST_FOREACH(inputList, tlist, ref) {
		if (ref && ref->id == handle) {
			if (ref->fdh)
				ecore_main_fd_handler_del(ref->fdh);
			inputList = eina_list_remove(inputList, ref);
			free(ref);
			return TRUE;
		}
	}
	
	return FALSE;
}

static PurpleEventLoopUiOps ecore_eventloops =
{
	ecore_purple_timeout_add,
	ecore_purple_timeout_remove,
	ecore_purple_input_add,
	ecore_purple_input_remove,
	NULL,
	NULL,
	NULL,
	NULL,
	NULL
};
/*** End of the eventloop functions. ***/

/*** Conversation uiops ***/
static void
null_write_conv(PurpleConversation *conv, const char *who, const char *alias,
			const char *message, PurpleMessageFlags flags, time_t mtime)
{
	const char *name;
	if (alias && *alias)
		name = alias;
	else if (who && *who)
		name = who;
	else
		name = NULL;

	printf("(%s) %s %s: %s\n", purple_conversation_get_name(conv),
			purple_utf8_strftime("(%H:%M:%S)", localtime(&mtime)),
			name, message);
}

static PurpleConversationUiOps null_conv_uiops = 
{
	NULL,                      /* create_conversation  */
	NULL,                      /* destroy_conversation */
	NULL,                      /* write_chat           */
	NULL,                      /* write_im             */
	null_write_conv,           /* write_conv           */
	NULL,                      /* chat_add_users       */
	NULL,                      /* chat_rename_user     */
	NULL,                      /* chat_remove_users    */
	NULL,                      /* chat_update_user     */
	NULL,                      /* present              */
	NULL,                      /* has_focus            */
	NULL,                      /* custom_smiley_add    */
	NULL,                      /* custom_smiley_write  */
	NULL,                      /* custom_smiley_close  */
	NULL,                      /* send_confirm         */
	NULL,
	NULL,
	NULL,
	NULL
};

static void
null_ui_init(void)
{
	/**
	 * This should initialize the UI components for all the modules. Here we
	 * just initialize the UI for conversations.
	 */
	purple_conversations_set_ui_ops(&null_conv_uiops);

}

static PurpleCoreUiOps null_core_uiops =
{
	NULL,
	NULL,
	null_ui_init,
	NULL,
	NULL,
	NULL,
	NULL,
	NULL
};

static void
init_libpurple(void)
{
	/* Set a custom user directory (optional) */
	purple_util_set_user_dir(CUSTOM_USER_DIRECTORY);

	/* We do not want any debugging for now to keep the noise to a minimum. */
	purple_debug_set_enabled(TRUE);

	/* Set the core-uiops, which is used to
	 * 	- initialize the ui specific preferences.
	 * 	- initialize the debug ui.
	 * 	- initialize the ui components for all the modules.
	 * 	- uninitialize the ui components for all the modules when the core terminates.
	 */
	purple_core_set_ui_ops(&null_core_uiops);

	/* Set the uiops for the eventloop. */
	purple_eventloop_set_ui_ops(&ecore_eventloops);

	/* Set path to search for plugins. The core (libpurple) takes care of loading the
	 * core-plugins, which includes the protocol-plugins. So it is not essential to add
	 * any path here, but it might be desired, especially for ui-specific plugins. */
	purple_plugins_add_search_path(CUSTOM_PLUGIN_PATH);

	/* Now that all the essential stuff has been set, let's try to init the core. It's
	 * necessary to provide a non-NULL name for the current ui to the core. This name
	 * is used by stuff that depends on this ui, for example the ui-specific plugins. */
	if (!purple_core_init(UI_ID)) {
		/* Initializing the core failed. Terminate. */
		fprintf(stderr,
				"libpurple initialization failed. Dumping core.\n"
				"Please report this!\n");
		abort();
	}

	/* Create and load the buddylist. */
	purple_set_blist(purple_blist_new());
	purple_blist_load();

	/* Load the preferences. */
	purple_prefs_load();

	/* Load the desired plugins. The client should save the list of loaded plugins in
	 * the preferences using purple_plugins_save_loaded(PLUGIN_SAVE_PREF) */
	purple_plugins_load_saved(PLUGIN_SAVE_PREF);

	/* Load the pounces. */
	purple_pounces_load();
}

static void
signed_on(PurpleConnection *gc, gpointer null)
{
	PurpleBlistNode *node;
	
	PurpleAccount *account = purple_connection_get_account(gc);
	printf("Account connected: %s %s\n", account->username, account->protocol_id);
}

static void loopListsfree(void) {
	fdRef *fref;
	timerRef *tref;
	Eina_List *tlist;
	
	EINA_LIST_FOREACH(timerList, tlist, fref) {
		if (fref && fref->fdh)
			ecore_main_fd_handler_del(fref->fdh);
		timerList = eina_list_remove(timerList, fref);
		free(fref);
	}
	
	EINA_LIST_FOREACH(inputList, tlist, tref) {
		if (tref && tref->timer)
			ecore_timer_del(tref->timer);
		inputList = eina_list_remove(inputList, tref);
		free(tref);
	}
	
	eina_list_free(timerList);
	eina_list_free(inputList);
}

static void signed_off(PurpleConnection *gc, gpointer null)
{
	printf("Connection is gone :(...\n ");
	ecore_main_loop_quit();
}

static void
connect_to_signals_for_demonstration_purposes_only(void)
{
	static int handle;
	purple_signal_connect(purple_connections_get_handle(), "signed-on", &handle,
				PURPLE_CALLBACK(signed_on), NULL);
				
	static int handle2;
	purple_signal_connect(purple_connections_get_handle(), "signed-off", &handle2,
				PURPLE_CALLBACK(signed_off), NULL);
}

int main(int argc, char *argv[])
{
	const char *prpl;
	char name[128];
	char *password;
	PurpleAccount *account;
	PurpleSavedStatus *status;
	char *res;

	/* libpurple's built-in DNS resolution forks processes to perform
	 * blocking lookups without blocking the main process.  It does not
	 * handle SIGCHLD itself, so if the UI does not you quickly get an army
	 * of zombie subprocesses marching around.
	 */
	signal(SIGCHLD, SIG_IGN);
	
	ecore_init();
	eina_list_init();

	init_libpurple();

	printf("libpurple initialized.\n");

	int i, num;
	GList *iter;
	Eina_List *names = NULL;
	iter = purple_plugins_get_protocols();
	for (i = 0; iter; iter = iter->next) {
		PurplePlugin *plugin = iter->data;
		PurplePluginInfo *info = plugin->info;
		if (info && info->name) {
			printf("\t%d: %s\n", i++, info->name);
			names = eina_list_append(names, info->id);
		}
	}
	
	printf("Select the protocol [0-%d]: ", i-1);
	res = fgets(name, sizeof(name), stdin);
	if (!res) {
		fprintf(stderr, "Failed to gets protocol selection.");
		abort();
	}
	sscanf(name, "%d", &num);
	prpl = eina_list_nth(names, num);

	printf("Username: ");
	res = fgets(name, sizeof(name), stdin);
	if (!res) {
		fprintf(stderr, "Failed to read user name.");
		abort();
	}
	name[strlen(name) - 1] = 0;  /* strip the \n at the end */

	/* Create the account */
	account = purple_account_new(name, prpl);

	/* Get the password for the account */
	password = getpass("Password: ");
	purple_account_set_password(account, password);

	/* It's necessary to enable the account first. */
	purple_account_set_enabled(account, UI_ID, TRUE);

	/* Now, to connect the account(s), create a status and activate it. */
	status = purple_savedstatus_new(NULL, PURPLE_STATUS_INVISIBLE);
	purple_savedstatus_activate(status);

	connect_to_signals_for_demonstration_purposes_only();

	ecore_main_loop_begin();

	ecore_main_loop_quit();
	
	eina_list_shutdown();
	
	return 0;
}
--- /a/nullclient-ecore.c	2009-04-15 18:34:47.000000000 +0200
+++ /b/nullclient-ecore.c	2009-04-15 18:39:37.000000000 +0200
@@ -103,42 +109,25 @@
 }
 
 static guint ecore_purple_input_add(int fd, PurpleInputCondition condition,
-				    PurpleInputFunction func, gpointer user_data) {
+							 PurpleInputFunction func, gpointer user_data) {
 	Ecore_Fd_Handler_Flags fdflags = 0;
-	Eina_List *tlist;
 	fdRef *ref;
 	
 	if (fd < 0)
 		return ++loopID;
 
-	if (condition & PURPLE_INPUT_READ) {
+	if (condition & PURPLE_INPUT_READ)
 		fdflags |= ECORE_FD_READ;
-		fdflags |= ECORE_FD_ERROR;
-	}
 
-	if (condition & PURPLE_INPUT_WRITE) {
+	if (condition & PURPLE_INPUT_WRITE)
 		fdflags |= ECORE_FD_WRITE;
-		fdflags |= ECORE_FD_ERROR;
-	}
-
-	EINA_LIST_FOREACH(inputList, tlist, ref) {
-		if (ref && ref->fdh) {
-			if (ecore_main_fd_handler_fd_get(ref->fdh) == fd)
-				break;
-		}
-	}
-
-	if (!ref) {
-		ref = calloc(1, sizeof(fdRef));
-		ref->fdh = ecore_main_fd_handler_add(fd, fdflags, ecore_purple_input_cb, ref, NULL, NULL);
-		inputList = eina_list_append(inputList, ref);
-	} else {
-		ecore_main_fd_handler_active_set(ref->fdh, fdflags);
-	}
 
+	ref = calloc(1, sizeof(fdRef));
+	ref->fdh = ecore_main_fd_handler_add(fd, fdflags, ecore_purple_input_cb, ref, NULL, NULL);
 	ref->id = ++loopID;
 	ref->data = user_data;
 	ref->function = func;
+	inputList = eina_list_append(inputList, ref);
 
 	return ref->id;
 }
------------------------------------------------------------------------------
This SF.net email is sponsored by:
High Quality Requirements in a Collaborative Environment.
Download a free trial of Rational Requirements Composer Now!
http://p.sf.net/sfu/www-ibm-com
_______________________________________________
enlightenment-devel mailing list
enlightenment-devel@lists.sourceforge.net
https://lists.sourceforge.net/lists/listinfo/enlightenment-devel

Reply via email to