Mercurial > pidgin
annotate plugins/perl/perl.c @ 5252:7c9393b1ea36
[gaim-migrate @ 5624]
"Shouldn't this give an error to the user so they know their stuff isn't
being saved?" -Rob McQueen
I'm not doing so well tonight.
committer: Tailor Script <tailor@pidgin.im>
author | Mark Doliner <mark@kingant.net> |
---|---|
date | Mon, 28 Apr 2003 07:21:19 +0000 |
parents | 6d1707dc8c3d |
children | 2af3224b329a |
rev | line source |
---|---|
5205 | 1 /* |
2 * gaim | |
3 * | |
4 * Copyright (C) 1998-1999, Mark Spencer <markster@marko.net> | |
5 * | |
6 * This program is free software; you can redistribute it and/or modify | |
7 * it under the terms of the GNU General Public License as published by | |
8 * the Free Software Foundation; either version 2 of the License, or | |
9 * (at your option) any later version. | |
10 * | |
11 * This program is distributed in the hope that it will be useful, | |
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
14 * GNU General Public License for more details. | |
15 * | |
16 * You should have received a copy of the GNU General Public License | |
17 * along with this program; if not, write to the Free Software | |
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | |
19 * | |
20 * This was taken almost exactly from X-Chat. The power of the GPL. | |
21 * Translated from X-Chat to Gaim by Eric Warmenhoven. | |
22 * Originally by Erik Scrafford <eriks@chilisoft.com>. | |
23 * X-Chat Copyright (C) 1998 Peter Zelezny. | |
24 * | |
25 */ | |
26 | |
27 #ifdef HAVE_CONFIG_H | |
28 #include <config.h> | |
29 #ifdef DEBUG | |
30 #undef DEBUG | |
31 #endif | |
32 #endif | |
33 #undef PACKAGE | |
34 | |
35 #define group perl_group | |
36 #ifdef _WIN32 | |
37 /* This took me an age to figure out.. without this __declspec(dllimport) | |
38 * will be ignored. | |
39 */ | |
40 #define HASATTRIBUTE | |
41 #endif | |
42 #include <EXTERN.h> | |
43 #ifndef _SEM_SEMUN_UNDEFINED | |
44 #define HAS_UNION_SEMUN | |
45 #endif | |
46 #include <perl.h> | |
47 #include <XSUB.h> | |
48 #ifndef _WIN32 | |
49 #include <sys/mman.h> | |
50 #endif | |
51 #include <sys/types.h> | |
52 #include <sys/stat.h> | |
53 #include <fcntl.h> | |
54 #undef PACKAGE | |
55 #include <stdio.h> | |
56 #ifndef _WIN32 | |
57 #include <dirent.h> | |
58 #else | |
59 /* We're using perl's win32 port of this */ | |
60 #define dirent direct | |
61 #endif | |
62 #include <string.h> | |
63 | |
64 #undef group | |
65 | |
66 /* perl module support */ | |
67 #ifdef OLD_PERL | |
68 extern void boot_DynaLoader _((CV * cv)); | |
69 #else | |
70 extern void boot_DynaLoader _((pTHX_ CV * cv)); /* perl is so wacky */ | |
71 #endif | |
72 | |
73 #undef _ | |
74 #ifdef DEBUG | |
75 #undef DEBUG | |
76 #endif | |
77 #ifdef _WIN32 | |
78 #undef pipe | |
79 #endif | |
80 #include "gaim.h" | |
81 #include "prpl.h" | |
82 #include "sound.h" | |
83 | |
84 #ifndef call_pv | |
85 # define call_pv(i,j) perl_call_pv((i), (j)) | |
86 #endif | |
87 | |
88 #define PERL_PLUGIN_ID "core-perl" | |
89 | |
90 struct perlscript { | |
91 char *name; | |
92 char *version; | |
93 char *shutdowncallback; /* bleh */ | |
94 GaimPlugin *plug; | |
95 }; | |
96 | |
97 struct _perl_event_handlers { | |
98 char *event_type; | |
99 char *handler_name; | |
100 GaimPlugin *plug; | |
101 }; | |
102 | |
103 struct _perl_timeout_handlers { | |
104 char *handler_name; | |
105 char *handler_args; | |
106 gint iotag; | |
107 GaimPlugin *plug; | |
108 }; | |
109 | |
110 static GList *perl_list = NULL; | |
111 static GList *perl_timeout_handlers = NULL; | |
112 static GList *perl_event_handlers = NULL; | |
113 static PerlInterpreter *my_perl = NULL; | |
114 static void perl_init(); | |
115 | |
116 /* dealing with gaim */ | |
117 XS(XS_GAIM_register); /* set up hooks for script */ | |
118 XS(XS_GAIM_get_info); /* version, last to attempt signon, protocol */ | |
119 XS(XS_GAIM_print); /* lemme figure this one out... */ | |
120 XS(XS_GAIM_write_to_conv); /* write into conversation window */ | |
121 | |
122 /* list stuff */ | |
123 XS(XS_GAIM_buddy_list); /* all buddies */ | |
124 XS(XS_GAIM_online_list); /* online buddies */ | |
125 | |
126 /* server stuff */ | |
127 XS(XS_GAIM_command); /* send command to server */ | |
128 XS(XS_GAIM_user_info); /* given name, return struct buddy members */ | |
129 XS(XS_GAIM_print_to_conv); /* send message to someone */ | |
130 XS(XS_GAIM_print_to_chat); /* send message to chat room */ | |
131 XS(XS_GAIM_serv_send_im); /* send message to someone (but do not display) */ | |
132 | |
133 /* handler commands */ | |
134 XS(XS_GAIM_add_event_handler); /* when servers talk */ | |
135 XS(XS_GAIM_remove_event_handler); /* remove a handler */ | |
136 XS(XS_GAIM_add_timeout_handler); /* figure it out */ | |
137 | |
138 /* play sound */ | |
139 XS(XS_GAIM_play_sound); /*play a sound */ | |
140 | |
141 static void | |
142 #ifdef OLD_PERL | |
143 xs_init() | |
144 #else | |
145 xs_init(pTHX) | |
146 #endif | |
147 { | |
148 char *file = __FILE__; | |
149 | |
150 /* This one allows dynamic loading of perl modules in perl | |
151 scripts by the 'use perlmod;' construction*/ | |
152 newXS ("DynaLoader::boot_DynaLoader", boot_DynaLoader, file); | |
153 | |
154 /* load up all the custom Gaim perl functions */ | |
155 newXS ("GAIM::register", XS_GAIM_register, "GAIM"); | |
156 newXS ("GAIM::get_info", XS_GAIM_get_info, "GAIM"); | |
157 newXS ("GAIM::print", XS_GAIM_print, "GAIM"); | |
158 newXS ("GAIM::write_to_conv", XS_GAIM_write_to_conv, "GAIM"); | |
159 | |
160 newXS ("GAIM::buddy_list", XS_GAIM_buddy_list, "GAIM"); | |
161 newXS ("GAIM::online_list", XS_GAIM_online_list, "GAIM"); | |
162 | |
163 newXS ("GAIM::command", XS_GAIM_command, "GAIM"); | |
164 newXS ("GAIM::user_info", XS_GAIM_user_info, "GAIM"); | |
165 newXS ("GAIM::print_to_conv", XS_GAIM_print_to_conv, "GAIM"); | |
166 newXS ("GAIM::print_to_chat", XS_GAIM_print_to_chat, "GAIM"); | |
167 newXS ("GAIM::serv_send_im", XS_GAIM_serv_send_im, "GAIM"); | |
168 | |
169 newXS ("GAIM::add_event_handler", XS_GAIM_add_event_handler, "GAIM"); | |
170 newXS ("GAIM::remove_event_handler", XS_GAIM_remove_event_handler, "GAIM"); | |
171 newXS ("GAIM::add_timeout_handler", XS_GAIM_add_timeout_handler, "GAIM"); | |
172 | |
173 newXS ("GAIM::play_sound", XS_GAIM_play_sound, "GAIM"); | |
174 } | |
175 | |
176 static char * | |
177 escape_quotes(const char *buf) | |
178 { | |
179 static char *tmp_buf = NULL; | |
180 const char *i; | |
181 char *j; | |
182 | |
183 if (tmp_buf) | |
184 g_free(tmp_buf); | |
185 | |
186 tmp_buf = g_malloc(strlen(buf) * 2 + 1); | |
187 | |
188 for (i = buf, j = tmp_buf; *i; i++, j++) { | |
189 if (*i == '\'' || *i == '\\') | |
190 *j++ = '\\'; | |
191 | |
192 *j = *i; | |
193 } | |
194 | |
195 *j = '\0'; | |
196 | |
197 return tmp_buf; | |
198 } | |
199 | |
200 /* | |
201 2003/02/06: execute_perl modified by Mark Doliner <mark@kingant.net> | |
202 Pass parameters by pushing them onto the stack rather than | |
203 passing an array of strings. This way, perl scripts can | |
204 modify the parameters and we can get the changed values | |
205 and then shoot ourselves. I mean, uh, use them. | |
206 | |
207 2001/06/14: execute_perl replaced by Martin Persson <mep@passagen.se> | |
208 previous use of perl_eval leaked memory, replaced with | |
209 a version that uses perl_call instead | |
210 | |
211 30/11/2002: execute_perl modified by Eric Timme <timothy@voidnet.com> | |
212 args changed to char** so that we can have preparsed | |
213 arguments again, and many headaches ensued! This essentially | |
214 means we replaced one hacked method with a messier hacked | |
215 method out of perceived necessity. Formerly execute_perl | |
216 required a single char_ptr, and it would insert it into an | |
217 array of character pointers and NULL terminate the new array. | |
218 Now we have to pass in pre-terminated character pointer arrays | |
219 to accomodate functions that want to pass in multiple arguments. | |
220 | |
221 Previously arguments were preparsed because an argument list | |
222 was constructed in the form 'arg one','arg two' and was | |
223 executed via a call like &funcname(arglist) (see .59.x), so | |
224 the arglist was magically pre-parsed because of the method. | |
225 With Martin Persson's change to perl_call we now need to | |
226 use a null terminated list of character pointers for arguments | |
227 if we wish them to be parsed. Lacking a better way to allow | |
228 for both single arguments and many I created a NULL terminated | |
229 array in every function that called execute_perl and passed | |
230 that list into the function. In the former version a single | |
231 character pointer was passed in, and was placed into an array | |
232 of character pointers with two elements, with a NULL element | |
233 tacked onto the back, but this method no longer seemed prudent. | |
234 | |
235 Enhancements in the future might be to get rid of pre-declaring | |
236 the array sizes? I am not comfortable enough with this | |
237 subject to attempt it myself and hope it to stand the test | |
238 of time. | |
239 */ | |
240 | |
241 static int | |
242 execute_perl(const char *function, int argc, char **args) | |
243 { | |
244 int count = 0, i, ret_value = 1; | |
245 SV *sv_args[argc]; | |
246 STRLEN na; | |
247 | |
248 /* | |
249 * Set up the perl environment, push arguments onto the | |
250 * perl stack, then call the given function | |
251 */ | |
252 dSP; | |
253 ENTER; | |
254 SAVETMPS; | |
255 PUSHMARK(sp); | |
256 | |
257 for (i = 0; i < argc; i++) { | |
258 if (args[i]) { | |
259 sv_args[i] = sv_2mortal(newSVpv(args[i], 0)); | |
260 XPUSHs(sv_args[i]); | |
261 } | |
262 } | |
263 | |
264 PUTBACK; | |
265 count = call_pv(function, G_EVAL | G_SCALAR); | |
266 SPAGAIN; | |
267 | |
268 /* | |
269 * Check for "die," make sure we have 1 argument, and set our | |
270 * return value. | |
271 */ | |
272 if (SvTRUE(ERRSV)) { | |
5227
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
273 gaim_debug(GAIM_DEBUG_ERROR, "perl", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
274 "Perl function %s exited abnormally: %s\n", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
275 function, SvPV(ERRSV, na)); |
5205 | 276 POPs; |
277 } | |
278 else if (count != 1) { | |
279 /* | |
280 * This should NEVER happen. G_SCALAR ensures that we WILL | |
281 * have 1 parameter. | |
282 */ | |
5227
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
283 gaim_debug(GAIM_DEBUG_ERROR, "perl", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
284 "Perl error from %s: expected 1 return value, " |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
285 "but got %d\n", function, count); |
5205 | 286 } |
287 else | |
288 ret_value = POPi; | |
289 | |
290 /* Check for changed arguments */ | |
291 for (i = 0; i < argc; i++) { | |
292 if (args[i] && strcmp(args[i], SvPVX(sv_args[i]))) { | |
293 /* | |
294 * Shizzel. So the perl script changed one of the parameters, | |
295 * and we want this change to affect the original parameters. | |
296 * args[i] is just a tempory little list of pointers. We don't | |
297 * want to free args[i] here because the new parameter doesn't | |
298 * overwrite the data that args[i] points to. That is done by | |
299 * the function that called execute_perl. I'm not explaining this | |
300 * very well. See, it's aggregate... Oh, but if 2 perl scripts | |
301 * both modify the data, _that's_ a memleak. This is really kind | |
302 * of hackish. I should fix it. Look how long this comment is. | |
303 * Holy crap. | |
304 */ | |
305 args[i] = g_strdup(SvPV(sv_args[i], na)); | |
306 } | |
307 } | |
308 | |
309 PUTBACK; | |
310 FREETMPS; | |
311 LEAVE; | |
312 | |
313 return ret_value; | |
314 } | |
315 | |
316 static void | |
317 perl_unload_file(GaimPlugin *plug) | |
318 { | |
319 char *atmp[2] = { "", NULL }; | |
320 struct perlscript *scp = NULL; | |
321 struct _perl_timeout_handlers *thn; | |
322 struct _perl_event_handlers *ehn; | |
323 GList *pl; | |
324 | |
325 for (pl = perl_list; pl != NULL; pl = pl->next) { | |
326 scp = pl->data; | |
327 | |
328 if (scp->plug == plug) { | |
329 perl_list = g_list_remove(perl_list, scp); | |
330 | |
331 if (scp->shutdowncallback[0]) | |
332 execute_perl(scp->shutdowncallback, 1, atmp); | |
333 | |
334 g_free(scp->name); | |
335 g_free(scp->version); | |
336 g_free(scp->shutdowncallback); | |
337 g_free(scp); | |
338 | |
339 break; | |
340 } | |
341 } | |
342 | |
343 for (pl = perl_timeout_handlers; pl != NULL; pl = pl->next) { | |
344 thn = pl->data; | |
345 | |
346 if (thn && thn->plug == plug) { | |
347 perl_timeout_handlers = g_list_remove(perl_timeout_handlers, thn); | |
348 | |
349 g_source_remove(thn->iotag); | |
350 g_free(thn->handler_args); | |
351 g_free(thn->handler_name); | |
352 g_free(thn); | |
353 } | |
354 } | |
355 | |
356 for (pl = perl_event_handlers; pl != NULL; pl = pl->next) { | |
357 ehn = pl->data; | |
358 | |
359 if (ehn && ehn->plug == plug) { | |
360 perl_event_handlers = g_list_remove(perl_event_handlers, ehn); | |
361 | |
362 g_free(ehn->event_type); | |
363 g_free(ehn->handler_name); | |
364 g_free(ehn); | |
365 } | |
366 } | |
367 } | |
368 | |
369 static int | |
370 perl_load_file(char *script_name, GaimPlugin *plugin) | |
371 { | |
372 char *atmp[2] = { script_name, NULL }; | |
373 GList *s; | |
374 struct perlscript *scp; | |
375 int ret; | |
376 | |
377 if (my_perl == NULL) | |
378 perl_init(); | |
379 | |
380 plugin->handle = plugin->path; | |
381 | |
382 ret = execute_perl("load_n_eval", 1, atmp); | |
383 | |
384 for (s = perl_list; s != NULL; s = s->next) { | |
385 scp = s->data; | |
386 | |
387 if (!strcmp(scp->name, plugin->info->name) && | |
388 !strcmp(scp->version, plugin->info->version)) { | |
389 | |
390 break; | |
391 } | |
392 } | |
393 | |
394 if (!s) { | |
395 plugin->error = g_strdup(_("GAIM::register not called with " | |
396 "proper arguments. Consult PERL-HOWTO.")); | |
397 | |
398 return 0; | |
399 } | |
400 | |
401 return ret; | |
402 } | |
403 | |
404 static void | |
405 perl_init(void) | |
406 { | |
407 /* changed the name of the variable from load_file to | |
408 perl_definitions since now it does much more than defining | |
409 the load_file sub. Moreover, deplaced the initialisation to | |
410 the xs_init function. (TheHobbit)*/ | |
411 char *perl_args[] = { "", "-e", "0", "-w" }; | |
412 char perl_definitions[] = | |
413 { | |
414 /* We use to function one to load a file the other to | |
415 execute the string obtained from the first and holding | |
416 the file conents. This allows to have a realy local $/ | |
417 without introducing temp variables to hold the old | |
418 value. Just a question of style:) */ | |
419 "sub load_file{" | |
420 "my $f_name=shift;" | |
421 "local $/=undef;" | |
422 "open FH,$f_name or return \"__FAILED__\";" | |
423 "$_=<FH>;" | |
424 "close FH;" | |
425 "return $_;" | |
426 "}" | |
427 "sub load_n_eval{" | |
428 "my $f_name=shift;" | |
429 "my $strin=load_file($f_name);" | |
430 "return 2 if($strin eq \"__FAILED__\");" | |
431 "eval $strin;" | |
432 "if($@){" | |
433 /*" #something went wrong\n"*/ | |
434 "GAIM::print(\"Errors loading file $f_name:\\n\",\"$@\");" | |
435 "return 1;" | |
436 "}" | |
437 "return 0;" | |
438 "}" | |
439 }; | |
440 | |
441 my_perl = perl_alloc(); | |
442 perl_construct(my_perl); | |
443 #ifdef DEBUG | |
444 perl_parse(my_perl, xs_init, 4, perl_args, NULL); | |
445 #else | |
446 perl_parse(my_perl, xs_init, 3, perl_args, NULL); | |
447 #endif | |
448 #ifdef HAVE_PERL_EVAL_PV | |
449 eval_pv(perl_definitions, TRUE); | |
450 #else | |
451 perl_eval_pv(perl_definitions, TRUE); /* deprecated */ | |
452 #endif | |
453 } | |
454 | |
455 static void | |
456 perl_end(void) | |
457 { | |
458 char *atmp[2] = { "", NULL }; | |
459 struct perlscript *scp; | |
460 struct _perl_timeout_handlers *thn; | |
461 struct _perl_event_handlers *ehn; | |
462 | |
463 while (perl_list) { | |
464 scp = perl_list->data; | |
465 perl_list = g_list_remove(perl_list, scp); | |
466 | |
467 if (scp->shutdowncallback[0]) | |
468 execute_perl(scp->shutdowncallback, 1, atmp); | |
469 | |
470 g_free(scp->name); | |
471 g_free(scp->version); | |
472 g_free(scp->shutdowncallback); | |
473 g_free(scp); | |
474 } | |
475 | |
476 while (perl_timeout_handlers) { | |
477 thn = perl_timeout_handlers->data; | |
478 perl_timeout_handlers = g_list_remove(perl_timeout_handlers, thn); | |
479 g_source_remove(thn->iotag); | |
480 g_free(thn->handler_args); | |
481 g_free(thn->handler_name); | |
482 g_free(thn); | |
483 } | |
484 | |
485 while (perl_event_handlers) { | |
486 ehn = perl_event_handlers->data; | |
487 perl_event_handlers = g_list_remove(perl_event_handlers, ehn); | |
488 g_free(ehn->event_type); | |
489 g_free(ehn->handler_name); | |
490 g_free(ehn); | |
491 } | |
492 | |
493 if (my_perl != NULL) { | |
494 perl_destruct(my_perl); | |
495 perl_free(my_perl); | |
496 my_perl = NULL; | |
497 } | |
498 } | |
499 | |
500 XS (XS_GAIM_register) | |
501 { | |
502 char *name, *ver, *callback, *unused; /* exactly like X-Chat, eh? :) */ | |
503 unsigned int junk; | |
504 struct perlscript *scp; | |
505 GaimPlugin *plug = NULL; | |
506 GList *pl; | |
507 | |
508 dXSARGS; | |
509 items = 0; | |
510 | |
511 name = SvPV(ST(0), junk); | |
512 ver = SvPV(ST(1), junk); | |
513 callback = SvPV(ST(2), junk); | |
514 unused = SvPV(ST(3), junk); | |
515 | |
5227
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
516 gaim_debug(GAIM_DEBUG_INFO, "perl", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
517 "GAIM::register(%s, %s)\n", name, ver); |
5205 | 518 |
519 for (pl = gaim_plugins_get_all(); pl != NULL; pl = pl->next) { | |
520 plug = pl->data; | |
521 | |
522 if (!strcmp(name, plug->info->name) && | |
523 !strcmp(ver, plug->info->version)) { | |
524 | |
525 break; | |
526 } | |
527 | |
528 plug = NULL; | |
529 } | |
530 | |
531 if (plug) { | |
532 scp = g_new0(struct perlscript, 1); | |
533 scp->name = g_strdup(name); | |
534 scp->version = g_strdup(ver); | |
535 scp->shutdowncallback = g_strdup(callback); | |
536 scp->plug = plug; | |
537 | |
538 perl_list = g_list_append(perl_list, scp); | |
539 | |
540 XST_mPV(0, plug->path); | |
541 } | |
542 else | |
543 XST_mPV(0, NULL); | |
544 | |
545 XSRETURN (1); | |
546 } | |
547 | |
548 XS (XS_GAIM_get_info) | |
549 { | |
550 int i = 0; | |
551 dXSARGS; | |
552 items = 0; | |
553 | |
554 switch(SvIV(ST(0))) { | |
555 case 0: | |
556 XST_mPV(0, VERSION); | |
557 i = 1; | |
558 break; | |
559 | |
560 case 1: | |
561 { | |
562 GSList *c = connections; | |
563 struct gaim_connection *gc; | |
564 | |
565 while (c) { | |
566 gc = (struct gaim_connection *)c->data; | |
567 XST_mIV(i++, (guint)gc); | |
568 c = c->next; | |
569 } | |
570 } | |
571 break; | |
572 | |
573 case 2: | |
574 { | |
575 struct gaim_connection *gc = | |
576 (struct gaim_connection *)SvIV(ST(1)); | |
577 | |
578 if (g_slist_find(connections, gc)) | |
579 XST_mIV(i++, gc->protocol); | |
580 else | |
581 XST_mIV(i++, -1); | |
582 } | |
583 break; | |
584 | |
585 case 3: | |
586 { | |
587 struct gaim_connection *gc = | |
588 (struct gaim_connection *)SvIV(ST(1)); | |
589 | |
590 if (g_slist_find(connections, gc)) | |
591 XST_mPV(i++, gc->username); | |
592 else | |
593 XST_mPV(i++, ""); | |
594 } | |
595 break; | |
596 | |
597 case 4: | |
598 { | |
599 struct gaim_connection *gc = | |
600 (struct gaim_connection *)SvIV(ST(1)); | |
601 | |
602 if (g_slist_find(connections, gc)) | |
603 XST_mIV(i++, g_slist_index(gaim_accounts, gc->account)); | |
604 else | |
605 XST_mIV(i++, -1); | |
606 } | |
607 break; | |
608 | |
609 case 5: | |
610 { | |
611 GSList *a = gaim_accounts; | |
612 while (a) { | |
613 struct gaim_account *account = a->data; | |
614 XST_mPV(i++, account->username); | |
615 a = a->next; | |
616 } | |
617 } | |
618 break; | |
619 | |
620 case 6: | |
621 { | |
622 GSList *a = gaim_accounts; | |
623 while (a) { | |
624 struct gaim_account *account = a->data; | |
625 XST_mIV(i++, account->protocol); | |
626 a = a->next; | |
627 } | |
628 } | |
629 break; | |
630 | |
631 case 7: | |
632 { | |
633 struct gaim_connection *gc = | |
634 (struct gaim_connection *)SvIV(ST(1)); | |
635 | |
636 if (g_slist_find(connections, gc)) | |
637 XST_mPV(i++, gc->prpl->info->name); | |
638 else | |
639 XST_mPV(i++, "Unknown"); | |
640 } | |
641 break; | |
642 | |
643 default: | |
644 XST_mPV(0, "Error2"); | |
645 i = 1; | |
646 } | |
647 | |
648 XSRETURN(i); | |
649 } | |
650 | |
651 XS (XS_GAIM_print) | |
652 { | |
653 char *title; | |
654 char *message; | |
655 unsigned int junk; | |
656 dXSARGS; | |
657 items = 0; | |
658 | |
659 title = SvPV(ST(0), junk); | |
660 message = SvPV(ST(1), junk); | |
661 do_error_dialog(title, message, GAIM_INFO); | |
662 XSRETURN(0); | |
663 } | |
664 | |
665 XS (XS_GAIM_buddy_list) | |
666 { | |
667 struct gaim_connection *gc; | |
668 struct buddy *buddy; | |
669 struct group *g; | |
670 GaimBlistNode *gnode,*bnode; | |
671 int i = 0; | |
672 dXSARGS; | |
673 items = 0; | |
674 | |
675 gc = (struct gaim_connection *)SvIV(ST(0)); | |
676 | |
677 for(gnode = gaim_get_blist()->root; gnode; gnode = gnode->next) { | |
678 if(!GAIM_BLIST_NODE_IS_GROUP(gnode)) | |
679 continue; | |
680 g = (struct group *)gnode; | |
681 for(bnode = gnode->child; bnode; bnode = bnode->next) { | |
682 if(!GAIM_BLIST_NODE_IS_BUDDY(bnode)) | |
683 continue; | |
684 buddy = (struct buddy *)bnode; | |
685 if(buddy->account == gc->account) | |
686 XST_mPV(i++, buddy->name); | |
687 } | |
688 } | |
689 XSRETURN(i); | |
690 } | |
691 | |
692 XS (XS_GAIM_online_list) | |
693 { | |
694 struct gaim_connection *gc; | |
695 struct buddy *b; | |
696 struct group *g; | |
697 GaimBlistNode *gnode,*bnode; | |
698 int i = 0; | |
699 dXSARGS; | |
700 items = 0; | |
701 | |
702 gc = (struct gaim_connection *)SvIV(ST(0)); | |
703 | |
704 for(gnode = gaim_get_blist()->root; gnode; gnode = gnode->next) { | |
705 if(!GAIM_BLIST_NODE_IS_GROUP(gnode)) | |
706 continue; | |
707 g = (struct group *)gnode; | |
708 for(bnode = gnode->child; bnode; bnode = bnode->next) { | |
709 if(!GAIM_BLIST_NODE_IS_BUDDY(bnode)) | |
710 continue; | |
711 b = (struct buddy *)bnode; | |
712 if (b->account == gc->account && GAIM_BUDDY_IS_ONLINE(b)) XST_mPV(i++, b->name); | |
713 } | |
714 } | |
715 XSRETURN(i); | |
716 } | |
717 | |
718 XS (XS_GAIM_command) | |
719 { | |
720 unsigned int junk; | |
721 char *command = NULL; | |
722 dXSARGS; | |
723 items = 0; | |
724 | |
725 command = SvPV(ST(0), junk); | |
726 if (!command) XSRETURN(0); | |
727 if (!strncasecmp(command, "signon", 6)) { | |
728 int index = SvIV(ST(1)); | |
729 if (g_slist_nth_data(gaim_accounts, index)) | |
730 serv_login(g_slist_nth_data(gaim_accounts, index)); | |
731 } else if (!strncasecmp(command, "signoff", 7)) { | |
732 struct gaim_connection *gc = (struct gaim_connection *)SvIV(ST(1)); | |
733 if (g_slist_find(connections, gc)) signoff(gc); | |
734 else signoff_all(NULL, NULL); | |
735 } else if (!strncasecmp(command, "info", 4)) { | |
736 struct gaim_connection *gc = (struct gaim_connection *)SvIV(ST(1)); | |
737 if (g_slist_find(connections, gc)) | |
738 serv_set_info(gc, SvPV(ST(2), junk)); | |
739 } else if (!strncasecmp(command, "away", 4)) { | |
740 char *message = SvPV(ST(1), junk); | |
741 static struct away_message a; | |
742 g_snprintf(a.message, sizeof(a.message), "%s", message); | |
743 do_away_message(NULL, &a); | |
744 } else if (!strncasecmp(command, "back", 4)) { | |
745 do_im_back(); | |
746 } else if (!strncasecmp(command, "idle", 4)) { | |
747 GSList *c = connections; | |
748 struct gaim_connection *gc; | |
749 | |
750 while (c) { | |
751 gc = (struct gaim_connection *)c->data; | |
752 serv_set_idle(gc, SvIV(ST(1))); | |
753 c = c->next; | |
754 } | |
755 } else if (!strncasecmp(command, "warn", 4)) { | |
756 GSList *c = connections; | |
757 struct gaim_connection *gc; | |
758 | |
759 while (c) { | |
760 gc = (struct gaim_connection *)c->data; | |
761 serv_warn(gc, SvPV(ST(1), junk), SvIV(ST(2))); | |
762 c = c->next; | |
763 } | |
764 } | |
765 | |
766 XSRETURN(0); | |
767 } | |
768 | |
769 XS (XS_GAIM_user_info) | |
770 { | |
771 struct gaim_connection *gc; | |
772 unsigned int junk; | |
773 struct buddy *buddy = NULL; | |
774 dXSARGS; | |
775 items = 0; | |
776 | |
777 gc = (struct gaim_connection *)SvIV(ST(0)); | |
778 if (g_slist_find(connections, gc)) | |
779 buddy = gaim_find_buddy(gc->account, SvPV(ST(1), junk)); | |
780 | |
781 if (!buddy) | |
782 XSRETURN(0); | |
783 XST_mPV(0, buddy->name); | |
784 XST_mPV(1, gaim_get_buddy_alias(buddy)); | |
785 XST_mPV(2, GAIM_BUDDY_IS_ONLINE(buddy) ? "Online" : "Offline"); | |
786 XST_mIV(3, buddy->evil); | |
787 XST_mIV(4, buddy->signon); | |
788 XST_mIV(5, buddy->idle); | |
789 XSRETURN(6); | |
790 } | |
791 | |
792 XS (XS_GAIM_write_to_conv) | |
793 { | |
794 char *nick, *who, *what; | |
795 struct gaim_conversation *c; | |
796 int junk; | |
797 int send, wflags; | |
798 dXSARGS; | |
799 items = 0; | |
800 | |
801 nick = SvPV(ST(0), junk); | |
802 send = SvIV(ST(1)); | |
803 what = SvPV(ST(2), junk); | |
804 who = SvPV(ST(3), junk); | |
805 | |
806 if (!*who) who=NULL; | |
807 | |
808 switch (send) { | |
809 case 0: wflags=WFLAG_SEND; break; | |
810 case 1: wflags=WFLAG_RECV; break; | |
811 case 2: wflags=WFLAG_SYSTEM; break; | |
812 default: wflags=WFLAG_RECV; | |
813 } | |
814 | |
815 c = gaim_find_conversation(nick); | |
816 | |
817 if (!c) | |
818 c = gaim_conversation_new(GAIM_CONV_IM, NULL, nick); | |
819 | |
820 gaim_conversation_write(c, who, what, -1, wflags, time(NULL)); | |
821 XSRETURN(0); | |
822 } | |
823 | |
824 XS (XS_GAIM_serv_send_im) | |
825 { | |
826 struct gaim_connection *gc; | |
827 char *nick, *what; | |
828 int isauto; | |
829 int junk; | |
830 dXSARGS; | |
831 items = 0; | |
832 | |
833 gc = (struct gaim_connection *)SvIV(ST(0)); | |
834 nick = SvPV(ST(1), junk); | |
835 what = SvPV(ST(2), junk); | |
836 isauto = SvIV(ST(3)); | |
837 | |
838 if (!g_slist_find(connections, gc)) { | |
839 XSRETURN(0); | |
840 return; | |
841 } | |
842 serv_send_im(gc, nick, what, -1, isauto); | |
843 XSRETURN(0); | |
844 } | |
845 | |
846 XS (XS_GAIM_print_to_conv) | |
847 { | |
848 struct gaim_connection *gc; | |
849 char *nick, *what; | |
850 int isauto; | |
851 struct gaim_conversation *c; | |
852 unsigned int junk; | |
853 dXSARGS; | |
854 items = 0; | |
855 | |
856 gc = (struct gaim_connection *)SvIV(ST(0)); | |
857 nick = SvPV(ST(1), junk); | |
858 what = SvPV(ST(2), junk); | |
859 isauto = SvIV(ST(3)); | |
860 if (!g_slist_find(connections, gc)) { | |
861 XSRETURN(0); | |
862 return; | |
863 } | |
864 | |
865 c = gaim_find_conversation(nick); | |
866 | |
867 if (!c) | |
868 c = gaim_conversation_new(GAIM_CONV_IM, gc->account, nick); | |
869 else | |
870 gaim_conversation_set_account(c, gc->account); | |
871 | |
872 gaim_conversation_write(c, NULL, what, -1, | |
873 (WFLAG_SEND | (isauto ? WFLAG_AUTO : 0)), time(NULL)); | |
874 serv_send_im(gc, nick, what, -1, isauto ? IM_FLAG_AWAY : 0); | |
875 XSRETURN(0); | |
876 } | |
877 | |
878 | |
879 | |
880 XS (XS_GAIM_print_to_chat) | |
881 { | |
882 struct gaim_connection *gc; | |
883 int id; | |
884 char *what; | |
885 struct gaim_conversation *b = NULL; | |
886 GSList *bcs; | |
887 unsigned int junk; | |
888 dXSARGS; | |
889 items = 0; | |
890 | |
891 gc = (struct gaim_connection *)SvIV(ST(0)); | |
892 id = SvIV(ST(1)); | |
893 what = SvPV(ST(2), junk); | |
894 | |
895 if (!g_slist_find(connections, gc)) { | |
896 XSRETURN(0); | |
897 return; | |
898 } | |
899 bcs = gc->buddy_chats; | |
900 while (bcs) { | |
901 b = (struct gaim_conversation *)bcs->data; | |
902 | |
903 if (gaim_chat_get_id(gaim_conversation_get_chat_data(b)) == id) | |
904 break; | |
905 bcs = bcs->next; | |
906 b = NULL; | |
907 } | |
908 if (b) | |
909 serv_chat_send(gc, id, what); | |
910 XSRETURN(0); | |
911 } | |
912 | |
913 static int | |
914 perl_event(GaimEvent event, void *unused, va_list args) | |
915 { | |
916 char *buf[5] = { NULL, NULL, NULL, NULL, NULL }; /* Maximum of 5 args */ | |
917 void *arg1 = NULL, *arg2 = NULL, *arg3 = NULL, *arg4 = NULL, *arg5 = NULL; | |
918 char tmpbuf1[16], tmpbuf2[16], tmpbuf3[1]; | |
919 GList *handler; | |
920 struct _perl_event_handlers *data; | |
921 int handler_return; | |
922 | |
923 arg1 = va_arg(args, void *); | |
924 arg2 = va_arg(args, void *); | |
925 arg3 = va_arg(args, void *); | |
926 arg4 = va_arg(args, void *); | |
927 arg5 = va_arg(args, void *); | |
928 | |
929 tmpbuf1[0] = '\0'; | |
930 tmpbuf2[0] = '\0'; | |
931 tmpbuf3[0] = '\0'; | |
932 | |
933 /* Make a pretty array of char*'s with which to call perl functions */ | |
934 switch (event) { | |
935 case event_signon: | |
936 case event_signoff: | |
937 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
938 buf[0] = tmpbuf1; | |
939 break; | |
940 case event_away: | |
941 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
942 buf[0] = tmpbuf1; | |
943 buf[1] = ((struct gaim_connection *)arg1)->away ? | |
944 ((struct gaim_connection *)arg1)->away : tmpbuf2; | |
945 break; | |
946 case event_im_recv: | |
947 if (!*(char**)arg2 || !*(char**)arg3) return 1; | |
948 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
949 buf[0] = tmpbuf1; | |
950 buf[1] = *(char **)arg2; | |
951 buf[2] = *(char **)arg3; | |
952 break; | |
953 case event_im_send: | |
954 if (!*(char**)arg3) return 1; | |
955 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
956 buf[0] = tmpbuf1; | |
957 buf[1] = arg2 ? arg2 : tmpbuf3; | |
958 buf[2] = *(char **)arg3; | |
959 break; | |
960 case event_buddy_signon: | |
961 case event_buddy_signoff: | |
962 case event_set_info: | |
963 case event_buddy_away: | |
964 case event_buddy_back: | |
965 case event_buddy_idle: | |
966 case event_buddy_unidle: | |
967 case event_got_typing: | |
968 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
969 buf[0] = tmpbuf1; | |
970 buf[1] = arg2; | |
971 break; | |
972 case event_chat_invited: | |
973 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
974 buf[0] = tmpbuf1; | |
975 buf[1] = arg2; | |
976 buf[2] = arg3; | |
977 buf[3] = arg4; | |
978 break; | |
979 case event_chat_join: | |
980 case event_chat_buddy_join: | |
981 case event_chat_buddy_leave: | |
982 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
983 buf[0] = tmpbuf1; | |
984 g_snprintf(tmpbuf2, 16, "%d", (int)arg2); | |
985 buf[1] = tmpbuf2; | |
986 buf[2] = arg3; | |
987 break; | |
988 case event_chat_leave: | |
989 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
990 buf[0] = tmpbuf1; | |
991 g_snprintf(tmpbuf2, 16, "%d", (int)arg2); | |
992 buf[1] = tmpbuf2; | |
993 break; | |
994 case event_chat_recv: | |
995 if (!*(char**)arg3 || !*(char**)arg4) return 1; | |
996 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
997 buf[0] = tmpbuf1; | |
998 g_snprintf(tmpbuf2, 16, "%d", (int)arg2); | |
999 buf[1] = tmpbuf2; | |
1000 buf[2] = *(char **)arg3; | |
1001 buf[3] = *(char **)arg4; | |
1002 break; | |
1003 case event_chat_send_invite: | |
1004 if (!*(char**)arg4) return 1; | |
1005 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
1006 buf[0] = tmpbuf1; | |
1007 g_snprintf(tmpbuf2, 16, "%d", (int)arg2); | |
1008 buf[1] = tmpbuf2; | |
1009 buf[2] = arg3; | |
1010 buf[3] = *(char **)arg4; | |
1011 break; | |
1012 case event_chat_send: | |
1013 if (!*(char**)arg3) return 1; | |
1014 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
1015 buf[0] = tmpbuf1; | |
1016 g_snprintf(tmpbuf2, 16, "%d", (int)arg2); | |
1017 buf[1] = tmpbuf2; | |
1018 buf[2] = *(char **)arg3; | |
1019 break; | |
1020 case event_warned: | |
1021 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
1022 buf[0] = tmpbuf1; | |
1023 buf[1] = arg2 ? arg2 : tmpbuf3; | |
1024 g_snprintf(tmpbuf2, 16, "%d", (int)arg3); | |
1025 buf[2] = tmpbuf2; | |
1026 break; | |
1027 case event_quit: | |
1028 case event_blist_update: | |
1029 buf[0] = tmpbuf3; | |
1030 break; | |
1031 case event_new_conversation: | |
1032 case event_del_conversation: | |
1033 buf[0] = arg1; | |
1034 break; | |
1035 case event_im_displayed_sent: | |
1036 if (!*(char**)arg3) return 1; | |
1037 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
1038 buf[0] = tmpbuf1; | |
1039 buf[1] = arg2; | |
1040 buf[2] = *(char **)arg3; | |
1041 break; | |
1042 case event_im_displayed_rcvd: | |
1043 g_snprintf(tmpbuf1, 16, "%lu", (unsigned long)arg1); | |
1044 buf[0] = tmpbuf1; | |
1045 buf[1] = arg2; | |
1046 buf[2] = arg3 ? arg3 : tmpbuf3; | |
1047 break; | |
1048 case event_draw_menu: | |
1049 /* we can't handle this usefully without gtk/perl bindings */ | |
1050 return 0; | |
1051 default: | |
5227
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1052 gaim_debug(GAIM_DEBUG_WARNING, "perl", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1053 "Someone forgot to handle %s in the perl binding\n", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1054 gaim_event_get_name(event)); |
5205 | 1055 return 0; |
1056 } | |
1057 | |
1058 /* Call any applicable functions */ | |
1059 for (handler = perl_event_handlers; | |
1060 handler != NULL; | |
1061 handler = handler->next) { | |
1062 | |
1063 data = handler->data; | |
1064 | |
1065 if (!strcmp(gaim_event_get_name(event), data->event_type)) { | |
1066 | |
1067 handler_return = execute_perl(data->handler_name, 5, buf); | |
1068 | |
1069 if (handler_return) | |
1070 return handler_return; | |
1071 } | |
1072 } | |
1073 | |
1074 /* Now make changes from perl scripts affect the real data */ | |
1075 switch (event) { | |
1076 case event_im_recv: | |
1077 if (buf[1] != *(char **)arg2) { | |
1078 free(*(char **)arg2); | |
1079 *(char **)arg2 = buf[1]; | |
1080 } | |
1081 if (buf[2] != *(char **)arg3) { | |
1082 free(*(char **)arg3); | |
1083 *(char **)arg3 = buf[2]; | |
1084 } | |
1085 break; | |
1086 case event_im_send: | |
1087 if (buf[2] != *(char **)arg3) { | |
1088 free(*(char **)arg3); | |
1089 *(char **)arg3 = buf[2]; | |
1090 } | |
1091 break; | |
1092 case event_chat_recv: | |
1093 if (buf[2] != *(char **)arg3) { | |
1094 free(*(char **)arg3); | |
1095 *(char **)arg3 = buf[2]; | |
1096 } | |
1097 if (buf[3] != *(char **)arg4) { | |
1098 free(*(char **)arg4); | |
1099 *(char **)arg4 = buf[3]; | |
1100 } | |
1101 break; | |
1102 case event_chat_send_invite: | |
1103 if (buf[3] != *(char **)arg4) { | |
1104 free(*(char **)arg4); | |
1105 *(char **)arg4 = buf[3]; | |
1106 } | |
1107 break; | |
1108 case event_chat_send: | |
1109 if (buf[2] != *(char **)arg3) { | |
1110 free(*(char **)arg3); | |
1111 *(char **)arg3 = buf[2]; | |
1112 } | |
1113 break; | |
1114 case event_im_displayed_sent: | |
1115 if (buf[2] != *(char **)arg3) { | |
1116 free(*(char **)arg3); | |
1117 *(char **)arg3 = buf[2]; | |
1118 } | |
1119 break; | |
1120 default: | |
1121 break; | |
1122 } | |
1123 | |
1124 return 0; | |
1125 } | |
1126 | |
1127 XS (XS_GAIM_add_event_handler) | |
1128 { | |
1129 unsigned int junk; | |
1130 struct _perl_event_handlers *handler; | |
1131 char *handle; | |
1132 GaimPlugin *plug; | |
1133 GList *p; | |
1134 dXSARGS; | |
1135 items = 0; | |
1136 | |
1137 handle = SvPV(ST(0), junk); | |
1138 | |
1139 for (p = gaim_plugins_get_all(); p != NULL; p = p->next) { | |
1140 plug = p->data; | |
1141 | |
1142 if (!strcmp(handle, plug->path)) | |
1143 break; | |
1144 } | |
1145 | |
1146 if (p) { | |
1147 handler = g_new0(struct _perl_event_handlers, 1); | |
1148 handler->event_type = g_strdup(SvPV(ST(1), junk)); | |
1149 handler->handler_name = g_strdup(SvPV(ST(2), junk)); | |
1150 handler->plug = plug; | |
1151 perl_event_handlers = g_list_append(perl_event_handlers, handler); | |
5227
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1152 gaim_debug(GAIM_DEBUG_INFO, "perl", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1153 "Registered perl event handler for %s\n", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1154 handler->event_type); |
5205 | 1155 } else { |
5227
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1156 gaim_debug(GAIM_DEBUG_ERROR, "perl", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1157 "Invalid handle (%s) registering perl event handler\n", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1158 handle); |
5205 | 1159 } |
1160 | |
1161 XSRETURN_EMPTY; | |
1162 } | |
1163 | |
1164 XS (XS_GAIM_remove_event_handler) | |
1165 { | |
1166 unsigned int junk; | |
1167 struct _perl_event_handlers *ehn; | |
1168 GList *cur = perl_event_handlers; | |
1169 dXSARGS; | |
1170 items = 0; | |
1171 | |
1172 while (cur) { | |
1173 GList *next = cur->next; | |
1174 ehn = cur->data; | |
1175 | |
1176 if (!strcmp(ehn->event_type, SvPV(ST(0), junk)) && | |
1177 !strcmp(ehn->handler_name, SvPV(ST(1), junk))) | |
1178 { | |
1179 perl_event_handlers = g_list_remove(perl_event_handlers, ehn); | |
1180 g_free(ehn->event_type); | |
1181 g_free(ehn->handler_name); | |
1182 g_free(ehn); | |
1183 } | |
1184 | |
1185 cur = next; | |
1186 } | |
1187 } | |
1188 | |
1189 static int | |
1190 perl_timeout(gpointer data) | |
1191 { | |
1192 char *atmp[2] = { NULL, NULL }; | |
1193 struct _perl_timeout_handlers *handler = data; | |
1194 | |
1195 atmp[0] = escape_quotes(handler->handler_args); | |
1196 execute_perl(handler->handler_name, 1, atmp); | |
1197 | |
1198 perl_timeout_handlers = g_list_remove(perl_timeout_handlers, handler); | |
1199 g_free(handler->handler_args); | |
1200 g_free(handler->handler_name); | |
1201 g_free(handler); | |
1202 | |
1203 return 0; /* returning zero removes the timeout handler */ | |
1204 } | |
1205 | |
1206 XS (XS_GAIM_add_timeout_handler) | |
1207 { | |
1208 unsigned int junk; | |
1209 long timeout; | |
1210 struct _perl_timeout_handlers *handler; | |
1211 char *handle; | |
1212 GaimPlugin *plug; | |
1213 GList *p; | |
1214 | |
1215 dXSARGS; | |
1216 items = 0; | |
1217 | |
1218 handle = SvPV(ST(0), junk); | |
1219 | |
1220 for (p = gaim_plugins_get_all(); p != NULL; p = p->next) { | |
1221 plug = p->data; | |
1222 | |
1223 if (!strcmp(handle, plug->path)) | |
1224 break; | |
1225 } | |
1226 | |
1227 if (p) { | |
1228 handler = g_new0(struct _perl_timeout_handlers, 1); | |
1229 timeout = 1000 * SvIV(ST(1)); | |
5227
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1230 gaim_debug(GAIM_DEBUG_INFO, "perl", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1231 "Adding timeout for %ld seconds.\n", timeout/1000); |
5205 | 1232 handler->plug = plug; |
1233 handler->handler_name = g_strdup(SvPV(ST(2), junk)); | |
1234 handler->handler_args = g_strdup(SvPV(ST(3), junk)); | |
1235 perl_timeout_handlers = g_list_append(perl_timeout_handlers, handler); | |
1236 handler->iotag = g_timeout_add(timeout, perl_timeout, handler); | |
1237 } else { | |
5227
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1238 gaim_debug(GAIM_DEBUG_ERROR, "perl", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1239 "Invalid handle (%s) in adding perl timeout handler.", |
6d1707dc8c3d
[gaim-migrate @ 5597]
Christian Hammond <chipx86@chipx86.com>
parents:
5205
diff
changeset
|
1240 handle); |
5205 | 1241 } |
1242 XSRETURN_EMPTY; | |
1243 } | |
1244 | |
1245 XS (XS_GAIM_play_sound) | |
1246 { | |
1247 int id; | |
1248 dXSARGS; | |
1249 | |
1250 items = 0; | |
1251 | |
1252 id = SvIV(ST(0)); | |
1253 | |
1254 gaim_sound_play_event(id); | |
1255 | |
1256 XSRETURN_EMPTY; | |
1257 } | |
1258 | |
1259 static gboolean | |
1260 probe_perl_plugin(GaimPlugin *plugin) | |
1261 { | |
1262 /* XXX This would be much faster if I didn't create a new | |
1263 * PerlInterpreter every time I probed a plugin */ | |
1264 | |
1265 GaimPluginInfo *info; | |
1266 PerlInterpreter *prober = perl_alloc(); | |
1267 char *argv[] = {"", plugin->path }; | |
1268 int count; | |
1269 gboolean status = TRUE; | |
1270 | |
1271 perl_construct(prober); | |
1272 perl_parse(prober, NULL, 2, argv, NULL); | |
1273 | |
1274 { | |
1275 dSP; | |
1276 ENTER; | |
1277 SAVETMPS; | |
1278 PUSHMARK(SP); | |
1279 | |
1280 count = perl_call_pv("description", G_NOARGS | G_ARRAY | G_EVAL); | |
1281 SPAGAIN; | |
1282 | |
1283 if (count == 6) { | |
1284 info = g_new0(GaimPluginInfo, 1); | |
1285 | |
1286 info->api_version = 2; | |
1287 info->type = GAIM_PLUGIN_STANDARD; | |
1288 | |
1289 info->dependencies = g_list_append(info->dependencies, | |
1290 PERL_PLUGIN_ID); | |
1291 | |
1292 POPp; /* iconfile */ | |
1293 | |
1294 info->homepage = g_strdup(POPp); | |
1295 info->author = g_strdup(POPp); | |
1296 info->description = g_strdup(POPp); | |
1297 info->version = g_strdup(POPp); | |
1298 info->name = g_strdup(POPp); | |
1299 | |
1300 plugin->info = info; | |
1301 | |
1302 if (!gaim_plugin_register(plugin)) | |
1303 status = FALSE; | |
1304 } | |
1305 else | |
1306 status = FALSE; | |
1307 | |
1308 PUTBACK; | |
1309 FREETMPS; | |
1310 LEAVE; | |
1311 } | |
1312 | |
1313 perl_destruct(prober); | |
1314 perl_free(prober); | |
1315 | |
1316 return status; | |
1317 } | |
1318 | |
1319 static gboolean | |
1320 load_perl_plugin(GaimPlugin *plugin) | |
1321 { | |
1322 perl_load_file(plugin->path, plugin); | |
1323 | |
1324 return TRUE; | |
1325 } | |
1326 | |
1327 static gboolean | |
1328 unload_perl_plugin(GaimPlugin *plugin) | |
1329 { | |
1330 perl_unload_file(plugin); | |
1331 | |
1332 return TRUE; | |
1333 } | |
1334 | |
1335 static void | |
1336 destroy_perl_plugin(GaimPlugin *plugin) | |
1337 { | |
1338 if (plugin->info != NULL) { | |
1339 g_free(plugin->info->name); | |
1340 g_free(plugin->info->version); | |
1341 g_free(plugin->info->description); | |
1342 g_free(plugin->info->author); | |
1343 g_free(plugin->info->homepage); | |
1344 } | |
1345 } | |
1346 | |
1347 static gboolean | |
1348 plugin_unload(GaimPlugin *plugin) | |
1349 { | |
1350 perl_end(); | |
1351 | |
1352 return TRUE; | |
1353 } | |
1354 | |
1355 static GaimPluginLoaderInfo loader_info = | |
1356 { | |
1357 NULL, /**< exts */ | |
1358 | |
1359 probe_perl_plugin, /**< probe */ | |
1360 load_perl_plugin, /**< load */ | |
1361 unload_perl_plugin, /**< unload */ | |
1362 destroy_perl_plugin, /**< destroy */ | |
1363 perl_event /**< broadcast */ | |
1364 }; | |
1365 | |
1366 static GaimPluginInfo info = | |
1367 { | |
1368 2, /**< api_version */ | |
1369 GAIM_PLUGIN_LOADER, /**< type */ | |
1370 NULL, /**< ui_requirement */ | |
1371 0, /**< flags */ | |
1372 NULL, /**< dependencies */ | |
1373 GAIM_PRIORITY_DEFAULT, /**< priority */ | |
1374 | |
1375 PERL_PLUGIN_ID, /**< id */ | |
1376 N_("Perl Plugin Loader"), /**< name */ | |
1377 VERSION, /**< version */ | |
1378 N_("Provides support for loading perl plugins."), /**< summary */ | |
1379 N_("Provides support for loading perl plugins."), /**< description */ | |
1380 "Christian Hammond <chipx86@gnupdate.org>", /**< author */ | |
1381 WEBSITE, /**< homepage */ | |
1382 | |
1383 NULL, /**< load */ | |
1384 plugin_unload, /**< unload */ | |
1385 NULL, /**< destroy */ | |
1386 | |
1387 NULL, /**< ui_info */ | |
1388 &loader_info /**< extra_info */ | |
1389 }; | |
1390 | |
1391 static void | |
1392 __init_plugin(GaimPlugin *plugin) | |
1393 { | |
1394 loader_info.exts = g_list_append(loader_info.exts, "pl"); | |
1395 } | |
1396 | |
1397 GAIM_INIT_PLUGIN(perl, __init_plugin, info); |