Mercurial > pidgin
annotate src/protocols/oscar/locate.c @ 9554:8b2451878e26
[gaim-migrate @ 10387]
" This patch adds chat user status icons (voice / halfop
/ op / founder) to chats
There's a screenshot here, showing ops, voices and
ignored ops and voices
http://nosnilmot.com/gaim/chatusers.png
This required some changes in how the core stores the
list of users in chats to be able to store the status
too, which are detailed below.
I also fixed up some memory leaks as I came across them
(string values returned by gtk_tree_model_get() not
being g_free()'d) and a minor bug in signals-test.c
Conversation API:
Changed:
gaim_conv_chat_add_user() (added flags parameter)
gaim_conv_chat_add_users() now (added GList of flags
parameter)
gaim_conv_chat_get_users() now returns a GList of
GaimChatBuddy's
gaim_conv_chat_set_users() now expects a GList of
GaimChatBuddy's
Added:
gaim_conv_chat_set_user_flags()
gaim_conv_chat_get_user_flags()
gaim_conv_chat_find_user()
gaim_conv_chat_cb_new()
gaim_conv_chat_cb_find()
gaim_conv_chat_cb_destroy()
gaim_conv_chat_cb_get_name()
Conversation UI ops:
added: chat_update_user()
Signals:
Changed: chat-buddy-joining & chat-buddy-joined now
include the user's flags
Added: chat-buddy-flags for when user's flags change
Added:
gaim_marshal_VOID__POINTER_POINTER_POINTER_UINT_UINT
(required for the new chat-buddy-flags signal)
Protocol Plugins:
All updated to work with above changes (obviously)
User flags support added to IRC, Jabber and SILC
New Files:
pixmaps/status/default/ voice.svg halfop.svg op.svg
founder.svg
" --Stu Tomlinson
committer: Tailor Script <tailor@pidgin.im>
author | Luke Schierer <lschiere@pidgin.im> |
---|---|
date | Sat, 17 Jul 2004 18:11:12 +0000 |
parents | 92cbf9713795 |
children | 9f358a718f38 |
rev | line source |
---|---|
7011 | 1 /* |
2 * Family 0x0002 - Locate. | |
3 * | |
4 * The functions here are responsible for requesting and parsing information- | |
5 * gathering SNACs. Or something like that. This family contains the SNACs | |
6 * for getting and setting info, away messages, directory profile thingy, etc. | |
7 */ | |
8 | |
9 #define FAIM_INTERNAL | |
10 #include <aim.h> | |
11 #ifdef _WIN32 | |
12 #include "win32dep.h" | |
13 #endif | |
14 | |
15 /* | |
16 * Capability blocks. | |
17 * | |
18 * These are CLSIDs. They should actually be of the form: | |
19 * | |
20 * {0x0946134b, 0x4c7f, 0x11d1, | |
21 * {0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}}, | |
22 * | |
23 * But, eh. | |
24 */ | |
25 static const struct { | |
26 fu32_t flag; | |
27 fu8_t data[16]; | |
28 } aim_caps[] = { | |
29 | |
30 /* | |
31 * These are in ascending numerical order. | |
32 */ | |
7253 | 33 |
34 /* | |
35 * Perhaps better called AIM_CAPS_SHORTCAPS | |
36 */ | |
7011 | 37 {AIM_CAPS_ICHAT, |
38 {0x09, 0x46, 0x00, 0x00, 0x4c, 0x7f, 0x11, 0xd1, | |
39 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
40 | |
41 {AIM_CAPS_SECUREIM, | |
42 {0x09, 0x46, 0x00, 0x01, 0x4c, 0x7f, 0x11, 0xd1, | |
43 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
44 | |
7582 | 45 {AIM_CAPS_VIDEO, |
46 {0x09, 0x46, 0x01, 0x00, 0x4c, 0x7f, 0x11, 0xd1, | |
47 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
48 | |
8092 | 49 /* "Live Video" support in Windows AIM 5.5.3501 and newer */ |
50 {AIM_CAPS_LIVEVIDEO, | |
51 {0x09, 0x46, 0x01, 0x01, 0x4c, 0x7f, 0x11, 0xd1, | |
52 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
53 | |
54 /* "Camera" support in Windows AIM 5.5.3501 and newer */ | |
55 {AIM_CAPS_CAMERA, | |
56 {0x09, 0x46, 0x01, 0x02, 0x4c, 0x7f, 0x11, 0xd1, | |
57 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
58 | |
59 /* In Windows AIM 5.5.3501 and newer */ | |
60 {AIM_CAPS_GENERICUNKNOWN, | |
61 {0x09, 0x46, 0x01, 0x03, 0x4c, 0x7f, 0x11, 0xd1, | |
62 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
63 | |
64 /* In iChatAV (version numbers...?) */ | |
7945 | 65 {AIM_CAPS_ICHATAV, |
66 {0x09, 0x46, 0x01, 0x05, 0x4c, 0x7f, 0x11, 0xd1, | |
67 0x82, 0x22, 0x44, 0x45, 0x45, 0x53, 0x54, 0x00}}, | |
68 | |
7025 | 69 /* |
70 * Not really sure about this one. In an email from | |
71 * 26 Sep 2003, Matthew Sachs suggested that, "this | |
72 * is probably the capability for the SMS features." | |
73 */ | |
74 {AIM_CAPS_SMS, | |
75 {0x09, 0x46, 0x01, 0xff, 0x4c, 0x7f, 0x11, 0xd1, | |
76 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
77 | |
7091 | 78 {AIM_CAPS_GENERICUNKNOWN, |
7025 | 79 {0x09, 0x46, 0xf0, 0x03, 0x4c, 0x7f, 0x11, 0xd1, |
80 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
81 | |
7091 | 82 {AIM_CAPS_GENERICUNKNOWN, |
7025 | 83 {0x09, 0x46, 0xf0, 0x04, 0x4c, 0x7f, 0x11, 0xd1, |
84 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
85 | |
7091 | 86 {AIM_CAPS_GENERICUNKNOWN, |
7025 | 87 {0x09, 0x46, 0xf0, 0x05, 0x4c, 0x7f, 0x11, 0xd1, |
88 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
89 | |
7011 | 90 {AIM_CAPS_HIPTOP, |
91 {0x09, 0x46, 0x13, 0x23, 0x4c, 0x7f, 0x11, 0xd1, | |
92 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
93 | |
8092 | 94 {AIM_CAPS_TALK, |
7011 | 95 {0x09, 0x46, 0x13, 0x41, 0x4c, 0x7f, 0x11, 0xd1, |
96 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
97 | |
98 {AIM_CAPS_SENDFILE, | |
99 {0x09, 0x46, 0x13, 0x43, 0x4c, 0x7f, 0x11, 0xd1, | |
100 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
101 | |
7141 | 102 {AIM_CAPS_ICQ_DIRECT, |
7011 | 103 {0x09, 0x46, 0x13, 0x44, 0x4c, 0x7f, 0x11, 0xd1, |
104 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
105 | |
106 {AIM_CAPS_DIRECTIM, | |
107 {0x09, 0x46, 0x13, 0x45, 0x4c, 0x7f, 0x11, 0xd1, | |
108 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
109 | |
110 {AIM_CAPS_BUDDYICON, | |
111 {0x09, 0x46, 0x13, 0x46, 0x4c, 0x7f, 0x11, 0xd1, | |
112 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
113 | |
8092 | 114 {AIM_CAPS_ADDINS, |
7011 | 115 {0x09, 0x46, 0x13, 0x47, 0x4c, 0x7f, 0x11, 0xd1, |
116 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
117 | |
118 {AIM_CAPS_GETFILE, | |
119 {0x09, 0x46, 0x13, 0x48, 0x4c, 0x7f, 0x11, 0xd1, | |
120 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
121 | |
122 {AIM_CAPS_ICQSERVERRELAY, | |
123 {0x09, 0x46, 0x13, 0x49, 0x4c, 0x7f, 0x11, 0xd1, | |
124 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
125 | |
126 /* | |
127 * Indeed, there are two of these. The former appears to be correct, | |
128 * but in some versions of winaim, the second one is set. Either they | |
129 * forgot to fix endianness, or they made a typo. It really doesn't | |
130 * matter which. | |
131 */ | |
132 {AIM_CAPS_GAMES, | |
133 {0x09, 0x46, 0x13, 0x4a, 0x4c, 0x7f, 0x11, 0xd1, | |
134 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
135 {AIM_CAPS_GAMES2, | |
136 {0x09, 0x46, 0x13, 0x4a, 0x4c, 0x7f, 0x11, 0xd1, | |
137 0x22, 0x82, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
138 | |
139 {AIM_CAPS_SENDBUDDYLIST, | |
140 {0x09, 0x46, 0x13, 0x4b, 0x4c, 0x7f, 0x11, 0xd1, | |
141 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
142 | |
143 /* | |
144 * Setting this lets AIM users receive messages from ICQ users, and ICQ | |
145 * users receive messages from AIM users. It also lets ICQ users show | |
146 * up in buddy lists for AIM users, and AIM users show up in buddy lists | |
147 * for ICQ users. And ICQ privacy/invisibility acts like AIM privacy, | |
148 * in that if you add a user to your deny list, you will not be able to | |
149 * see them as online (previous you could still see them, but they | |
150 * couldn't see you. | |
151 */ | |
152 {AIM_CAPS_INTEROPERATE, | |
153 {0x09, 0x46, 0x13, 0x4d, 0x4c, 0x7f, 0x11, 0xd1, | |
154 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
155 | |
156 {AIM_CAPS_ICQUTF8, | |
157 {0x09, 0x46, 0x13, 0x4e, 0x4c, 0x7f, 0x11, 0xd1, | |
158 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
159 | |
160 {AIM_CAPS_ICQUTF8OLD, | |
161 {0x2e, 0x7a, 0x64, 0x75, 0xfa, 0xdf, 0x4d, 0xc8, | |
162 0x88, 0x6f, 0xea, 0x35, 0x95, 0xfd, 0xb6, 0xdf}}, | |
163 | |
164 /* | |
165 * Chat is oddball. | |
166 */ | |
167 {AIM_CAPS_CHAT, | |
168 {0x74, 0x8f, 0x24, 0x20, 0x62, 0x87, 0x11, 0xd1, | |
169 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}}, | |
170 | |
171 /* | |
172 {AIM_CAPS_ICQ2GO, | |
173 {0x56, 0x3f, 0xc8, 0x09, 0x0b, 0x6f, 0x41, 0xbd, | |
174 0x9f, 0x79, 0x42, 0x26, 0x09, 0xdf, 0xa2, 0xf3}}, | |
175 */ | |
176 | |
177 {AIM_CAPS_ICQRTF, | |
178 {0x97, 0xb1, 0x27, 0x51, 0x24, 0x3c, 0x43, 0x34, | |
179 0xad, 0x22, 0xd6, 0xab, 0xf7, 0x3f, 0x14, 0x92}}, | |
180 | |
7091 | 181 /* This is added by the servers and it only shows up for ourselves... */ |
182 {AIM_CAPS_GENERICUNKNOWN, | |
7011 | 183 {0x97, 0xb1, 0x27, 0x51, 0x24, 0x3c, 0x43, 0x34, |
7091 | 184 0xad, 0x22, 0xd6, 0xab, 0xf7, 0x3f, 0x14, 0x09}}, |
7011 | 185 |
186 {AIM_CAPS_APINFO, | |
187 {0xaa, 0x4a, 0x32, 0xb5, 0xf8, 0x84, 0x48, 0xc6, | |
188 0xa3, 0xd7, 0x8c, 0x50, 0x97, 0x19, 0xfd, 0x5b}}, | |
189 | |
190 {AIM_CAPS_TRILLIANCRYPT, | |
191 {0xf2, 0xe7, 0xc7, 0xf4, 0xfe, 0xad, 0x4d, 0xfb, | |
192 0xb2, 0x35, 0x36, 0x79, 0x8b, 0xdf, 0x00, 0x00}}, | |
193 | |
194 {AIM_CAPS_EMPTY, | |
195 {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | |
196 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}}, | |
197 | |
198 {AIM_CAPS_LAST} | |
199 }; | |
200 | |
201 /* | |
202 * Add the userinfo to our linked list. If we already have userinfo | |
203 * for this buddy, then just overwrite parts of the old data. | |
204 * @param userinfo Contains the new information for the buddy. | |
205 */ | |
7045 | 206 static void aim_locate_adduserinfo(aim_session_t *sess, aim_userinfo_t *userinfo) { |
7011 | 207 aim_userinfo_t *cur; |
208 | |
7045 | 209 cur = aim_locate_finduserinfo(sess, userinfo->sn); |
7011 | 210 |
211 if (cur == NULL) { | |
212 cur = (aim_userinfo_t *)calloc(1, sizeof(aim_userinfo_t)); | |
213 cur->sn = strdup(userinfo->sn); | |
7045 | 214 cur->next = sess->locate.userinfo; |
215 sess->locate.userinfo = cur; | |
7011 | 216 } |
217 | |
218 cur->warnlevel = userinfo->warnlevel; | |
219 cur->idletime = userinfo->idletime; | |
220 if (userinfo->flags != 0) | |
221 cur->flags = userinfo->flags; | |
222 if (userinfo->createtime != 0) | |
223 cur->createtime = userinfo->createtime; | |
224 if (userinfo->membersince != 0) | |
225 cur->membersince = userinfo->membersince; | |
226 if (userinfo->onlinesince != 0) | |
227 cur->onlinesince = userinfo->onlinesince; | |
228 if (userinfo->sessionlen != 0) | |
229 cur->sessionlen = userinfo->sessionlen; | |
230 if (userinfo->capabilities != 0) | |
231 cur->capabilities = userinfo->capabilities; | |
232 cur->present |= userinfo->present; | |
233 | |
7046 | 234 if (userinfo->iconcsumlen > 0) { |
235 free(cur->iconcsum); | |
236 cur->iconcsum = (fu8_t *)malloc(userinfo->iconcsumlen); | |
237 memcpy(cur->iconcsum, userinfo->iconcsum, userinfo->iconcsumlen); | |
238 cur->iconcsumlen = userinfo->iconcsumlen; | |
7011 | 239 } |
240 | |
7301 | 241 if (userinfo->info != NULL) { |
7011 | 242 free(cur->info); |
243 free(cur->info_encoding); | |
8393 | 244 if (userinfo->info_len > 0) { |
245 cur->info = (char *)malloc(userinfo->info_len); | |
246 memcpy(cur->info, userinfo->info, userinfo->info_len); | |
8394 | 247 } else |
248 cur->info = NULL; | |
7235 | 249 cur->info_encoding = strdup(userinfo->info_encoding); |
7011 | 250 cur->info_len = userinfo->info_len; |
251 } | |
7046 | 252 |
7301 | 253 if (userinfo->away != NULL) { |
7046 | 254 free(cur->away); |
255 free(cur->away_encoding); | |
8393 | 256 if (userinfo->away_len > 0) { |
257 cur->away = (char *)malloc(userinfo->away_len); | |
258 memcpy(cur->away, userinfo->away, userinfo->away_len); | |
8394 | 259 } else |
260 cur->away = NULL; | |
7235 | 261 cur->away_encoding = strdup(userinfo->away_encoding); |
7046 | 262 cur->away_len = userinfo->away_len; |
263 } | |
7011 | 264 } |
265 | |
8341 | 266 faim_export void aim_locate_dorequest(aim_session_t *sess) { |
7466 | 267 struct userinfo_node *cur = sess->locate.torequest; |
7011 | 268 |
269 if (cur == NULL) | |
270 return; | |
271 | |
7045 | 272 if (sess->locate.waiting_for_response == TRUE) |
7011 | 273 return; |
274 | |
7045 | 275 sess->locate.waiting_for_response = TRUE; |
7166 | 276 aim_locate_getinfoshort(sess, cur->sn, 0x00000003); |
7466 | 277 |
278 /* Move this node to the "requested" queue */ | |
279 sess->locate.torequest = cur->next; | |
280 cur->next = sess->locate.requested; | |
281 sess->locate.requested = cur; | |
7011 | 282 } |
283 | |
7447 | 284 /** |
8735
92cbf9713795
[gaim-migrate @ 9490]
Christian Hammond <chipx86@chipx86.com>
parents:
8394
diff
changeset
|
285 * Remove this screen name from our queue. If this info was requested |
7447 | 286 * by our info request queue, then pop the next element off of the queue. |
7259 | 287 * |
288 * @param sess The aim session. | |
289 * @param sn Screen name of the info we just received. | |
290 * @return True if the request was explicit (client requested the info), | |
291 * false if the request was implicit (libfaim request the info). | |
292 */ | |
293 static int aim_locate_gotuserinfo(aim_session_t *sess, const char *sn) { | |
294 struct userinfo_node *cur, *del; | |
295 int was_explicit = TRUE; | |
296 | |
7466 | 297 while ((sess->locate.requested != NULL) && (aim_sncmp(sn, sess->locate.requested->sn) == 0)) { |
298 del = sess->locate.requested; | |
299 sess->locate.requested = del->next; | |
7259 | 300 was_explicit = FALSE; |
301 free(del->sn); | |
302 free(del); | |
303 } | |
304 | |
7466 | 305 cur = sess->locate.requested; |
7259 | 306 while ((cur != NULL) && (cur->next != NULL)) { |
307 if (aim_sncmp(sn, cur->next->sn) == 0) { | |
308 del = cur->next; | |
309 cur->next = del->next; | |
310 was_explicit = FALSE; | |
311 free(del->sn); | |
312 free(del); | |
313 } else | |
314 cur = cur->next; | |
315 } | |
316 | |
317 if (!was_explicit) { | |
8341 | 318 aim_conn_t *conn = aim_conn_findbygroup(sess, AIM_CB_FAM_LOC); |
319 aim_rxcallback_t userfunc; | |
320 | |
7259 | 321 sess->locate.waiting_for_response = FALSE; |
8341 | 322 |
323 if ((userfunc = aim_callhandler(sess, conn, AIM_CB_FAM_LOC, AIM_CB_LOC_REQUESTINFOTIMEOUT))) | |
324 userfunc(sess, NULL); | |
8342 | 325 else |
326 aim_locate_dorequest(sess); | |
7259 | 327 } |
328 | |
329 return was_explicit; | |
330 } | |
331 | |
7011 | 332 faim_internal void aim_locate_requestuserinfo(aim_session_t *sess, const char *sn) { |
7045 | 333 struct userinfo_node *cur; |
7011 | 334 |
7266 | 335 /* Make sure we aren't already requesting info for this buddy */ |
7466 | 336 cur = sess->locate.torequest; |
7266 | 337 while (cur != NULL) { |
338 if (aim_sncmp(sn, cur->sn) == 0) | |
339 return; | |
340 cur = cur->next; | |
341 } | |
342 | |
343 /* Add a new node to our request queue */ | |
7045 | 344 cur = (struct userinfo_node *)malloc(sizeof(struct userinfo_node)); |
7011 | 345 cur->sn = strdup(sn); |
7466 | 346 cur->next = sess->locate.torequest; |
347 sess->locate.torequest = cur; | |
7011 | 348 |
7266 | 349 /* Actually request some info up in this piece */ |
7011 | 350 aim_locate_dorequest(sess); |
351 } | |
352 | |
7045 | 353 faim_export aim_userinfo_t *aim_locate_finduserinfo(aim_session_t *sess, const char *sn) { |
354 aim_userinfo_t *cur = sess->locate.userinfo; | |
7011 | 355 |
356 while (cur != NULL) { | |
357 if (aim_sncmp(cur->sn, sn) == 0) | |
358 return cur; | |
359 cur = cur->next; | |
360 } | |
361 | |
362 return NULL; | |
363 } | |
364 | |
7334 | 365 faim_internal fu32_t aim_locate_getcaps(aim_session_t *sess, aim_bstream_t *bs, int len) |
7011 | 366 { |
367 fu32_t flags = 0; | |
368 int offset; | |
369 | |
370 for (offset = 0; aim_bstream_empty(bs) && (offset < len); offset += 0x10) { | |
371 fu8_t *cap; | |
372 int i, identified; | |
373 | |
374 cap = aimbs_getraw(bs, 0x10); | |
375 | |
376 for (i = 0, identified = 0; !(aim_caps[i].flag & AIM_CAPS_LAST); i++) { | |
377 if (memcmp(&aim_caps[i].data, cap, 0x10) == 0) { | |
378 flags |= aim_caps[i].flag; | |
379 identified++; | |
380 break; /* should only match once... */ | |
381 } | |
382 } | |
383 | |
7334 | 384 if (!identified) |
7011 | 385 faimdprintf(sess, 0, "unknown capability: {%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x}\n", |
386 cap[0], cap[1], cap[2], cap[3], | |
387 cap[4], cap[5], | |
388 cap[6], cap[7], | |
389 cap[8], cap[9], | |
390 cap[10], cap[11], cap[12], cap[13], | |
391 cap[14], cap[15]); | |
7334 | 392 |
393 free(cap); | |
394 } | |
395 | |
396 return flags; | |
397 } | |
398 | |
399 faim_internal fu32_t aim_locate_getcaps_short(aim_session_t *sess, aim_bstream_t *bs, int len) | |
400 { | |
401 fu32_t flags = 0; | |
402 int offset; | |
403 | |
404 for (offset = 0; aim_bstream_empty(bs) && (offset < len); offset += 0x02) { | |
405 fu8_t *cap; | |
406 int i, identified; | |
407 | |
408 cap = aimbs_getraw(bs, 0x02); | |
409 | |
410 for (i = 0, identified = 0; !(aim_caps[i].flag & AIM_CAPS_LAST); i++) { | |
411 if (memcmp(&aim_caps[i].data[2], cap, 0x02) == 0) { | |
412 flags |= aim_caps[i].flag; | |
413 identified++; | |
414 break; /* should only match once... */ | |
415 } | |
7011 | 416 } |
417 | |
7334 | 418 if (!identified) |
419 faimdprintf(sess, 0, "unknown short capability: {%02x%02x}\n", cap[0], cap[1]); | |
420 | |
7011 | 421 free(cap); |
422 } | |
423 | |
424 return flags; | |
425 } | |
426 | |
427 faim_internal int aim_putcap(aim_bstream_t *bs, fu32_t caps) | |
428 { | |
429 int i; | |
430 | |
431 if (!bs) | |
432 return -EINVAL; | |
433 | |
434 for (i = 0; aim_bstream_empty(bs); i++) { | |
435 | |
436 if (aim_caps[i].flag == AIM_CAPS_LAST) | |
437 break; | |
438 | |
439 if (caps & aim_caps[i].flag) | |
440 aimbs_putraw(bs, aim_caps[i].data, 0x10); | |
441 | |
442 } | |
443 | |
444 return 0; | |
445 } | |
446 | |
447 static void dumptlv(aim_session_t *sess, fu16_t type, aim_bstream_t *bs, fu8_t len) | |
448 { | |
449 int i; | |
450 | |
451 if (!sess || !bs || !len) | |
452 return; | |
453 | |
454 faimdprintf(sess, 0, "userinfo: type =0x%04x\n", type); | |
455 faimdprintf(sess, 0, "userinfo: length=0x%04x\n", len); | |
456 faimdprintf(sess, 0, "userinfo: value:\n"); | |
457 | |
458 for (i = 0; i < len; i++) { | |
459 if ((i % 8) == 0) | |
460 faimdprintf(sess, 0, "\nuserinfo: "); | |
461 faimdprintf(sess, 0, "0x%2x ", aimbs_get8(bs)); | |
462 } | |
463 | |
464 faimdprintf(sess, 0, "\n"); | |
465 | |
466 return; | |
467 } | |
468 | |
469 faim_internal void aim_info_free(aim_userinfo_t *info) | |
470 { | |
471 free(info->sn); | |
472 free(info->iconcsum); | |
473 free(info->info); | |
474 free(info->info_encoding); | |
475 free(info->avail); | |
476 free(info->avail_encoding); | |
477 free(info->away); | |
478 free(info->away_encoding); | |
479 } | |
480 | |
481 /* | |
482 * AIM is fairly regular about providing user info. This is a generic | |
483 * routine to extract it in its standard form. | |
484 */ | |
485 faim_internal int aim_info_extract(aim_session_t *sess, aim_bstream_t *bs, aim_userinfo_t *outinfo) | |
486 { | |
487 int curtlv, tlvcnt; | |
488 fu8_t snlen; | |
489 | |
490 if (!bs || !outinfo) | |
491 return -EINVAL; | |
492 | |
493 /* Clear out old data first */ | |
494 memset(outinfo, 0x00, sizeof(aim_userinfo_t)); | |
495 | |
496 /* | |
497 * Screen name. Stored as an unterminated string prepended with a | |
498 * byte containing its length. | |
499 */ | |
500 snlen = aimbs_get8(bs); | |
501 outinfo->sn = aimbs_getstr(bs, snlen); | |
502 | |
503 /* | |
504 * Warning Level. Stored as an unsigned short. | |
505 */ | |
506 outinfo->warnlevel = aimbs_get16(bs); | |
507 | |
508 /* | |
509 * TLV Count. Unsigned short representing the number of | |
510 * Type-Length-Value triples that follow. | |
511 */ | |
512 tlvcnt = aimbs_get16(bs); | |
513 | |
514 /* | |
515 * Parse out the Type-Length-Value triples as they're found. | |
516 */ | |
517 for (curtlv = 0; curtlv < tlvcnt; curtlv++) { | |
518 int endpos; | |
519 fu16_t type, length; | |
520 | |
521 type = aimbs_get16(bs); | |
522 length = aimbs_get16(bs); | |
523 | |
524 endpos = aim_bstream_curpos(bs) + length; | |
525 | |
526 if (type == 0x0001) { | |
527 /* | |
528 * Type = 0x0001: User flags | |
529 * | |
530 * Specified as any of the following ORed together: | |
531 * 0x0001 Trial (user less than 60days) | |
532 * 0x0002 Unknown bit 2 | |
533 * 0x0004 AOL Main Service user | |
534 * 0x0008 Unknown bit 4 | |
535 * 0x0010 Free (AIM) user | |
536 * 0x0020 Away | |
537 * 0x0400 ActiveBuddy | |
538 * | |
539 */ | |
540 outinfo->flags = aimbs_get16(bs); | |
541 outinfo->present |= AIM_USERINFO_PRESENT_FLAGS; | |
542 | |
543 } else if (type == 0x0002) { | |
544 /* | |
545 * Type = 0x0002: Account creation time. | |
546 * | |
547 * The time/date that the user originally registered for | |
548 * the service, stored in time_t format. | |
549 * | |
550 * I'm not sure how this differs from type 5 ("member | |
551 * since"). | |
552 * | |
553 * Note: This is the field formerly known as "member | |
554 * since". All these years and I finally found out | |
555 * that I got the name wrong. | |
556 */ | |
557 outinfo->createtime = aimbs_get32(bs); | |
558 outinfo->present |= AIM_USERINFO_PRESENT_CREATETIME; | |
559 | |
560 } else if (type == 0x0003) { | |
561 /* | |
562 * Type = 0x0003: On-Since date. | |
563 * | |
564 * The time/date that the user started their current | |
565 * session, stored in time_t format. | |
566 */ | |
567 outinfo->onlinesince = aimbs_get32(bs); | |
568 outinfo->present |= AIM_USERINFO_PRESENT_ONLINESINCE; | |
569 | |
570 } else if (type == 0x0004) { | |
571 /* | |
572 * Type = 0x0004: Idle time. | |
573 * | |
574 * Number of minutes since the user actively used the | |
575 * service. | |
576 * | |
577 * Note that the client tells the server when to start | |
578 * counting idle times, so this may or may not be | |
579 * related to reality. | |
580 */ | |
581 outinfo->idletime = aimbs_get16(bs); | |
582 outinfo->present |= AIM_USERINFO_PRESENT_IDLE; | |
583 | |
584 } else if (type == 0x0005) { | |
585 /* | |
586 * Type = 0x0005: Member since date. | |
587 * | |
588 * The time/date that the user originally registered for | |
589 * the service, stored in time_t format. | |
590 * | |
591 * This is sometimes sent instead of type 2 ("account | |
592 * creation time"), particularly in the self-info. | |
593 * And particularly for ICQ? | |
594 */ | |
595 outinfo->membersince = aimbs_get32(bs); | |
596 outinfo->present |= AIM_USERINFO_PRESENT_MEMBERSINCE; | |
597 | |
598 } else if (type == 0x0006) { | |
599 /* | |
600 * Type = 0x0006: ICQ Online Status | |
601 * | |
602 * ICQ's Away/DND/etc "enriched" status. Some decoding | |
603 * of values done by Scott <darkagl@pcnet.com> | |
604 */ | |
605 aimbs_get16(bs); | |
606 outinfo->icqinfo.status = aimbs_get16(bs); | |
607 outinfo->present |= AIM_USERINFO_PRESENT_ICQEXTSTATUS; | |
608 | |
7141 | 609 } else if (type == 0x0008) { |
610 /* | |
611 * Type = 0x0008 | |
612 * | |
613 * Client type, or some such. | |
614 */ | |
615 | |
7011 | 616 } else if (type == 0x000a) { |
617 /* | |
618 * Type = 0x000a | |
619 * | |
620 * ICQ User IP Address. | |
621 * Ahh, the joy of ICQ security. | |
622 */ | |
623 outinfo->icqinfo.ipaddr = aimbs_get32(bs); | |
624 outinfo->present |= AIM_USERINFO_PRESENT_ICQIPADDR; | |
625 | |
626 } else if (type == 0x000c) { | |
627 /* | |
628 * Type = 0x000c | |
629 * | |
630 * random crap containing the IP address, | |
631 * apparently a port number, and some Other Stuff. | |
632 * | |
633 * Format is: | |
634 * 4 bytes - Our IP address, 0xc0 a8 01 2b for 192.168.1.43 | |
635 * | |
636 * | |
637 */ | |
638 aimbs_getrawbuf(bs, outinfo->icqinfo.crap, 0x25); | |
639 outinfo->present |= AIM_USERINFO_PRESENT_ICQDATA; | |
640 | |
641 } else if (type == 0x000d) { | |
642 /* | |
643 * Type = 0x000d | |
644 * | |
7141 | 645 * OSCAR Capability information. |
7011 | 646 * |
647 */ | |
7334 | 648 outinfo->capabilities |= aim_locate_getcaps(sess, bs, length); |
7011 | 649 outinfo->present |= AIM_USERINFO_PRESENT_CAPABILITIES; |
650 | |
651 } else if (type == 0x000e) { | |
652 /* | |
653 * Type = 0x000e | |
654 * | |
7141 | 655 * AOL capability information. |
7011 | 656 * |
657 */ | |
658 | |
659 } else if ((type == 0x000f) || (type == 0x0010)) { | |
660 /* | |
661 * Type = 0x000f: Session Length. (AIM) | |
662 * Type = 0x0010: Session Length. (AOL) | |
663 * | |
664 * The duration, in seconds, of the user's current | |
665 * session. | |
666 * | |
667 * Which TLV type this comes in depends on the | |
668 * service the user is using (AIM or AOL). | |
669 * | |
670 */ | |
671 outinfo->sessionlen = aimbs_get32(bs); | |
672 outinfo->present |= AIM_USERINFO_PRESENT_SESSIONLEN; | |
673 | |
674 } else if (type == 0x0019) { | |
7141 | 675 /* |
676 * Type = 0x0019 | |
677 * | |
678 * OSCAR short capability information. A shortened | |
679 * form of the normal capabilities. | |
680 */ | |
7334 | 681 outinfo->capabilities |= aim_locate_getcaps_short(sess, bs, length); |
682 outinfo->present |= AIM_USERINFO_PRESENT_CAPABILITIES; | |
7011 | 683 |
684 } else if (type == 0x001b) { | |
7141 | 685 /* |
686 * Type = 0x001a | |
687 * | |
688 * AOL short capability information. A shortened | |
689 * form of the normal capabilities. | |
690 */ | |
691 | |
692 } else if (type == 0x001b) { | |
693 /* | |
694 * Type = 0x0019 | |
695 * | |
696 * Encryption certification MD5 checksum. | |
697 */ | |
7011 | 698 |
699 } else if (type == 0x001d) { | |
700 /* | |
701 * Type = 0x001d | |
702 * | |
703 * Buddy icon information and available messages. | |
704 * | |
705 * This almost seems like the AIM protocol guys gave | |
706 * the iChat guys a Type, and the iChat guys tried to | |
707 * cram as much cool shit into it as possible. Then | |
708 * the Windows AIM guys were like, "hey, that's | |
709 * pretty neat, let's copy those prawns." | |
710 * | |
711 * In that spirit, this can contain a custom message, | |
712 * kind of like an away message, but you're not away | |
713 * (it's called an "available" message). Or it can | |
714 * contain information about the buddy icon the user | |
715 * has stored on the server. | |
716 */ | |
717 int type2, number, length2; | |
718 | |
719 while (aim_bstream_curpos(bs) < endpos) { | |
720 type2 = aimbs_get16(bs); | |
721 number = aimbs_get8(bs); | |
722 length2 = aimbs_get8(bs); | |
723 | |
724 switch (type2) { | |
725 case 0x0000: { /* This is an official buddy icon? */ | |
726 /* This is always 5 bytes of "0x02 01 d2 04 72"? */ | |
727 aim_bstream_advance(bs, length2); | |
728 } break; | |
729 | |
730 case 0x0001: { /* A buddy icon checksum */ | |
731 if ((length2 > 0) && (number == 0x01)) { | |
732 free(outinfo->iconcsum); | |
733 outinfo->iconcsum = aimbs_getraw(bs, length2); | |
734 outinfo->iconcsumlen = length2; | |
735 } else | |
736 aim_bstream_advance(bs, length2); | |
737 } break; | |
738 | |
739 case 0x0002: { /* An available message */ | |
740 if (length2 > 4) { | |
741 free(outinfo->avail); | |
742 outinfo->avail_len = aimbs_get16(bs); | |
743 outinfo->avail = aimbs_getstr(bs, outinfo->avail_len); | |
744 if (aimbs_get16(bs) == 0x0001) { /* We have an encoding */ | |
745 aimbs_get16(bs); | |
746 outinfo->avail_encoding = aimbs_getstr(bs, aimbs_get16(bs)); | |
747 } else { | |
748 /* No explicit encoding, client should use UTF-8 */ | |
749 outinfo->avail_encoding = NULL; | |
750 } | |
751 } else | |
752 aim_bstream_advance(bs, length2); | |
753 } break; | |
754 | |
755 default: { | |
756 aim_bstream_advance(bs, length2); | |
757 } break; | |
758 } | |
759 } | |
760 | |
761 } else if (type == 0x001e) { | |
762 /* | |
763 * Type 30: Unknown. | |
764 * | |
765 * Always four bytes, but it doesn't look like an int. | |
766 */ | |
7025 | 767 |
768 } else if (type == 0x001f) { | |
769 /* | |
770 * Type 31: Unknown. | |
771 * | |
772 * Seen on a buddy using DeadAIM. Data was 4 bytes: | |
773 * 0x00 00 00 10 | |
774 */ | |
775 | |
7011 | 776 } else { |
777 | |
778 /* | |
779 * Reaching here indicates that either AOL has | |
780 * added yet another TLV for us to deal with, | |
781 * or the parsing has gone Terribly Wrong. | |
782 * | |
783 * Either way, inform the owner and attempt | |
784 * recovery. | |
785 * | |
786 */ | |
787 faimdprintf(sess, 0, "userinfo: **warning: unexpected TLV:\n"); | |
788 faimdprintf(sess, 0, "userinfo: sn =%s\n", outinfo->sn); | |
789 dumptlv(sess, type, bs, length); | |
790 } | |
791 | |
792 /* Save ourselves. */ | |
793 aim_bstream_setpos(bs, endpos); | |
794 } | |
795 | |
7045 | 796 aim_locate_adduserinfo(sess, outinfo); |
7011 | 797 |
798 return 0; | |
799 } | |
800 | |
801 /* | |
802 * Inverse of aim_info_extract() | |
803 */ | |
804 faim_internal int aim_putuserinfo(aim_bstream_t *bs, aim_userinfo_t *info) | |
805 { | |
806 aim_tlvlist_t *tlvlist = NULL; | |
807 | |
808 if (!bs || !info) | |
809 return -EINVAL; | |
810 | |
811 aimbs_put8(bs, strlen(info->sn)); | |
812 aimbs_putraw(bs, info->sn, strlen(info->sn)); | |
813 | |
814 aimbs_put16(bs, info->warnlevel); | |
815 | |
816 if (info->present & AIM_USERINFO_PRESENT_FLAGS) | |
7167 | 817 aim_tlvlist_add_16(&tlvlist, 0x0001, info->flags); |
7011 | 818 if (info->present & AIM_USERINFO_PRESENT_MEMBERSINCE) |
7167 | 819 aim_tlvlist_add_32(&tlvlist, 0x0002, info->membersince); |
7011 | 820 if (info->present & AIM_USERINFO_PRESENT_ONLINESINCE) |
7167 | 821 aim_tlvlist_add_32(&tlvlist, 0x0003, info->onlinesince); |
7011 | 822 if (info->present & AIM_USERINFO_PRESENT_IDLE) |
7167 | 823 aim_tlvlist_add_16(&tlvlist, 0x0004, info->idletime); |
7011 | 824 |
825 /* XXX - So, ICQ_OSCAR_SUPPORT is never defined anywhere... */ | |
826 #if ICQ_OSCAR_SUPPORT | |
827 if (atoi(info->sn) != 0) { | |
828 if (info->present & AIM_USERINFO_PRESENT_ICQEXTSTATUS) | |
7167 | 829 aim_tlvlist_add_16(&tlvlist, 0x0006, info->icqinfo.status); |
7011 | 830 if (info->present & AIM_USERINFO_PRESENT_ICQIPADDR) |
7167 | 831 aim_tlvlist_add_32(&tlvlist, 0x000a, info->icqinfo.ipaddr); |
7011 | 832 } |
833 #endif | |
834 | |
835 if (info->present & AIM_USERINFO_PRESENT_CAPABILITIES) | |
7167 | 836 aim_tlvlist_add_caps(&tlvlist, 0x000d, info->capabilities); |
7011 | 837 |
838 if (info->present & AIM_USERINFO_PRESENT_SESSIONLEN) | |
7167 | 839 aim_tlvlist_add_32(&tlvlist, (fu16_t)((info->flags & AIM_FLAG_AOL) ? 0x0010 : 0x000f), info->sessionlen); |
7011 | 840 |
7167 | 841 aimbs_put16(bs, aim_tlvlist_count(&tlvlist)); |
842 aim_tlvlist_write(bs, &tlvlist); | |
843 aim_tlvlist_free(&tlvlist); | |
7011 | 844 |
845 return 0; | |
846 } | |
847 | |
848 /* | |
7259 | 849 * Subtype 0x0001 |
850 */ | |
851 static int error(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs) | |
852 { | |
853 int ret = 0; | |
854 aim_rxcallback_t userfunc; | |
855 aim_snac_t *snac2; | |
856 fu16_t reason; | |
857 char *sn; | |
858 int was_explicit; | |
859 | |
860 if (!(snac2 = aim_remsnac(sess, snac->id))) { | |
861 faimdprintf(sess, 0, "faim: locate.c, error(): received response from unknown request!\n"); | |
862 return 0; | |
863 } | |
864 | |
865 if ((snac2->family != 0x0002) && (snac2->type != 0x0015)) { | |
866 faimdprintf(sess, 0, "faim: locate.c, error(): received response from invalid request! %d\n", snac2->family); | |
867 return 0; | |
868 } | |
869 | |
870 if (!(sn = snac2->data)) { | |
871 faimdprintf(sess, 0, "faim: locate.c, error(): received response from request without a screen name!\n"); | |
872 return 0; | |
873 } | |
874 | |
875 reason = aimbs_get16(bs); | |
876 | |
877 /* | |
878 * Remove this screen name from our queue. If the client requested | |
879 * this buddy's info explicitly, then notify them that we do not have | |
880 * info for this buddy. | |
881 */ | |
882 was_explicit = aim_locate_gotuserinfo(sess, sn); | |
883 if (was_explicit == TRUE) | |
884 if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype))) | |
885 ret = userfunc(sess, rx, reason, sn); | |
886 | |
887 if (snac2) | |
888 free(snac2->data); | |
889 free(snac2); | |
890 | |
891 return ret; | |
892 } | |
893 | |
894 /* | |
7011 | 895 * Subtype 0x0002 |
896 * | |
897 * Request Location services rights. | |
898 * | |
899 */ | |
900 faim_export int aim_locate_reqrights(aim_session_t *sess) | |
901 { | |
902 aim_conn_t *conn; | |
903 | |
904 if (!sess || !(conn = aim_conn_findbygroup(sess, AIM_CB_FAM_LOC))) | |
905 return -EINVAL; | |
906 | |
7282 | 907 return aim_genericreq_n_snacid(sess, conn, AIM_CB_FAM_LOC, AIM_CB_LOC_REQRIGHTS); |
7011 | 908 } |
909 | |
910 /* | |
911 * Subtype 0x0003 | |
912 * | |
913 * Normally contains: | |
914 * t(0001) - short containing max profile length (value = 1024) | |
915 * t(0002) - short - unknown (value = 16) [max MIME type length?] | |
916 * t(0003) - short - unknown (value = 10) | |
917 * t(0004) - short - unknown (value = 2048) [ICQ only?] | |
918 */ | |
919 static int rights(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs) | |
920 { | |
921 aim_tlvlist_t *tlvlist; | |
922 aim_rxcallback_t userfunc; | |
923 int ret = 0; | |
924 fu16_t maxsiglen = 0; | |
925 | |
7167 | 926 tlvlist = aim_tlvlist_read(bs); |
7011 | 927 |
7167 | 928 if (aim_tlv_gettlv(tlvlist, 0x0001, 1)) |
929 maxsiglen = aim_tlv_get16(tlvlist, 0x0001, 1); | |
7011 | 930 |
931 if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype))) | |
932 ret = userfunc(sess, rx, maxsiglen); | |
933 | |
7167 | 934 aim_tlvlist_free(&tlvlist); |
7011 | 935 |
936 return ret; | |
937 } | |
938 | |
939 /* | |
940 * Subtype 0x0004 | |
941 * | |
942 * Gives BOS your profile. | |
943 * | |
944 * profile_encoding and awaymsg_encoding MUST be set if profile or | |
945 * away are set, respectively, and their value may or may not be | |
946 * restricted to a few choices. I am currently aware of: | |
947 * | |
948 * us-ascii Just that | |
949 * unicode-2-0 UCS2-BE | |
950 * | |
951 * profile_len and awaymsg_len MUST be set similarly, and they MUST | |
952 * be the length of their respective strings in bytes. | |
953 * | |
954 * To get the previous behavior of awaymsg == "" un-setting the away | |
955 * message, set awaymsg non-NULL and awaymsg_len to 0 (this is the | |
956 * obvious equivalent). | |
957 * | |
958 */ | |
959 faim_export int aim_locate_setprofile(aim_session_t *sess, | |
960 const char *profile_encoding, const char *profile, const int profile_len, | |
7334 | 961 const char *awaymsg_encoding, const char *awaymsg, const int awaymsg_len) |
7011 | 962 { |
963 aim_conn_t *conn; | |
964 aim_frame_t *fr; | |
965 aim_snacid_t snacid; | |
966 aim_tlvlist_t *tl = NULL; | |
967 char *encoding; | |
968 static const char defencoding[] = {"text/aolrtf; charset=\"%s\""}; | |
969 | |
970 if (!sess || !(conn = aim_conn_findbygroup(sess, AIM_CB_FAM_LOC))) | |
971 return -EINVAL; | |
972 | |
7510 | 973 if (!profile && !awaymsg) |
974 return -EINVAL; | |
975 | |
7011 | 976 if ((profile && profile_encoding == NULL) || (awaymsg && awaymsg_len && awaymsg_encoding == NULL)) { |
977 return -EINVAL; | |
978 } | |
979 | |
980 /* Build the packet first to get real length */ | |
981 if (profile) { | |
982 /* no + 1 here because of %s */ | |
983 encoding = malloc(strlen(defencoding) + strlen(profile_encoding)); | |
984 if (encoding == NULL) { | |
985 return -ENOMEM; | |
986 } | |
987 snprintf(encoding, strlen(defencoding) + strlen(profile_encoding), defencoding, profile_encoding); | |
7167 | 988 aim_tlvlist_add_raw(&tl, 0x0001, strlen(encoding), encoding); |
989 aim_tlvlist_add_raw(&tl, 0x0002, profile_len, profile); | |
7011 | 990 free(encoding); |
991 } | |
992 | |
993 /* | |
994 * So here's how this works: | |
995 * - You are away when you have a non-zero-length type 4 TLV stored. | |
996 * - You become unaway when you clear the TLV with a zero-length | |
997 * type 4 TLV. | |
998 * - If you do not send the type 4 TLV, your status does not change | |
999 * (that is, if you were away, you'll remain away). | |
1000 */ | |
1001 if (awaymsg) { | |
1002 if (awaymsg_len) { | |
1003 encoding = malloc(strlen(defencoding) + strlen(awaymsg_encoding)); | |
1004 if (encoding == NULL) { | |
1005 return -ENOMEM; | |
1006 } | |
1007 snprintf(encoding, strlen(defencoding) + strlen(awaymsg_encoding), defencoding, awaymsg_encoding); | |
7167 | 1008 aim_tlvlist_add_raw(&tl, 0x0003, strlen(encoding), encoding); |
1009 aim_tlvlist_add_raw(&tl, 0x0004, awaymsg_len, awaymsg); | |
7011 | 1010 free(encoding); |
1011 } else | |
7167 | 1012 aim_tlvlist_add_noval(&tl, 0x0004); |
7011 | 1013 } |
1014 | |
7334 | 1015 if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 10 + aim_tlvlist_size(&tl)))) |
1016 return -ENOMEM; | |
1017 | |
1018 snacid = aim_cachesnac(sess, 0x0002, 0x0004, 0x0000, NULL, 0); | |
1019 aim_putsnac(&fr->data, 0x0002, 0x004, 0x0000, snacid); | |
1020 | |
1021 aim_tlvlist_write(&fr->data, &tl); | |
1022 aim_tlvlist_free(&tl); | |
1023 | |
1024 aim_tx_enqueue(sess, fr); | |
1025 | |
1026 return 0; | |
1027 } | |
1028 | |
1029 /* | |
1030 * Subtype 0x0004 - Set your client's capabilities. | |
1031 */ | |
1032 faim_export int aim_locate_setcaps(aim_session_t *sess, fu32_t caps) | |
1033 { | |
1034 aim_conn_t *conn; | |
1035 aim_frame_t *fr; | |
1036 aim_snacid_t snacid; | |
1037 aim_tlvlist_t *tl = NULL; | |
1038 | |
1039 if (!sess || !(conn = aim_conn_findbygroup(sess, AIM_CB_FAM_LOC))) | |
1040 return -EINVAL; | |
1041 | |
1042 aim_tlvlist_add_caps(&tl, 0x0005, caps); | |
7011 | 1043 |
7167 | 1044 if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 10 + aim_tlvlist_size(&tl)))) |
7011 | 1045 return -ENOMEM; |
1046 | |
1047 snacid = aim_cachesnac(sess, 0x0002, 0x0004, 0x0000, NULL, 0); | |
1048 aim_putsnac(&fr->data, 0x0002, 0x004, 0x0000, snacid); | |
1049 | |
7167 | 1050 aim_tlvlist_write(&fr->data, &tl); |
1051 aim_tlvlist_free(&tl); | |
7011 | 1052 |
1053 aim_tx_enqueue(sess, fr); | |
1054 | |
1055 return 0; | |
1056 } | |
1057 | |
1058 /* | |
1059 * Subtype 0x0005 - Request info of another AIM user. | |
1060 * | |
1061 * @param sn The screenname whose info you wish to request. | |
1062 * @param infotype The type of info you wish to request. | |
1063 * 0x0001 - Info/profile | |
1064 * 0x0003 - Away message | |
1065 * 0x0004 - Capabilities | |
1066 */ | |
1067 faim_export int aim_locate_getinfo(aim_session_t *sess, const char *sn, fu16_t infotype) | |
1068 { | |
1069 aim_conn_t *conn; | |
1070 aim_frame_t *fr; | |
1071 aim_snacid_t snacid; | |
1072 | |
1073 if (!sess || !(conn = aim_conn_findbygroup(sess, AIM_CB_FAM_LOC)) || !sn) | |
1074 return -EINVAL; | |
1075 | |
1076 if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 12+1+strlen(sn)))) | |
1077 return -ENOMEM; | |
1078 | |
1079 snacid = aim_cachesnac(sess, 0x0002, 0x0005, 0x0000, NULL, 0); | |
1080 | |
1081 aim_putsnac(&fr->data, 0x0002, 0x0005, 0x0000, snacid); | |
1082 aimbs_put16(&fr->data, infotype); | |
1083 aimbs_put8(&fr->data, strlen(sn)); | |
1084 aimbs_putraw(&fr->data, sn, strlen(sn)); | |
1085 | |
1086 aim_tx_enqueue(sess, fr); | |
1087 | |
1088 return 0; | |
1089 } | |
1090 | |
1091 /* Subtype 0x0006 */ | |
1092 static int userinfo(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs) | |
1093 { | |
1094 int ret = 0; | |
1095 aim_rxcallback_t userfunc; | |
1096 aim_userinfo_t *userinfo, *userinfo2; | |
1097 aim_tlvlist_t *tlvlist; | |
1098 aim_tlv_t *tlv = NULL; | |
1099 int was_explicit; | |
1100 | |
1101 userinfo = (aim_userinfo_t *)malloc(sizeof(aim_userinfo_t)); | |
1102 aim_info_extract(sess, bs, userinfo); | |
7167 | 1103 tlvlist = aim_tlvlist_read(bs); |
7011 | 1104 |
1105 /* Profile will be 1 and 2 */ | |
7167 | 1106 userinfo->info_encoding = aim_tlv_getstr(tlvlist, 0x0001, 1); |
1107 if ((tlv = aim_tlv_gettlv(tlvlist, 0x0002, 1))) { | |
7011 | 1108 userinfo->info = (char *)malloc(tlv->length); |
1109 memcpy(userinfo->info, tlv->value, tlv->length); | |
1110 userinfo->info_len = tlv->length; | |
1111 } | |
1112 | |
1113 /* Away message will be 3 and 4 */ | |
7167 | 1114 userinfo->away_encoding = aim_tlv_getstr(tlvlist, 0x0003, 1); |
1115 if ((tlv = aim_tlv_gettlv(tlvlist, 0x0004, 1))) { | |
7011 | 1116 userinfo->away = (char *)malloc(tlv->length); |
1117 memcpy(userinfo->away, tlv->value, tlv->length); | |
1118 userinfo->away_len = tlv->length; | |
1119 } | |
1120 | |
1121 /* Caps will be 5 */ | |
7167 | 1122 if ((tlv = aim_tlv_gettlv(tlvlist, 0x0005, 1))) { |
7011 | 1123 aim_bstream_t cbs; |
1124 aim_bstream_init(&cbs, tlv->value, tlv->length); | |
7334 | 1125 userinfo->capabilities = aim_locate_getcaps(sess, &cbs, tlv->length); |
7011 | 1126 userinfo->present = AIM_USERINFO_PRESENT_CAPABILITIES; |
1127 } | |
7167 | 1128 aim_tlvlist_free(&tlvlist); |
7011 | 1129 |
7045 | 1130 aim_locate_adduserinfo(sess, userinfo); |
1131 userinfo2 = aim_locate_finduserinfo(sess, userinfo->sn); | |
7011 | 1132 aim_info_free(userinfo); |
1133 free(userinfo); | |
1134 | |
1135 /* | |
1136 * Remove this screen name from our queue. If the client requested | |
1137 * this buddy's info explicitly, then notify them that we have info | |
1138 * for this buddy. | |
1139 */ | |
7259 | 1140 was_explicit = aim_locate_gotuserinfo(sess, userinfo2->sn); |
1141 if (was_explicit == TRUE) | |
7011 | 1142 if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype))) |
1143 ret = userfunc(sess, rx, userinfo2); | |
1144 | |
1145 return ret; | |
1146 } | |
1147 | |
1148 /* | |
1149 * Subtype 0x0009 - Set directory profile data. | |
1150 * | |
1151 * This is not the same as aim_location_setprofile! | |
1152 * privacy: 1 to allow searching, 0 to disallow. | |
1153 * | |
1154 */ | |
1155 faim_export int aim_locate_setdirinfo(aim_session_t *sess, const char *first, const char *middle, const char *last, const char *maiden, const char *nickname, const char *street, const char *city, const char *state, const char *zip, int country, fu16_t privacy) | |
1156 { | |
1157 aim_conn_t *conn; | |
1158 aim_frame_t *fr; | |
1159 aim_snacid_t snacid; | |
1160 aim_tlvlist_t *tl = NULL; | |
1161 | |
1162 if (!sess || !(conn = aim_conn_findbygroup(sess, AIM_CB_FAM_LOC))) | |
1163 return -EINVAL; | |
1164 | |
7167 | 1165 aim_tlvlist_add_16(&tl, 0x000a, privacy); |
7011 | 1166 |
1167 if (first) | |
7167 | 1168 aim_tlvlist_add_raw(&tl, 0x0001, strlen(first), first); |
7011 | 1169 if (last) |
7167 | 1170 aim_tlvlist_add_raw(&tl, 0x0002, strlen(last), last); |
7011 | 1171 if (middle) |
7167 | 1172 aim_tlvlist_add_raw(&tl, 0x0003, strlen(middle), middle); |
7011 | 1173 if (maiden) |
7167 | 1174 aim_tlvlist_add_raw(&tl, 0x0004, strlen(maiden), maiden); |
7011 | 1175 |
1176 if (state) | |
7167 | 1177 aim_tlvlist_add_raw(&tl, 0x0007, strlen(state), state); |
7011 | 1178 if (city) |
7167 | 1179 aim_tlvlist_add_raw(&tl, 0x0008, strlen(city), city); |
7011 | 1180 |
1181 if (nickname) | |
7167 | 1182 aim_tlvlist_add_raw(&tl, 0x000c, strlen(nickname), nickname); |
7011 | 1183 if (zip) |
7167 | 1184 aim_tlvlist_add_raw(&tl, 0x000d, strlen(zip), zip); |
7011 | 1185 |
1186 if (street) | |
7167 | 1187 aim_tlvlist_add_raw(&tl, 0x0021, strlen(street), street); |
7011 | 1188 |
7167 | 1189 if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 10+aim_tlvlist_size(&tl)))) |
7011 | 1190 return -ENOMEM; |
1191 | |
1192 snacid = aim_cachesnac(sess, 0x0002, 0x0009, 0x0000, NULL, 0); | |
1193 | |
1194 aim_putsnac(&fr->data, 0x0002, 0x0009, 0x0000, snacid); | |
7167 | 1195 aim_tlvlist_write(&fr->data, &tl); |
1196 aim_tlvlist_free(&tl); | |
7011 | 1197 |
1198 aim_tx_enqueue(sess, fr); | |
1199 | |
1200 return 0; | |
1201 } | |
1202 | |
1203 /* | |
1204 * Subtype 0x000b - Huh? What is this? | |
1205 */ | |
1206 faim_export int aim_locate_000b(aim_session_t *sess, const char *sn) | |
1207 { | |
1208 aim_conn_t *conn; | |
1209 aim_frame_t *fr; | |
1210 aim_snacid_t snacid; | |
1211 | |
1212 return -EINVAL; | |
1213 | |
1214 if (!sess || !(conn = aim_conn_findbygroup(sess, AIM_CB_FAM_LOC)) || !sn) | |
1215 return -EINVAL; | |
1216 | |
1217 if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 10+1+strlen(sn)))) | |
1218 return -ENOMEM; | |
1219 | |
1220 snacid = aim_cachesnac(sess, 0x0002, 0x000b, 0x0000, NULL, 0); | |
1221 | |
1222 aim_putsnac(&fr->data, 0x0002, 0x000b, 0x0000, snacid); | |
1223 aimbs_put8(&fr->data, strlen(sn)); | |
1224 aimbs_putraw(&fr->data, sn, strlen(sn)); | |
1225 | |
1226 aim_tx_enqueue(sess, fr); | |
1227 | |
1228 return 0; | |
1229 } | |
1230 | |
1231 /* | |
1232 * Subtype 0x000f | |
1233 * | |
1234 * XXX pass these in better | |
1235 * | |
1236 */ | |
1237 faim_export int aim_locate_setinterests(aim_session_t *sess, const char *interest1, const char *interest2, const char *interest3, const char *interest4, const char *interest5, fu16_t privacy) | |
1238 { | |
1239 aim_conn_t *conn; | |
1240 aim_frame_t *fr; | |
1241 aim_snacid_t snacid; | |
1242 aim_tlvlist_t *tl = NULL; | |
1243 | |
1244 if (!sess || !(conn = aim_conn_findbygroup(sess, AIM_CB_FAM_LOC))) | |
1245 return -EINVAL; | |
1246 | |
1247 /* ?? privacy ?? */ | |
7167 | 1248 aim_tlvlist_add_16(&tl, 0x000a, privacy); |
7011 | 1249 |
1250 if (interest1) | |
7167 | 1251 aim_tlvlist_add_raw(&tl, 0x0000b, strlen(interest1), interest1); |
7011 | 1252 if (interest2) |
7167 | 1253 aim_tlvlist_add_raw(&tl, 0x0000b, strlen(interest2), interest2); |
7011 | 1254 if (interest3) |
7167 | 1255 aim_tlvlist_add_raw(&tl, 0x0000b, strlen(interest3), interest3); |
7011 | 1256 if (interest4) |
7167 | 1257 aim_tlvlist_add_raw(&tl, 0x0000b, strlen(interest4), interest4); |
7011 | 1258 if (interest5) |
7167 | 1259 aim_tlvlist_add_raw(&tl, 0x0000b, strlen(interest5), interest5); |
7011 | 1260 |
7167 | 1261 if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 10+aim_tlvlist_size(&tl)))) |
7011 | 1262 return -ENOMEM; |
1263 | |
1264 snacid = aim_cachesnac(sess, 0x0002, 0x000f, 0x0000, NULL, 0); | |
1265 | |
1266 aim_putsnac(&fr->data, 0x0002, 0x000f, 0x0000, 0); | |
7167 | 1267 aim_tlvlist_write(&fr->data, &tl); |
1268 aim_tlvlist_free(&tl); | |
7011 | 1269 |
1270 aim_tx_enqueue(sess, fr); | |
1271 | |
1272 return 0; | |
1273 } | |
1274 | |
1275 /* | |
1276 * Subtype 0x0015 - Request the info a user using the short method. This is | |
1277 * what iChat uses. It normally is VERY leniently rate limited. | |
1278 * | |
1279 * @param sn The screen name whose info you wish to request. | |
1280 * @param flags The bitmask which specifies the type of info you wish to request. | |
1281 * 0x00000001 - Info/profile. | |
1282 * 0x00000002 - Away message. | |
1283 * 0x00000004 - Capabilities. | |
1284 * 0x00000008 - Certification. | |
1285 * @return Return 0 if no errors, otherwise return the error number. | |
1286 */ | |
1287 faim_export int aim_locate_getinfoshort(aim_session_t *sess, const char *sn, fu32_t flags) | |
1288 { | |
1289 aim_conn_t *conn; | |
1290 aim_frame_t *fr; | |
1291 aim_snacid_t snacid; | |
1292 | |
1293 if (!sess || !(conn = aim_conn_findbygroup(sess, AIM_CB_FAM_LOC)) || !sn) | |
1294 return -EINVAL; | |
1295 | |
1296 if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 10+4+1+strlen(sn)))) | |
1297 return -ENOMEM; | |
1298 | |
7259 | 1299 snacid = aim_cachesnac(sess, 0x0002, 0x0015, 0x0000, sn, strlen(sn)+1); |
7011 | 1300 |
7259 | 1301 aim_putsnac(&fr->data, 0x0002, 0x0015, 0x0000, snacid); |
7011 | 1302 aimbs_put32(&fr->data, flags); |
1303 aimbs_put8(&fr->data, strlen(sn)); | |
1304 aimbs_putraw(&fr->data, sn, strlen(sn)); | |
1305 | |
1306 aim_tx_enqueue(sess, fr); | |
1307 | |
1308 return 0; | |
1309 } | |
1310 | |
1311 static int snachandler(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs) | |
1312 { | |
1313 | |
7259 | 1314 if (snac->subtype == 0x0001) |
1315 return error(sess, mod, rx, snac, bs); | |
1316 else if (snac->subtype == 0x0003) | |
7011 | 1317 return rights(sess, mod, rx, snac, bs); |
1318 else if (snac->subtype == 0x0006) | |
1319 return userinfo(sess, mod, rx, snac, bs); | |
1320 | |
1321 return 0; | |
1322 } | |
1323 | |
1324 static void locate_shutdown(aim_session_t *sess, aim_module_t *mod) | |
1325 { | |
1326 aim_userinfo_t *del; | |
1327 | |
7045 | 1328 while (sess->locate.userinfo) { |
1329 del = sess->locate.userinfo; | |
1330 sess->locate.userinfo = sess->locate.userinfo->next; | |
7046 | 1331 aim_info_free(del); |
7011 | 1332 free(del); |
1333 } | |
1334 } | |
1335 | |
1336 faim_internal int locate_modfirst(aim_session_t *sess, aim_module_t *mod) | |
1337 { | |
1338 | |
1339 mod->family = AIM_CB_FAM_LOC; | |
1340 mod->version = 0x0001; | |
1341 mod->toolid = 0x0110; | |
1342 mod->toolversion = 0x0629; | |
1343 mod->flags = 0; | |
1344 strncpy(mod->name, "locate", sizeof(mod->name)); | |
1345 mod->snachandler = snachandler; | |
1346 mod->shutdown = locate_shutdown; | |
1347 | |
1348 return 0; | |
1349 } |