source: protocols/nogaim.h @ d28fe1c4

Last change on this file since d28fe1c4 was d28fe1c4, checked in by jgeboski <jgeboski@…>, at 2016-05-26T02:48:08Z

Implemented plugin information for external plugins

As of now, bitlbee will load any plugin regardless of the ABI it was
built against. This is really problematic when structures or symbols
are changed within bitlbee. This often leads to the plugin not loading
or the plugin acting in an undefined way. Typically a simple rebuild of
the plugin will resolve such issues, but many users have no idea that
this is required after they have updated bitlbee.

Furthermore, it is often times impossible to determine the version of
a plugin, without relying on the package manager of the system. This is
quite a problem when users are reporting bugs for external plugins, and
they have no idea what version of the plugin they are running. This is
also an opportunity to provide additional metadata for each plugin that
can then be displayed to the user.

Solving these issues is done by adding a new required function to each
plugin. The init_plugin_info() function must now be implemented along
with the init_plugin() function. This function then returns a static
structure, which retains all of the metadata for the plugin. Then this
is used by bitlbee to check the ABI version and provide information to
the user.

The introduction of the new function is required as bitlbee needs to
obtain the ABI version before calling init_plugin().

The boiler-plate implementation of init_plugin_info():

#ifdef BITLBEE_ABI_VERSION_CODE
struct plugin_info *init_plugin_info(void)
{

static struct plugin_info info = {

BITLBEE_ABI_VERSION_CODE, /* Required */
"plugin-name", /* Required */
"1.3.3.7", /* Required */
"A short description of the plugin", /* Optional */
"First Last <alias@…>", /* Optional */
"http://www.domain.tld" /* Optional */

};

return &info;

}
#endif

The example wraps the function declaration in an if block for backwards
compatibility with older bitlbee versions.

Displaying the plugin metadata is done via the newly added "plugins"
command, which simply dumps formatted data to the root channel.

  • Property mode set to 100644
File size: 14.5 KB
RevLine 
[5ebff60]1/********************************************************************\
[b7d3cc34]2  * BitlBee -- An IRC to other IM-networks gateway                     *
3  *                                                                    *
[0e788f5]4  * Copyright 2002-2012 Wilmer van der Gaast and others                *
[b7d3cc34]5  \********************************************************************/
6
7/*
[6bbb939]8 * nogaim, soon to be known as im_api. Not a separate product (unless
9 * someone would be interested in such a thing), just a new name.
[b7d3cc34]10 *
11 * Gaim without gaim - for BitlBee
12 *
13 * This file contains functions called by the Gaim IM-modules. It contains
14 * some struct and type definitions from Gaim.
15 *
16 * Copyright (C) 1998-1999, Mark Spencer <markster@marko.net>
17 *                          (and possibly other members of the Gaim team)
[0e788f5]18 * Copyright 2002-2012 Wilmer van der Gaast <wilmer@gaast.net>
[b7d3cc34]19 */
20
21/*
22  This program is free software; you can redistribute it and/or modify
23  it under the terms of the GNU General Public License as published by
24  the Free Software Foundation; either version 2 of the License, or
25  (at your option) any later version.
26
27  This program is distributed in the hope that it will be useful,
28  but WITHOUT ANY WARRANTY; without even the implied warranty of
29  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
30  GNU General Public License for more details.
31
32  You should have received a copy of the GNU General Public License with
33  the Debian GNU/Linux distribution in /usr/share/common-licenses/GPL;
[6f10697]34  if not, write to the Free Software Foundation, Inc., 51 Franklin St.,
35  Fifth Floor, Boston, MA  02110-1301  USA
[b7d3cc34]36*/
37
38#ifndef _NOGAIM_H
39#define _NOGAIM_H
40
[5ebff60]41#if (__sun)
[daae10f]42#include <inttypes.h>
43#else
[4cf80bb]44#include <stdint.h>
[daae10f]45#endif
46
[b7d3cc34]47#include "bitlbee.h"
[0a3c243]48#include "account.h"
[b7d3cc34]49#include "proxy.h"
[9143aeb]50#include "query.h"
[b7d3cc34]51#include "md5.h"
[2c2df7d]52#include "ft.h"
[b7d3cc34]53
54#define BUDDY_ALIAS_MAXLEN 388   /* because MSN names can be 387 characters */
55
[1c8a7a2]56#define WEBSITE "http://www.bitlbee.org/"
[b7d3cc34]57
[9624fdf]58/* Sharing flags between all kinds of things. I just hope I won't hit any
59   limits before 32-bit machines become extinct. ;-) */
[6bbb939]60#define OPT_LOGGED_IN   0x00000001
61#define OPT_LOGGING_OUT 0x00000002
62#define OPT_AWAY        0x00000004
[0ebf919]63#define OPT_MOBILE      0x00000008
[6bbb939]64#define OPT_DOES_HTML   0x00000010
[6286f80]65#define OPT_LOCALBUDDY  0x00000020 /* For nicks local to one groupchat */
[748bcdd]66#define OPT_SLOW_LOGIN  0x00000040 /* I.e. Twitter Oauth @ login time */
[df1fb67]67#define OPT_TYPING      0x00000100 /* Some pieces of code make assumptions */
68#define OPT_THINKING    0x00000200 /* about these values... Stupid me! */
[be999a5]69#define OPT_NOOTR       0x00001000 /* protocol not suitable for OTR */
[e132b60]70#define OPT_PONGS       0x00010000 /* Service sends us keep-alives */
71#define OPT_PONGED      0x00020000 /* Received a keep-alive during last interval */
[345577b]72#define OPT_SELFMESSAGE 0x00080000 /* A message sent by self from another location */
[fb62f81f]73
[b7d3cc34]74/* ok. now the fun begins. first we create a connection structure */
[5ebff60]75struct im_connection {
[0a3c243]76        account_t *acc;
[52744f8]77        uint32_t flags;
[5ebff60]78
[192b80a]79        /* each connection then can have its own protocol-specific data */
80        void *proto_data;
[5ebff60]81
[b7d3cc34]82        /* all connections need an input watcher */
83        int inpa;
[192b80a]84        guint keepalive;
[5ebff60]85
[b7d3cc34]86        /* buddy list stuff. there is still a global groups for the buddy list, but
87         * we need to maintain our own set of buddies, and our own permit/deny lists */
88        GSList *permit;
89        GSList *deny;
90        int permdeny;
[5ebff60]91
[b7d3cc34]92        char *away;
[5ebff60]93
[b7d3cc34]94        /* BitlBee */
[81e04e1]95        bee_t *bee;
[5ebff60]96
[aea8b68]97        GSList *groupchats;
[b7d3cc34]98};
99
[0da65d5]100struct groupchat {
101        struct im_connection *ic;
[b7d3cc34]102
103        /* stuff used just for chat */
[acd61b9]104        /* The in_room variable is a list of handles (not nicks!), kind of
105         * "nick list". This is how you can check who is in the group chat
106         * already, for example to avoid adding somebody two times. */
[fa29d093]107        GList *in_room;
[aea8b68]108        //GList *ignored;
[5ebff60]109
[aea8b68]110        //struct groupchat *next;
[acd61b9]111        /* The title variable contains the ID you gave when you created the
112         * chat using imcb_chat_new(). */
[fa29d093]113        char *title;
[50e1776]114        /* Use imcb_chat_topic() to change this variable otherwise the user
115         * won't notice the topic change. */
116        char *topic;
[fa29d093]117        char joined;
[acd61b9]118        /* This is for you, you can add your own structure here to extend this
119         * structure for your protocol's needs. */
[fa29d093]120        void *data;
[aea8b68]121        void *ui_data;
[b7d3cc34]122};
123
124struct buddy {
125        char name[80];
126        char show[BUDDY_ALIAS_MAXLEN];
[fa29d093]127        int present;
[b7d3cc34]128        time_t signon;
129        time_t idle;
[fa29d093]130        int uc;
[b7d3cc34]131        guint caps; /* woohoo! */
132        void *proto_data; /* what a hack */
[0da65d5]133        struct im_connection *ic; /* the connection it belongs to */
[b7d3cc34]134};
135
[d88c92a]136struct buddy_action {
137        char *name;
138        char *description;
139};
140
[b7d3cc34]141struct prpl {
142        int options;
[acd61b9]143        /* You should set this to the name of your protocol.
144         * - The user sees this name ie. when imcb_log() is used. */
[7b23afd]145        const char *name;
[cd741d8]146        void *data;
[1dd3470]147        /* Maximum Message Size of this protocol.
148         * - Introduced for OTR, in order to fragment large protocol messages.
149         * - 0 means "unlimited". */
150        unsigned int mms;
[b7d3cc34]151
[0da65d5]152        /* Added this one to be able to add per-account settings, don't think
[acd61b9]153         * it should be used for anything else. You are supposed to use the
154         * set_add() function to add new settings. */
[5ebff60]155        void (* init)           (account_t *);
[acd61b9]156        /* The typical usage of the login() function:
157         * - Create an im_connection using imcb_new() from the account_t parameter.
158         * - Initialize your myproto_data struct - you should store all your protocol-specific data there.
159         * - Save your custom structure to im_connection->proto_data.
160         * - Use proxy_connect() to connect to the server.
161         */
[5ebff60]162        void (* login)          (account_t *);
[acd61b9]163        /* Implementing this function is optional. */
[5ebff60]164        void (* keepalive)      (struct im_connection *);
[acd61b9]165        /* In this function you should:
166         * - Tell the server about you are logging out.
167         * - g_free() your myproto_data struct as BitlBee does not know how to
168         *   properly do so.
169         */
[5ebff60]170        void (* logout)         (struct im_connection *);
171
[acd61b9]172        /* This function is called when the user wants to send a message to a handle.
173         * - 'to' is a handle, not a nick
174         * - 'flags' may be ignored
175         */
[5ebff60]176        int (* buddy_msg)      (struct im_connection *, char *to, char *message, int flags);
[acd61b9]177        /* This function is called then the user uses the /away IRC command.
178         * - 'state' contains the away reason.
179         * - 'message' may be ignored if your protocol does not support it.
180         */
[5ebff60]181        void (* set_away)       (struct im_connection *, char *state, char *message);
[acd61b9]182        /* Implementing this function is optional. */
[5ebff60]183        int (* send_typing)    (struct im_connection *, char *who, int flags);
184
[acd61b9]185        /* 'name' is a handle to add/remove. For now BitlBee doesn't really
186         * handle groups, just set it to NULL, so you can ignore that
187         * parameter. */
[5ebff60]188        void (* add_buddy)      (struct im_connection *, char *name, char *group);
189        void (* remove_buddy)   (struct im_connection *, char *name, char *group);
190
[acd61b9]191        /* Block list stuff. Implementing these are optional. */
[5ebff60]192        void (* add_permit)     (struct im_connection *, char *who);
193        void (* add_deny)       (struct im_connection *, char *who);
194        void (* rem_permit)     (struct im_connection *, char *who);
195        void (* rem_deny)       (struct im_connection *, char *who);
196
[0da65d5]197        /* Request profile info. Free-formatted stuff, the IM module gives back
[acd61b9]198           this info via imcb_log(). Implementing these are optional. */
[5ebff60]199        void (* get_info)       (struct im_connection *, char *who);
200
[192b80a]201        /* Group chat stuff. */
[acd61b9]202        /* This is called when the user uses the /invite IRC command.
203         * - 'who' may be ignored
204         * - 'message' is a handle to invite
205         */
[5ebff60]206        void (* chat_invite)    (struct groupchat *, char *who, char *message);
[7821ee8]207        /* This is called when the user uses the /kick IRC command.
208         * - 'who' is a handle to kick
209         * - 'message' is a kick message or NULL
210         */
[5ebff60]211        void (* chat_kick)      (struct groupchat *, char *who, const char *message);
[acd61b9]212        /* This is called when the user uses the /part IRC command in a group
213         * chat. You just should tell the user about it, nothing more. */
[5ebff60]214        void (* chat_leave)     (struct groupchat *);
[acd61b9]215        /* This is called when the user sends a message to the groupchat.
216         * 'flags' may be ignored. */
[5ebff60]217        void (* chat_msg)       (struct groupchat *, char *message, int flags);
[acd61b9]218        /* This is called when the user uses the /join #nick IRC command.
219         * - 'who' is the handle of the nick
220         */
[0da65d5]221        struct groupchat *
[5ebff60]222        (* chat_with)      (struct im_connection *, char *who);
[acd61b9]223        /* This is used when the user uses the /join #channel IRC command.  If
224         * your protocol does not support publicly named group chats, then do
225         * not implement this. */
[0da65d5]226        struct groupchat *
[5ebff60]227        (* chat_join)      (struct im_connection *, const char *room,
228                            const char *nick, const char *password, set_t **sets);
[50e1776]229        /* Change the topic, if supported. Note that BitlBee expects the IM
230           server to confirm the topic change with a regular topic change
231           event. If it doesn't do that, you have to fake it to make it
232           visible to the user. */
[5ebff60]233        void (* chat_topic)     (struct groupchat *, char *topic);
234
[03f3828]235        /* If your protocol module needs any special info for joining chatrooms
236           other than a roomname + nickname, add them here. */
[5ebff60]237        void (* chat_add_settings)      (account_t *acc, set_t **head);
238        void (* chat_free_settings)     (account_t *acc, set_t **head);
239
[acd61b9]240        /* You can tell what away states your protocol supports, so that
[2bc8ac0]241         * BitlBee will try to map the IRC away reasons to them. If your
242         * protocol doesn't have any, just return one generic "Away". */
[0da65d5]243        GList *(* away_states)(struct im_connection *ic);
[5ebff60]244
[acd61b9]245        /* Mainly for AOL, since they think "Bung hole" == "Bu ngho le". *sigh*
246         * - Most protocols will just want to set this to g_strcasecmp().*/
[5b52a48]247        int (* handle_cmp) (const char *who1, const char *who2);
[2ff2076]248
[fa295e36]249        /* Implement these callbacks if you want to use imcb_ask_auth() */
[5ebff60]250        void (* auth_allow)     (struct im_connection *, const char *who);
251        void (* auth_deny)      (struct im_connection *, const char *who);
[2288705]252
[2ff2076]253        /* Incoming transfer request */
[5ebff60]254        void (* transfer_request) (struct im_connection *, file_transfer_t *ft, char *handle);
255
[203a2d2]256        void (* buddy_data_add) (struct bee_user *bu);
257        void (* buddy_data_free) (struct bee_user *bu);
[5ebff60]258
[d88c92a]259        GList *(* buddy_action_list) (struct bee_user *bu);
260        void *(* buddy_action) (struct bee_user *bu, const char *action, char * const args[], void *data);
[5ebff60]261
[be1efa3]262        /* If null, equivalent to handle_cmp( ic->acc->user, who ) */
263        gboolean (* handle_is_self) (struct im_connection *, const char *who);
264
[03f3828]265        /* Some placeholders so eventually older plugins may cooperate with newer BitlBees. */
266        void *resv1;
267        void *resv2;
268        void *resv3;
269        void *resv4;
270        void *resv5;
[b7d3cc34]271};
272
[d28fe1c4]273struct plugin_info
274{
275        guint abiver;
276        const char *name;
277        const char *version;
278        const char *description;
279        const char *author;
280        const char *url;
281};
282
283#ifdef WITH_PLUGINS
284G_MODULE_EXPORT GList *get_plugins();
285#endif
286
[84b045d]287/* im_api core stuff. */
288void nogaim_init();
[b7d3cc34]289G_MODULE_EXPORT GSList *get_connections();
[5ebff60]290G_MODULE_EXPORT struct prpl *find_protocol(const char *name);
[ad9ac5d]291G_MODULE_EXPORT gboolean is_protocol_disabled(const char *name);
[acd61b9]292/* When registering a new protocol, you should allocate space for a new prpl
293 * struct, initialize it (set the function pointers to point to your
294 * functions), finally call this function. */
[5ebff60]295G_MODULE_EXPORT void register_protocol(struct prpl *);
[b7d3cc34]296
[84b045d]297/* Connection management. */
[acd61b9]298/* You will need this function in prpl->login() to get an im_connection from
299 * the account_t parameter. */
[5ebff60]300G_MODULE_EXPORT struct im_connection *imcb_new(account_t *acc);
301G_MODULE_EXPORT void imc_free(struct im_connection *ic);
[acd61b9]302/* Once you're connected, you should call this function, so that the user will
303 * see the success. */
[5ebff60]304G_MODULE_EXPORT void imcb_connected(struct im_connection *ic);
[acd61b9]305/* This can be used to disconnect when something went wrong (ie. read error
306 * from the server). You probably want to set the second parameter to TRUE. */
[5ebff60]307G_MODULE_EXPORT void imc_logout(struct im_connection *ic, int allow_reconnect);
[b7d3cc34]308
[84b045d]309/* Communicating with the user. */
[acd61b9]310/* A printf()-like function to tell the user anything you want. */
[5ebff60]311G_MODULE_EXPORT void imcb_log(struct im_connection *ic, char *format, ...) G_GNUC_PRINTF(2, 3);
[acd61b9]312/* To tell the user an error, ie. before logging out when an error occurs. */
[5ebff60]313G_MODULE_EXPORT void imcb_error(struct im_connection *ic, char *format, ...) G_GNUC_PRINTF(2, 3);
[fa295e36]314
[acd61b9]315/* To ask a your about something.
316 * - 'msg' is the question.
317 * - 'data' can be your custom struct - it will be passed to the callbacks.
318 * - 'doit' or 'dont' will be called depending of the answer of the user.
319 */
[5ebff60]320G_MODULE_EXPORT void imcb_ask(struct im_connection *ic, char *msg, void *data, query_callback doit,
321                              query_callback dont);
322G_MODULE_EXPORT void imcb_ask_with_free(struct im_connection *ic, char *msg, void *data, query_callback doit,
323                                        query_callback dont, query_callback myfree);
[fa295e36]324
325/* Two common questions you may want to ask:
326 * - X added you to his contact list, allow?
327 * - X is not in your contact list, want to add?
328 */
[5ebff60]329G_MODULE_EXPORT void imcb_ask_auth(struct im_connection *ic, const char *handle, const char *realname);
330G_MODULE_EXPORT void imcb_ask_add(struct im_connection *ic, const char *handle, const char *realname);
[b7d3cc34]331
[84b045d]332/* Buddy management */
[acd61b9]333/* This function should be called for each handle which are visible to the
[527360f]334 * user, usually after a login, or if the user added a buddy and the IM
335 * server confirms that the add was successful. Don't forget to do this! */
[5ebff60]336G_MODULE_EXPORT void imcb_add_buddy(struct im_connection *ic, const char *handle, const char *group);
337G_MODULE_EXPORT void imcb_remove_buddy(struct im_connection *ic, const char *handle, char *group);
338G_MODULE_EXPORT void imcb_rename_buddy(struct im_connection *ic, const char *handle, const char *realname);
339G_MODULE_EXPORT void imcb_buddy_nick_hint(struct im_connection *ic, const char *handle, const char *nick);
[a42fda4]340G_MODULE_EXPORT void imcb_buddy_nick_change(struct im_connection *ic, const char *handle, const char *nick);
[5ebff60]341G_MODULE_EXPORT void imcb_buddy_action_response(bee_user_t *bu, const char *action, char * const args[], void *data);
[84b045d]342
[345577b]343G_MODULE_EXPORT void imcb_buddy_typing(struct im_connection *ic, const char *handle, guint32 flags);
[5ebff60]344G_MODULE_EXPORT struct bee_user *imcb_buddy_by_handle(struct im_connection *ic, const char *handle);
[d6e2aa8]345
346G_GNUC_DEPRECATED G_MODULE_EXPORT void imcb_clean_handle(struct im_connection *ic, char *handle);
[84b045d]347
348/* Actions, or whatever. */
[5ebff60]349int imc_away_send_update(struct im_connection *ic);
350int imc_chat_msg(struct groupchat *c, char *msg, int flags);
[84b045d]351
[5ebff60]352void imc_add_allow(struct im_connection *ic, char *handle);
353void imc_rem_allow(struct im_connection *ic, char *handle);
354void imc_add_block(struct im_connection *ic, char *handle);
355void imc_rem_block(struct im_connection *ic, char *handle);
[84b045d]356
357/* Misc. stuff */
[5ebff60]358char *set_eval_timezone(set_t *set, char *value);
359gboolean auto_reconnect(gpointer data, gint fd, b_input_condition cond);
360void cancel_auto_reconnect(struct account *a);
[84b045d]361
[b7d3cc34]362#endif
Note: See TracBrowser for help on using the repository browser.