tcl.c

来自「Linux下的多协议即时通讯程序源代码」· C语言 代码 · 共 554 行 · 第 1/2 页

C
554
字号
/** * @file tcl.c Purple Tcl plugin bindings * * purple * * Copyright (C) 2003 Ethan Blanton <eblanton@cs.purdue.edu> * * 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 "config.h"#include <tcl.h>#ifdef HAVE_TK#include <tk.h>#endif#include <stdio.h>#include <sys/types.h>#include <sys/stat.h>#include <unistd.h>#include <string.h>#include "tcl_glib.h"#include "tcl_purple.h"#include "internal.h"#include "connection.h"#include "plugin.h"#include "signals.h"#include "debug.h"#include "util.h"#include "version.h"struct tcl_plugin_data {	PurplePlugin *plugin;	Tcl_Interp *interp;};PurpleStringref *PurpleTclRefAccount;PurpleStringref *PurpleTclRefConnection;PurpleStringref *PurpleTclRefConversation;PurpleStringref *PurpleTclRefPointer;PurpleStringref *PurpleTclRefPlugin;PurpleStringref *PurpleTclRefPresence;PurpleStringref *PurpleTclRefStatus;PurpleStringref *PurpleTclRefStatusAttr;PurpleStringref *PurpleTclRefStatusType;PurpleStringref *PurpleTclRefXfer;static GHashTable *tcl_plugins = NULL;PurplePlugin *_tcl_plugin;static gboolean tcl_loaded = FALSE;PurplePlugin *tcl_interp_get_plugin(Tcl_Interp *interp){	struct tcl_plugin_data *data;	if (tcl_plugins == NULL)		return NULL;	data = g_hash_table_lookup(tcl_plugins, (gpointer)interp);	return data != NULL ? data->plugin : NULL;}static int tcl_init_interp(Tcl_Interp *interp){	char *rcfile;	char init[] =		"namespace eval ::purple {\n"		"	namespace export account buddy connection conversation\n"		"	namespace export core debug notify prefs send_im\n"		"	namespace export signal unload\n"		"	namespace eval _callback { }\n"		"\n"		"	proc conv_send { account who text } {\n"		"		set gc [purple::account connection $account]\n"		"		set convo [purple::conversation new $account $who]\n"		"		set myalias [purple::account alias $account]\n"		"\n"		"		if {![string length $myalias]} {\n"		"			set myalias [purple::account username $account]\n"		"		}\n"		"\n"		"		purple::send_im $gc $who $text\n"		"		purple::conversation write $convo	send $myalias $text\n"		"	}\n"		"}\n"		"\n"		"proc bgerror { message } {\n"		"	global errorInfo\n"		"	purple::notify -error \"Tcl Error\" \"Tcl Error: $message\" \"$errorInfo\"\n"		"}\n";	if (Tcl_EvalEx(interp, init, -1, TCL_EVAL_GLOBAL) != TCL_OK) {		return 1;	}	Tcl_SetVar(interp, "argc", "0", TCL_GLOBAL_ONLY);	Tcl_SetVar(interp, "argv0", "purple", TCL_GLOBAL_ONLY);	Tcl_SetVar(interp, "tcl_interactive", "0", TCL_GLOBAL_ONLY);	rcfile = g_strdup_printf("%s" G_DIR_SEPARATOR_S "tclrc", purple_user_dir());	Tcl_SetVar(interp, "tcl_rcFileName", rcfile, TCL_GLOBAL_ONLY);	g_free(rcfile);	Tcl_SetVar(interp, "::purple::version", VERSION, TCL_GLOBAL_ONLY);	Tcl_SetVar(interp, "::purple::user_dir", purple_user_dir(), TCL_GLOBAL_ONLY);#ifdef HAVE_TK	Tcl_SetVar(interp, "::purple::tk_available", "1", TCL_GLOBAL_ONLY);#else	Tcl_SetVar(interp, "::purple::tk_available", "0", TCL_GLOBAL_ONLY);#endif /* HAVE_TK */	Tcl_CreateObjCommand(interp, "::purple::account", tcl_cmd_account, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::buddy", tcl_cmd_buddy, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::cmd", tcl_cmd_cmd, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::connection", tcl_cmd_connection, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::conversation", tcl_cmd_conversation, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::core", tcl_cmd_core, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::debug", tcl_cmd_debug, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::notify", tcl_cmd_notify, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::plugins", tcl_cmd_plugins, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::prefs", tcl_cmd_prefs, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::presence", tcl_cmd_presence, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::send_im", tcl_cmd_send_im, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::savedstatus", tcl_cmd_savedstatus, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::signal", tcl_cmd_signal, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::status", tcl_cmd_status, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::status_attr", tcl_cmd_status_attr, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::status_type", tcl_cmd_status_type, (ClientData)NULL, NULL);	Tcl_CreateObjCommand(interp, "::purple::unload", tcl_cmd_unload, (ClientData)NULL, NULL);	return 0;}static Tcl_Interp *tcl_create_interp(){	Tcl_Interp *interp;	interp = Tcl_CreateInterp();	if (Tcl_Init(interp) == TCL_ERROR) {		Tcl_DeleteInterp(interp);		return NULL;	}	if (tcl_init_interp(interp)) {		Tcl_DeleteInterp(interp);		return NULL;	}	Tcl_StaticPackage(interp, "purple", tcl_init_interp, NULL);	return interp;}static gboolean tcl_probe_plugin(PurplePlugin *plugin){	PurplePluginInfo *info;	Tcl_Interp *interp;	Tcl_Parse parse;	Tcl_Obj *result, **listitems;	struct stat st;	FILE *fp;	char *buf, *cur;	const char *next;	int len, found = 0, err = 0, nelems;	gboolean status = FALSE;	if ((fp = g_fopen(plugin->path, "r")) == NULL)		return FALSE;	if (fstat(fileno(fp), &st)) {		fclose(fp);		return FALSE;	}	len = st.st_size;	buf = g_malloc(len + 1);	cur = buf;	while (fgets(cur, GPOINTER_TO_INT(buf) - (buf - cur), fp)) {		cur += strlen(cur);		if (feof(fp))			break;	}	if (ferror(fp)) {		purple_debug(PURPLE_DEBUG_ERROR, "tcl", "error reading %s (%s)\n", plugin->path, strerror(errno));		g_free(buf);		fclose(fp);		return FALSE;	}	fclose(fp);	if ((interp = tcl_create_interp()) == NULL) {		return FALSE;	}	next = buf;	do {		if (Tcl_ParseCommand(interp, next, len, 0, &parse) == TCL_ERROR) {			purple_debug(PURPLE_DEBUG_ERROR, "tcl", "parse error in %s: %s\n", plugin->path,				   Tcl_GetString(Tcl_GetObjResult(interp)));			err = 1;			break;		}		if (parse.tokenPtr[0].type == TCL_TOKEN_SIMPLE_WORD		    && !strncmp(parse.tokenPtr[0].start, "proc", parse.tokenPtr[0].size)) {			if (!strncmp(parse.tokenPtr[2].start, "plugin_init", parse.tokenPtr[2].size)) {				if (Tcl_EvalEx(interp, parse.commandStart, parse.commandSize, TCL_EVAL_GLOBAL) != TCL_OK) {					Tcl_FreeParse(&parse);					break;				}				found = 1;				/* We'll continue parsing the file, just in case */			}		}		len -= (parse.commandStart + parse.commandSize) - next;		next = parse.commandStart + parse.commandSize;		Tcl_FreeParse(&parse);	} while (len);	if (found && !err) {		if (Tcl_EvalEx(interp, "plugin_init", -1, TCL_EVAL_GLOBAL) == TCL_OK) {			result = Tcl_GetObjResult(interp);			if (Tcl_ListObjGetElements(interp, result, &nelems, &listitems) == TCL_OK) {				if ((nelems == 6) || (nelems == 7)) {					info = g_new0(PurplePluginInfo, 1);					info->magic = PURPLE_PLUGIN_MAGIC;					info->major_version = PURPLE_MAJOR_VERSION;					info->minor_version = PURPLE_MINOR_VERSION;					info->type = PURPLE_PLUGIN_STANDARD;					info->dependencies = g_list_append(info->dependencies, "core-tcl");					info->name = g_strdup(Tcl_GetString(listitems[0]));					info->version = g_strdup(Tcl_GetString(listitems[1]));					info->summary = g_strdup(Tcl_GetString(listitems[2]));					info->description = g_strdup(Tcl_GetString(listitems[3]));					info->author = g_strdup(Tcl_GetString(listitems[4]));					info->homepage = g_strdup(Tcl_GetString(listitems[5]));					if (nelems == 6)						info->id = g_strdup_printf("tcl-%s", Tcl_GetString(listitems[0]));					else if (nelems == 7)						info->id = g_strdup_printf("tcl-%s", Tcl_GetString(listitems[6]));					plugin->info = info;					if (purple_plugin_register(plugin))						status = TRUE;				}			}		}	}	Tcl_DeleteInterp(interp);	g_free(buf);	return status;}static gboolean tcl_load_plugin(PurplePlugin *plugin){	struct tcl_plugin_data *data;

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?