]>
jfr.im git - solanum.git/blob - modules/m_cap.c
3 * Copyright (C) 2005 Lee Hardy <lee@leeh.co.uk>
4 * Copyright (C) 2005 ircd-ratbox development team
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions are
10 * 1.Redistributions of source code must retain the above copyright notice,
11 * this list of conditions and the following disclaimer.
12 * 2.Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3.The name of the author may not be used to endorse or promote products
16 * derived from this software without specific prior written permission.
18 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21 * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
22 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
24 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
26 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
27 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28 * POSSIBILITY OF SUCH DAMAGE.
30 * $Id: m_cap.c 676 2006-02-03 20:05:09Z gxti $
47 typedef int (*bqcmp
)(const void *, const void *);
49 static int m_cap(struct Client
*, struct Client
*, int, const char **);
50 static int modinit(void);
52 struct Message cap_msgtab
= {
53 "CAP", 0, 0, 0, MFLG_SLOW
,
54 {{m_cap
, 2}, {m_cap
, 2}, mg_ignore
, mg_ignore
, mg_ignore
, {m_cap
, 2}}
57 mapi_clist_av1 cap_clist
[] = { &cap_msgtab
, NULL
};
58 DECLARE_MODULE_AV1(cap
, modinit
, NULL
, cap_clist
, NULL
, NULL
, "$Revision: 676 $");
60 #define _CLICAP(name, capserv, capclient, caprequired, flags) \
61 { (name), (capserv), (capclient), (caprequired), (flags), sizeof(name) - 1 }
63 #define CLICAP_FLAGS_STICKY 0x001
68 int cap_serv
; /* for altering s->c */
69 int cap_cli
; /* for altering c->s */
70 int cap_required_serv
; /* required dependency cap */
74 _CLICAP("multi-prefix", CLICAP_MULTI_PREFIX
, 0, 0, 0),
75 _CLICAP("sasl", CLICAP_SASL
, 0, 0, 0),
76 _CLICAP("account-notify", CLICAP_ACCOUNT_NOTIFY
, 0, 0, 0),
77 _CLICAP("extended-join", CLICAP_EXTENDED_JOIN
, 0, 0, 0),
78 _CLICAP("away-notify", CLICAP_AWAY_NOTIFY
, 0, 0, 0),
79 _CLICAP("tls", CLICAP_TLS
, 0, 0, 0),
80 _CLICAP("userhost-in-names", CLICAP_USERHOST_IN_NAMES
, 0, 0, 0),
81 _CLICAP("cap-notify", CLICAP_CAP_NOTIFY
, 0, 0, 0),
84 #define CLICAP_LIST_LEN (sizeof(clicap_list) / sizeof(struct clicap))
86 static int clicap_sort(struct clicap
*, struct clicap
*);
91 qsort(clicap_list
, CLICAP_LIST_LEN
, sizeof(struct clicap
),
97 clicap_sort(struct clicap
*one
, struct clicap
*two
)
99 return irccmp(one
->name
, two
->name
);
103 clicap_compare(const char *name
, struct clicap
*cap
)
105 return irccmp(name
, cap
->name
);
109 * Used iteratively over a buffer, extracts individual cap tokens.
111 * Inputs: buffer to start iterating over (NULL to iterate over existing buf)
112 * int pointer to whether the cap token is negated
113 * int pointer to whether we finish with success
114 * Ouputs: Cap entry if found, NULL otherwise.
116 static struct clicap
*
117 clicap_find(const char *data
, int *negate
, int *finished
)
119 static char buf
[BUFSIZE
];
128 rb_strlcpy(buf
, data
, sizeof(buf
));
135 /* skip any whitespace */
136 while(*p
&& IsSpace(*p
))
150 /* someone sent a '-' without a parameter.. */
155 if((s
= strchr(p
, ' ')))
158 if((cap
= bsearch(p
, clicap_list
, CLICAP_LIST_LEN
,
159 sizeof(struct clicap
), (bqcmp
) clicap_compare
)))
171 * Generates a list of capabilities.
173 * Inputs: client to send to, subcmd to send,
174 * flags to match against: 0 to do none, -1 if client has no flags,
175 * int to whether we are doing CAP CLEAR
179 clicap_generate(struct Client
*source_p
, const char *subcmd
, int flags
, int clear
)
182 char capbuf
[BUFSIZE
];
188 mlen
= rb_sprintf(buf
, ":%s CAP %s %s",
190 EmptyString(source_p
->name
) ? "*" : source_p
->name
,
196 /* shortcut, nothing to do */
199 sendto_one(source_p
, "%s :", buf
);
203 for(i
= 0; i
< CLICAP_LIST_LEN
; i
++)
207 if(!IsCapable(source_p
, clicap_list
[i
].cap_serv
))
209 /* they are capable of this, check sticky */
210 else if(clear
&& clicap_list
[i
].flags
& CLICAP_FLAGS_STICKY
)
214 if (clicap_list
[i
].cap_serv
== CLICAP_SASL
)
216 struct Client
*agent_p
= NULL
;
218 if (!ConfigFileEntry
.sasl_service
)
221 agent_p
= find_named_client(ConfigFileEntry
.sasl_service
);
222 if (agent_p
== NULL
|| !IsService(agent_p
))
226 /* \r\n\0, possible "-~=", space, " *" */
227 if(buflen
+ clicap_list
[i
].namelen
>= BUFSIZE
- 10)
229 /* remove our trailing space -- if buflen == mlen
230 * here, we didnt even succeed in adding one.
237 sendto_one(source_p
, "%s * :%s", buf
, capbuf
);
247 /* needs a client ack */
248 if(clicap_list
[i
].cap_cli
&&
249 IsCapable(source_p
, clicap_list
[i
].cap_cli
))
257 if(clicap_list
[i
].flags
& CLICAP_FLAGS_STICKY
)
263 /* if we're doing an LS, then we only send this if
266 if(clicap_list
[i
].cap_cli
&&
267 (!flags
|| !IsCapable(source_p
, clicap_list
[i
].cap_cli
)))
274 curlen
= rb_sprintf(p
, "%s ", clicap_list
[i
].name
);
279 /* remove trailing space */
285 sendto_one(source_p
, "%s :%s", buf
, capbuf
);
289 cap_ack(struct Client
*source_p
, const char *arg
)
292 int capadd
= 0, capdel
= 0;
293 int finished
= 0, negate
;
298 for(cap
= clicap_find(arg
, &negate
, &finished
); cap
;
299 cap
= clicap_find(NULL
, &negate
, &finished
))
301 /* sent an ACK for something they havent REQd */
302 if(!IsCapable(source_p
, cap
->cap_serv
))
307 /* dont let them ack something sticky off */
308 if(cap
->flags
& CLICAP_FLAGS_STICKY
)
311 capdel
|= cap
->cap_cli
;
314 capadd
|= cap
->cap_cli
;
317 source_p
->localClient
->caps
|= capadd
;
318 source_p
->localClient
->caps
&= ~capdel
;
322 cap_clear(struct Client
*source_p
, const char *arg
)
324 clicap_generate(source_p
, "ACK",
325 source_p
->localClient
->caps
? source_p
->localClient
->caps
: -1, 1);
327 /* XXX - sticky capabs */
329 source_p
->localClient
->caps
= source_p
->localClient
->caps
& CLICAP_STICKY
;
331 source_p
->localClient
->caps
= 0;
336 cap_end(struct Client
*source_p
, const char *arg
)
338 if(IsRegistered(source_p
))
341 source_p
->flags
&= ~FLAGS_CLICAP
;
343 if(source_p
->name
[0] && source_p
->flags
& FLAGS_SENTUSER
)
346 rb_strlcpy(buf
, source_p
->username
, sizeof(buf
));
347 register_local_user(source_p
, source_p
, buf
);
352 cap_list(struct Client
*source_p
, const char *arg
)
354 /* list of what theyre currently using */
355 clicap_generate(source_p
, "LIST",
356 source_p
->localClient
->caps
? source_p
->localClient
->caps
: -1, 0);
360 cap_ls(struct Client
*source_p
, const char *arg
)
362 if(!IsRegistered(source_p
))
363 source_p
->flags
|= FLAGS_CLICAP
;
365 /* list of what we support */
366 clicap_generate(source_p
, "LS", 0, 0);
370 cap_req(struct Client
*source_p
, const char *arg
)
373 char pbuf
[2][BUFSIZE
];
377 int capadd
= 0, capdel
= 0;
378 int finished
= 0, negate
;
380 if(!IsRegistered(source_p
))
381 source_p
->flags
|= FLAGS_CLICAP
;
386 buflen
= rb_snprintf(buf
, sizeof(buf
), ":%s CAP %s ACK",
387 me
.name
, EmptyString(source_p
->name
) ? "*" : source_p
->name
);
392 for(cap
= clicap_find(arg
, &negate
, &finished
); cap
;
393 cap
= clicap_find(NULL
, &negate
, &finished
))
395 /* filled the first array, but cant send it in case the
396 * request fails. one REQ should never fill more than two
399 if(buflen
+ plen
+ cap
->namelen
+ 6 >= BUFSIZE
)
408 if(cap
->flags
& CLICAP_FLAGS_STICKY
)
414 strcat(pbuf
[i
], "-");
417 capdel
|= cap
->cap_serv
;
421 if(cap
->cap_required_serv
&& !((capadd
& cap
->cap_required_serv
) == cap
->cap_required_serv
|| IsCapable(source_p
, cap
->cap_required_serv
)))
427 if (cap
->cap_serv
== CLICAP_SASL
)
429 struct Client
*agent_p
= NULL
;
431 if (!ConfigFileEntry
.sasl_service
)
437 agent_p
= find_named_client(ConfigFileEntry
.sasl_service
);
438 if (agent_p
== NULL
|| !IsService(agent_p
))
445 if(cap
->flags
& CLICAP_FLAGS_STICKY
)
447 strcat(pbuf
[i
], "=");
451 capadd
|= cap
->cap_serv
;
456 strcat(pbuf
[i
], "~");
460 strcat(pbuf
[i
], cap
->name
);
461 strcat(pbuf
[i
], " ");
462 plen
+= (cap
->namelen
+ 1);
467 sendto_one(source_p
, ":%s CAP %s NAK :%s",
468 me
.name
, EmptyString(source_p
->name
) ? "*" : source_p
->name
, arg
);
474 sendto_one(source_p
, "%s * :%s", buf
, pbuf
[0]);
475 sendto_one(source_p
, "%s :%s", buf
, pbuf
[1]);
478 sendto_one(source_p
, "%s :%s", buf
, pbuf
[0]);
480 source_p
->localClient
->caps
|= capadd
;
481 source_p
->localClient
->caps
&= ~capdel
;
484 static struct clicap_cmd
487 void (*func
)(struct Client
*source_p
, const char *arg
);
488 } clicap_cmdlist
[] = {
489 /* This list *MUST* be in alphabetical order */
491 { "CLEAR", cap_clear
},
493 { "LIST", cap_list
},
499 clicap_cmd_search(const char *command
, struct clicap_cmd
*entry
)
501 return irccmp(command
, entry
->cmd
);
505 m_cap(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
507 struct clicap_cmd
*cmd
;
509 if(!(cmd
= bsearch(parv
[1], clicap_cmdlist
,
510 sizeof(clicap_cmdlist
) / sizeof(struct clicap_cmd
),
511 sizeof(struct clicap_cmd
), (bqcmp
) clicap_cmd_search
)))
513 sendto_one(source_p
, form_str(ERR_INVALIDCAPCMD
),
514 me
.name
, EmptyString(source_p
->name
) ? "*" : source_p
->name
,
519 (cmd
->func
)(source_p
, parv
[2]);