Mercurial > pidgin.yaz
comparison libpurple/protocols/jabber/google/google_session.c @ 30906:34f586bffe4e
Added new files in sub directory google
author | Marcus Lundblad <ml@update.uu.se> |
---|---|
date | Thu, 25 Mar 2010 20:18:54 +0000 |
parents | |
children | 721b257bcd8c |
comparison
equal
deleted
inserted
replaced
30905:ae615b3d3e47 | 30906:34f586bffe4e |
---|---|
1 /** | |
2 * Purple is the legal property of its developers, whose names are too numerous | |
3 * to list here. Please refer to the COPYRIGHT file distributed with this | |
4 * source distribution. | |
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., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA | |
19 */ | |
20 | |
21 #include "internal.h" | |
22 #include "debug.h" | |
23 #include "google_session.h" | |
24 | |
25 #include "jingle/jingle.h" | |
26 | |
27 #ifdef USE_VV | |
28 | |
29 typedef struct { | |
30 char *id; | |
31 char *initiator; | |
32 } GoogleSessionId; | |
33 | |
34 typedef enum { | |
35 UNINIT, | |
36 SENT_INITIATE, | |
37 RECEIVED_INITIATE, | |
38 IN_PRORESS, | |
39 TERMINATED | |
40 } GoogleSessionState; | |
41 | |
42 typedef struct { | |
43 GoogleSessionId id; | |
44 GoogleSessionState state; | |
45 PurpleMedia *media; | |
46 JabberStream *js; | |
47 char *remote_jid; | |
48 gboolean video; | |
49 } GoogleSession; | |
50 | |
51 static gboolean | |
52 google_session_id_equal(gconstpointer a, gconstpointer b) | |
53 { | |
54 GoogleSessionId *c = (GoogleSessionId*)a; | |
55 GoogleSessionId *d = (GoogleSessionId*)b; | |
56 | |
57 return !strcmp(c->id, d->id) && !strcmp(c->initiator, d->initiator); | |
58 } | |
59 | |
60 static void | |
61 google_session_destroy(GoogleSession *session) | |
62 { | |
63 g_free(session->id.id); | |
64 g_free(session->id.initiator); | |
65 g_free(session->remote_jid); | |
66 g_free(session); | |
67 } | |
68 | |
69 static xmlnode * | |
70 google_session_create_xmlnode(GoogleSession *session, const char *type) | |
71 { | |
72 xmlnode *node = xmlnode_new("session"); | |
73 xmlnode_set_namespace(node, NS_GOOGLE_SESSION); | |
74 xmlnode_set_attrib(node, "id", session->id.id); | |
75 xmlnode_set_attrib(node, "initiator", session->id.initiator); | |
76 xmlnode_set_attrib(node, "type", type); | |
77 return node; | |
78 } | |
79 | |
80 static void | |
81 google_session_send_candidates(PurpleMedia *media, gchar *session_id, | |
82 gchar *participant, GoogleSession *session) | |
83 { | |
84 GList *candidates = purple_media_get_local_candidates( | |
85 session->media, session_id, session->remote_jid), *iter; | |
86 PurpleMediaCandidate *transport; | |
87 gboolean video = FALSE; | |
88 | |
89 if (!strcmp(session_id, "google-video")) | |
90 video = TRUE; | |
91 | |
92 for (iter = candidates; iter; iter = iter->next) { | |
93 JabberIq *iq; | |
94 gchar *ip, *port, *username, *password; | |
95 gchar pref[16]; | |
96 PurpleMediaCandidateType type; | |
97 xmlnode *sess; | |
98 xmlnode *candidate; | |
99 guint component_id; | |
100 transport = PURPLE_MEDIA_CANDIDATE(iter->data); | |
101 component_id = purple_media_candidate_get_component_id( | |
102 transport); | |
103 | |
104 iq = jabber_iq_new(session->js, JABBER_IQ_SET); | |
105 sess = google_session_create_xmlnode(session, "candidates"); | |
106 xmlnode_insert_child(iq->node, sess); | |
107 xmlnode_set_attrib(iq->node, "to", session->remote_jid); | |
108 | |
109 candidate = xmlnode_new("candidate"); | |
110 | |
111 ip = purple_media_candidate_get_ip(transport); | |
112 port = g_strdup_printf("%d", | |
113 purple_media_candidate_get_port(transport)); | |
114 g_ascii_dtostr(pref, 16, | |
115 purple_media_candidate_get_priority(transport) / 1000.0); | |
116 username = purple_media_candidate_get_username(transport); | |
117 password = purple_media_candidate_get_password(transport); | |
118 type = purple_media_candidate_get_candidate_type(transport); | |
119 | |
120 xmlnode_set_attrib(candidate, "address", ip); | |
121 xmlnode_set_attrib(candidate, "port", port); | |
122 xmlnode_set_attrib(candidate, "name", | |
123 component_id == PURPLE_MEDIA_COMPONENT_RTP ? | |
124 video ? "video_rtp" : "rtp" : | |
125 component_id == PURPLE_MEDIA_COMPONENT_RTCP ? | |
126 video ? "video_rtcp" : "rtcp" : "none"); | |
127 xmlnode_set_attrib(candidate, "username", username); | |
128 /* | |
129 * As of this writing, Farsight 2 in Google compatibility | |
130 * mode doesn't provide a password. The Gmail client | |
131 * requires this to be set. | |
132 */ | |
133 xmlnode_set_attrib(candidate, "password", | |
134 password != NULL ? password : ""); | |
135 xmlnode_set_attrib(candidate, "preference", pref); | |
136 xmlnode_set_attrib(candidate, "protocol", | |
137 purple_media_candidate_get_protocol(transport) | |
138 == PURPLE_MEDIA_NETWORK_PROTOCOL_UDP ? | |
139 "udp" : "tcp"); | |
140 xmlnode_set_attrib(candidate, "type", type == | |
141 PURPLE_MEDIA_CANDIDATE_TYPE_HOST ? "local" : | |
142 type == | |
143 PURPLE_MEDIA_CANDIDATE_TYPE_SRFLX ? "stun" : | |
144 type == | |
145 PURPLE_MEDIA_CANDIDATE_TYPE_RELAY ? "relay" : | |
146 NULL); | |
147 xmlnode_set_attrib(candidate, "generation", "0"); | |
148 xmlnode_set_attrib(candidate, "network", "0"); | |
149 xmlnode_insert_child(sess, candidate); | |
150 | |
151 g_free(ip); | |
152 g_free(port); | |
153 g_free(username); | |
154 g_free(password); | |
155 | |
156 jabber_iq_send(iq); | |
157 } | |
158 purple_media_candidate_list_free(candidates); | |
159 } | |
160 | |
161 static void | |
162 google_session_ready(GoogleSession *session) | |
163 { | |
164 PurpleMedia *media = session->media; | |
165 if (purple_media_codecs_ready(media, NULL) && | |
166 purple_media_candidates_prepared(media, NULL, NULL)) { | |
167 gchar *me = g_strdup_printf("%s@%s/%s", | |
168 session->js->user->node, | |
169 session->js->user->domain, | |
170 session->js->user->resource); | |
171 JabberIq *iq; | |
172 xmlnode *sess, *desc, *payload; | |
173 GList *codecs, *iter; | |
174 gboolean is_initiator = !strcmp(session->id.initiator, me); | |
175 | |
176 if (!is_initiator && | |
177 !purple_media_accepted(media, NULL, NULL)) { | |
178 g_free(me); | |
179 return; | |
180 } | |
181 | |
182 iq = jabber_iq_new(session->js, JABBER_IQ_SET); | |
183 | |
184 if (is_initiator) { | |
185 xmlnode_set_attrib(iq->node, "to", session->remote_jid); | |
186 xmlnode_set_attrib(iq->node, "from", session->id.initiator); | |
187 sess = google_session_create_xmlnode(session, "initiate"); | |
188 } else { | |
189 google_session_send_candidates(session->media, | |
190 "google-voice", session->remote_jid, | |
191 session); | |
192 google_session_send_candidates(session->media, | |
193 "google-video", session->remote_jid, | |
194 session); | |
195 xmlnode_set_attrib(iq->node, "to", session->remote_jid); | |
196 xmlnode_set_attrib(iq->node, "from", me); | |
197 sess = google_session_create_xmlnode(session, "accept"); | |
198 } | |
199 xmlnode_insert_child(iq->node, sess); | |
200 desc = xmlnode_new_child(sess, "description"); | |
201 if (session->video) | |
202 xmlnode_set_namespace(desc, NS_GOOGLE_SESSION_VIDEO); | |
203 else | |
204 xmlnode_set_namespace(desc, NS_GOOGLE_SESSION_PHONE); | |
205 | |
206 codecs = purple_media_get_codecs(media, "google-video"); | |
207 | |
208 for (iter = codecs; iter; iter = g_list_next(iter)) { | |
209 PurpleMediaCodec *codec = (PurpleMediaCodec*)iter->data; | |
210 gchar *id = g_strdup_printf("%d", | |
211 purple_media_codec_get_id(codec)); | |
212 gchar *encoding_name = | |
213 purple_media_codec_get_encoding_name(codec); | |
214 payload = xmlnode_new_child(desc, "payload-type"); | |
215 xmlnode_set_attrib(payload, "id", id); | |
216 xmlnode_set_attrib(payload, "name", encoding_name); | |
217 xmlnode_set_attrib(payload, "width", "320"); | |
218 xmlnode_set_attrib(payload, "height", "200"); | |
219 xmlnode_set_attrib(payload, "framerate", "30"); | |
220 g_free(encoding_name); | |
221 g_free(id); | |
222 } | |
223 purple_media_codec_list_free(codecs); | |
224 | |
225 codecs = purple_media_get_codecs(media, "google-voice"); | |
226 | |
227 for (iter = codecs; iter; iter = g_list_next(iter)) { | |
228 PurpleMediaCodec *codec = (PurpleMediaCodec*)iter->data; | |
229 gchar *id = g_strdup_printf("%d", | |
230 purple_media_codec_get_id(codec)); | |
231 gchar *encoding_name = | |
232 purple_media_codec_get_encoding_name(codec); | |
233 gchar *clock_rate = g_strdup_printf("%d", | |
234 purple_media_codec_get_clock_rate(codec)); | |
235 payload = xmlnode_new_child(desc, "payload-type"); | |
236 if (session->video) | |
237 xmlnode_set_namespace(payload, NS_GOOGLE_SESSION_PHONE); | |
238 xmlnode_set_attrib(payload, "id", id); | |
239 /* | |
240 * Hack to make Gmail accept speex as the codec. | |
241 * It shouldn't have to be case sensitive. | |
242 */ | |
243 if (purple_strequal(encoding_name, "SPEEX")) | |
244 xmlnode_set_attrib(payload, "name", "speex"); | |
245 else | |
246 xmlnode_set_attrib(payload, "name", encoding_name); | |
247 xmlnode_set_attrib(payload, "clockrate", clock_rate); | |
248 g_free(clock_rate); | |
249 g_free(encoding_name); | |
250 g_free(id); | |
251 } | |
252 purple_media_codec_list_free(codecs); | |
253 | |
254 jabber_iq_send(iq); | |
255 | |
256 if (is_initiator) { | |
257 google_session_send_candidates(session->media, | |
258 "google-voice", session->remote_jid, | |
259 session); | |
260 google_session_send_candidates(session->media, | |
261 "google-video", session->remote_jid, | |
262 session); | |
263 } | |
264 | |
265 g_signal_handlers_disconnect_by_func(G_OBJECT(session->media), | |
266 G_CALLBACK(google_session_ready), session); | |
267 } | |
268 } | |
269 | |
270 static void | |
271 google_session_state_changed_cb(PurpleMedia *media, PurpleMediaState state, | |
272 gchar *sid, gchar *name, GoogleSession *session) | |
273 { | |
274 if (sid == NULL && name == NULL) { | |
275 if (state == PURPLE_MEDIA_STATE_END) { | |
276 google_session_destroy(session); | |
277 } | |
278 } | |
279 } | |
280 | |
281 static void | |
282 google_session_stream_info_cb(PurpleMedia *media, PurpleMediaInfoType type, | |
283 gchar *sid, gchar *name, gboolean local, | |
284 GoogleSession *session) | |
285 { | |
286 if (sid != NULL || name != NULL) | |
287 return; | |
288 | |
289 if (type == PURPLE_MEDIA_INFO_HANGUP) { | |
290 xmlnode *sess; | |
291 JabberIq *iq = jabber_iq_new(session->js, JABBER_IQ_SET); | |
292 | |
293 xmlnode_set_attrib(iq->node, "to", session->remote_jid); | |
294 sess = google_session_create_xmlnode(session, "terminate"); | |
295 xmlnode_insert_child(iq->node, sess); | |
296 | |
297 jabber_iq_send(iq); | |
298 } else if (type == PURPLE_MEDIA_INFO_REJECT) { | |
299 xmlnode *sess; | |
300 JabberIq *iq = jabber_iq_new(session->js, JABBER_IQ_SET); | |
301 | |
302 xmlnode_set_attrib(iq->node, "to", session->remote_jid); | |
303 sess = google_session_create_xmlnode(session, "reject"); | |
304 xmlnode_insert_child(iq->node, sess); | |
305 | |
306 jabber_iq_send(iq); | |
307 } else if (type == PURPLE_MEDIA_INFO_ACCEPT && local == TRUE) { | |
308 google_session_ready(session); | |
309 } | |
310 } | |
311 | |
312 static GParameter * | |
313 jabber_google_session_get_params(JabberStream *js, guint *num) | |
314 { | |
315 guint num_params; | |
316 GParameter *params = jingle_get_params(js, &num_params); | |
317 GParameter *new_params = g_new0(GParameter, num_params + 1); | |
318 | |
319 memcpy(new_params, params, sizeof(GParameter) * num_params); | |
320 | |
321 purple_debug_info("jabber", "setting Google jingle compatibility param\n"); | |
322 new_params[num_params].name = "compatibility-mode"; | |
323 g_value_init(&new_params[num_params].value, G_TYPE_UINT); | |
324 g_value_set_uint(&new_params[num_params].value, 1); /* NICE_COMPATIBILITY_GOOGLE */ | |
325 | |
326 g_free(params); | |
327 *num = num_params + 1; | |
328 return new_params; | |
329 } | |
330 | |
331 | |
332 gboolean | |
333 jabber_google_session_initiate(JabberStream *js, const gchar *who, PurpleMediaSessionType type) | |
334 { | |
335 GoogleSession *session; | |
336 JabberBuddy *jb; | |
337 JabberBuddyResource *jbr; | |
338 gchar *jid; | |
339 GParameter *params; | |
340 guint num_params; | |
341 | |
342 /* construct JID to send to */ | |
343 jb = jabber_buddy_find(js, who, FALSE); | |
344 if (!jb) { | |
345 purple_debug_error("jingle-rtp", | |
346 "Could not find Jabber buddy\n"); | |
347 return FALSE; | |
348 } | |
349 jbr = jabber_buddy_find_resource(jb, NULL); | |
350 if (!jbr) { | |
351 purple_debug_error("jingle-rtp", | |
352 "Could not find buddy's resource\n"); | |
353 } | |
354 | |
355 if ((strchr(who, '/') == NULL) && jbr && (jbr->name != NULL)) { | |
356 jid = g_strdup_printf("%s/%s", who, jbr->name); | |
357 } else { | |
358 jid = g_strdup(who); | |
359 } | |
360 | |
361 session = g_new0(GoogleSession, 1); | |
362 session->id.id = jabber_get_next_id(js); | |
363 session->id.initiator = g_strdup_printf("%s@%s/%s", js->user->node, | |
364 js->user->domain, js->user->resource); | |
365 session->state = SENT_INITIATE; | |
366 session->js = js; | |
367 session->remote_jid = jid; | |
368 | |
369 if (type & PURPLE_MEDIA_VIDEO) | |
370 session->video = TRUE; | |
371 | |
372 session->media = purple_media_manager_create_media( | |
373 purple_media_manager_get(), | |
374 purple_connection_get_account(js->gc), | |
375 "fsrtpconference", session->remote_jid, TRUE); | |
376 | |
377 purple_media_set_prpl_data(session->media, session); | |
378 | |
379 g_signal_connect_swapped(G_OBJECT(session->media), | |
380 "candidates-prepared", | |
381 G_CALLBACK(google_session_ready), session); | |
382 g_signal_connect_swapped(G_OBJECT(session->media), "codecs-changed", | |
383 G_CALLBACK(google_session_ready), session); | |
384 g_signal_connect(G_OBJECT(session->media), "state-changed", | |
385 G_CALLBACK(google_session_state_changed_cb), session); | |
386 g_signal_connect(G_OBJECT(session->media), "stream-info", | |
387 G_CALLBACK(google_session_stream_info_cb), session); | |
388 | |
389 params = jabber_google_session_get_params(js, &num_params); | |
390 | |
391 if (purple_media_add_stream(session->media, "google-voice", | |
392 session->remote_jid, PURPLE_MEDIA_AUDIO, | |
393 TRUE, "nice", num_params, params) == FALSE || | |
394 (session->video && purple_media_add_stream( | |
395 session->media, "google-video", | |
396 session->remote_jid, PURPLE_MEDIA_VIDEO, | |
397 TRUE, "nice", num_params, params) == FALSE)) { | |
398 purple_media_error(session->media, "Error adding stream."); | |
399 purple_media_end(session->media, NULL, NULL); | |
400 g_free(params); | |
401 return FALSE; | |
402 } | |
403 | |
404 g_free(params); | |
405 | |
406 return (session->media != NULL) ? TRUE : FALSE; | |
407 } | |
408 | |
409 static gboolean | |
410 google_session_handle_initiate(JabberStream *js, GoogleSession *session, xmlnode *sess, const char *iq_id) | |
411 { | |
412 JabberIq *result; | |
413 GList *codecs = NULL, *video_codecs = NULL; | |
414 xmlnode *desc_element, *codec_element; | |
415 PurpleMediaCodec *codec; | |
416 const char *xmlns; | |
417 GParameter *params; | |
418 guint num_params; | |
419 | |
420 if (session->state != UNINIT) { | |
421 purple_debug_error("jabber", "Received initiate for active session.\n"); | |
422 return FALSE; | |
423 } | |
424 | |
425 desc_element = xmlnode_get_child(sess, "description"); | |
426 xmlns = xmlnode_get_namespace(desc_element); | |
427 | |
428 if (purple_strequal(xmlns, NS_GOOGLE_SESSION_PHONE)) | |
429 session->video = FALSE; | |
430 else if (purple_strequal(xmlns, NS_GOOGLE_SESSION_VIDEO)) | |
431 session->video = TRUE; | |
432 else { | |
433 purple_debug_error("jabber", "Received initiate with " | |
434 "invalid namespace %s.\n", xmlns); | |
435 return FALSE; | |
436 } | |
437 | |
438 session->media = purple_media_manager_create_media( | |
439 purple_media_manager_get(), | |
440 purple_connection_get_account(js->gc), | |
441 "fsrtpconference", session->remote_jid, FALSE); | |
442 | |
443 purple_media_set_prpl_data(session->media, session); | |
444 | |
445 g_signal_connect_swapped(G_OBJECT(session->media), | |
446 "candidates-prepared", | |
447 G_CALLBACK(google_session_ready), session); | |
448 g_signal_connect_swapped(G_OBJECT(session->media), "codecs-changed", | |
449 G_CALLBACK(google_session_ready), session); | |
450 g_signal_connect(G_OBJECT(session->media), "state-changed", | |
451 G_CALLBACK(google_session_state_changed_cb), session); | |
452 g_signal_connect(G_OBJECT(session->media), "stream-info", | |
453 G_CALLBACK(google_session_stream_info_cb), session); | |
454 | |
455 params = jabber_google_session_get_params(js, &num_params); | |
456 | |
457 if (purple_media_add_stream(session->media, "google-voice", | |
458 session->remote_jid, PURPLE_MEDIA_AUDIO, FALSE, | |
459 "nice", num_params, params) == FALSE || | |
460 (session->video && purple_media_add_stream( | |
461 session->media, "google-video", | |
462 session->remote_jid, PURPLE_MEDIA_VIDEO, | |
463 FALSE, "nice", num_params, params) == FALSE)) { | |
464 purple_media_error(session->media, "Error adding stream."); | |
465 purple_media_stream_info(session->media, | |
466 PURPLE_MEDIA_INFO_REJECT, NULL, NULL, TRUE); | |
467 g_free(params); | |
468 return FALSE; | |
469 } | |
470 | |
471 g_free(params); | |
472 | |
473 for (codec_element = xmlnode_get_child(desc_element, "payload-type"); | |
474 codec_element; codec_element = codec_element->next) { | |
475 const char *id, *encoding_name, *clock_rate, | |
476 *width, *height, *framerate; | |
477 gboolean video; | |
478 if (codec_element->name && | |
479 strcmp(codec_element->name, "payload-type")) | |
480 continue; | |
481 | |
482 xmlns = xmlnode_get_namespace(codec_element); | |
483 encoding_name = xmlnode_get_attrib(codec_element, "name"); | |
484 id = xmlnode_get_attrib(codec_element, "id"); | |
485 | |
486 if (!session->video || | |
487 (xmlns && !strcmp(xmlns, NS_GOOGLE_SESSION_PHONE))) { | |
488 clock_rate = xmlnode_get_attrib( | |
489 codec_element, "clockrate"); | |
490 video = FALSE; | |
491 } else { | |
492 width = xmlnode_get_attrib(codec_element, "width"); | |
493 height = xmlnode_get_attrib(codec_element, "height"); | |
494 framerate = xmlnode_get_attrib( | |
495 codec_element, "framerate"); | |
496 clock_rate = "90000"; | |
497 video = TRUE; | |
498 } | |
499 | |
500 if (id) { | |
501 codec = purple_media_codec_new(atoi(id), encoding_name, | |
502 video ? PURPLE_MEDIA_VIDEO : | |
503 PURPLE_MEDIA_AUDIO, | |
504 clock_rate ? atoi(clock_rate) : 0); | |
505 if (video) | |
506 video_codecs = g_list_append( | |
507 video_codecs, codec); | |
508 else | |
509 codecs = g_list_append(codecs, codec); | |
510 } | |
511 } | |
512 | |
513 if (codecs) | |
514 purple_media_set_remote_codecs(session->media, "google-voice", | |
515 session->remote_jid, codecs); | |
516 if (video_codecs) | |
517 purple_media_set_remote_codecs(session->media, "google-video", | |
518 session->remote_jid, video_codecs); | |
519 | |
520 purple_media_codec_list_free(codecs); | |
521 purple_media_codec_list_free(video_codecs); | |
522 | |
523 result = jabber_iq_new(js, JABBER_IQ_RESULT); | |
524 jabber_iq_set_id(result, iq_id); | |
525 xmlnode_set_attrib(result->node, "to", session->remote_jid); | |
526 jabber_iq_send(result); | |
527 | |
528 return TRUE; | |
529 } | |
530 | |
531 static void | |
532 google_session_handle_candidates(JabberStream *js, GoogleSession *session, xmlnode *sess, const char *iq_id) | |
533 { | |
534 JabberIq *result; | |
535 GList *list = NULL, *video_list = NULL; | |
536 xmlnode *cand; | |
537 static int name = 0; | |
538 char n[4]; | |
539 | |
540 for (cand = xmlnode_get_child(sess, "candidate"); cand; | |
541 cand = xmlnode_get_next_twin(cand)) { | |
542 PurpleMediaCandidate *info; | |
543 const gchar *cname = xmlnode_get_attrib(cand, "name"); | |
544 const gchar *type = xmlnode_get_attrib(cand, "type"); | |
545 const gchar *protocol = xmlnode_get_attrib(cand, "protocol"); | |
546 const gchar *address = xmlnode_get_attrib(cand, "address"); | |
547 const gchar *port = xmlnode_get_attrib(cand, "port"); | |
548 guint component_id; | |
549 | |
550 if (cname && type && address && port) { | |
551 PurpleMediaCandidateType candidate_type; | |
552 | |
553 g_snprintf(n, sizeof(n), "S%d", name++); | |
554 | |
555 if (g_str_equal(type, "local")) | |
556 candidate_type = PURPLE_MEDIA_CANDIDATE_TYPE_HOST; | |
557 else if (g_str_equal(type, "stun")) | |
558 candidate_type = PURPLE_MEDIA_CANDIDATE_TYPE_PRFLX; | |
559 else if (g_str_equal(type, "relay")) | |
560 candidate_type = PURPLE_MEDIA_CANDIDATE_TYPE_RELAY; | |
561 else | |
562 candidate_type = PURPLE_MEDIA_CANDIDATE_TYPE_HOST; | |
563 | |
564 if (purple_strequal(cname, "rtcp") || | |
565 purple_strequal(cname, "video_rtcp")) | |
566 component_id = PURPLE_MEDIA_COMPONENT_RTCP; | |
567 else | |
568 component_id = PURPLE_MEDIA_COMPONENT_RTP; | |
569 | |
570 info = purple_media_candidate_new(n, component_id, | |
571 candidate_type, | |
572 purple_strequal(protocol, "udp") ? | |
573 PURPLE_MEDIA_NETWORK_PROTOCOL_UDP : | |
574 PURPLE_MEDIA_NETWORK_PROTOCOL_TCP, | |
575 address, | |
576 atoi(port)); | |
577 g_object_set(info, "username", xmlnode_get_attrib(cand, "username"), | |
578 "password", xmlnode_get_attrib(cand, "password"), NULL); | |
579 if (!strncmp(cname, "video_", 6)) | |
580 video_list = g_list_append(video_list, info); | |
581 else | |
582 list = g_list_append(list, info); | |
583 } | |
584 } | |
585 | |
586 if (list) | |
587 purple_media_add_remote_candidates( | |
588 session->media, "google-voice", | |
589 session->remote_jid, list); | |
590 if (video_list) | |
591 purple_media_add_remote_candidates( | |
592 session->media, "google-video", | |
593 session->remote_jid, video_list); | |
594 purple_media_candidate_list_free(list); | |
595 purple_media_candidate_list_free(video_list); | |
596 | |
597 result = jabber_iq_new(js, JABBER_IQ_RESULT); | |
598 jabber_iq_set_id(result, iq_id); | |
599 xmlnode_set_attrib(result->node, "to", session->remote_jid); | |
600 jabber_iq_send(result); | |
601 } | |
602 | |
603 static void | |
604 google_session_handle_accept(JabberStream *js, GoogleSession *session, xmlnode *sess, const char *iq_id) | |
605 { | |
606 xmlnode *desc_element = xmlnode_get_child(sess, "description"); | |
607 xmlnode *codec_element = xmlnode_get_child( | |
608 desc_element, "payload-type"); | |
609 GList *codecs = NULL, *video_codecs = NULL; | |
610 JabberIq *result = NULL; | |
611 const gchar *xmlns = xmlnode_get_namespace(desc_element); | |
612 gboolean video = (xmlns && !strcmp(xmlns, NS_GOOGLE_SESSION_VIDEO)); | |
613 | |
614 for (; codec_element; codec_element = codec_element->next) { | |
615 const gchar *xmlns, *encoding_name, *id, | |
616 *clock_rate, *width, *height, *framerate; | |
617 gboolean video_codec = FALSE; | |
618 | |
619 if (!purple_strequal(codec_element->name, "payload-type")) | |
620 continue; | |
621 | |
622 xmlns = xmlnode_get_namespace(codec_element); | |
623 encoding_name = xmlnode_get_attrib(codec_element, "name"); | |
624 id = xmlnode_get_attrib(codec_element, "id"); | |
625 | |
626 if (!video || purple_strequal(xmlns, NS_GOOGLE_SESSION_PHONE)) | |
627 clock_rate = xmlnode_get_attrib( | |
628 codec_element, "clockrate"); | |
629 else { | |
630 clock_rate = "90000"; | |
631 width = xmlnode_get_attrib(codec_element, "width"); | |
632 height = xmlnode_get_attrib(codec_element, "height"); | |
633 framerate = xmlnode_get_attrib( | |
634 codec_element, "framerate"); | |
635 video_codec = TRUE; | |
636 } | |
637 | |
638 if (id && encoding_name) { | |
639 PurpleMediaCodec *codec = purple_media_codec_new( | |
640 atoi(id), encoding_name, | |
641 video_codec ? PURPLE_MEDIA_VIDEO : | |
642 PURPLE_MEDIA_AUDIO, | |
643 clock_rate ? atoi(clock_rate) : 0); | |
644 if (video_codec) | |
645 video_codecs = g_list_append( | |
646 video_codecs, codec); | |
647 else | |
648 codecs = g_list_append(codecs, codec); | |
649 } | |
650 } | |
651 | |
652 if (codecs) | |
653 purple_media_set_remote_codecs(session->media, "google-voice", | |
654 session->remote_jid, codecs); | |
655 if (video_codecs) | |
656 purple_media_set_remote_codecs(session->media, "google-video", | |
657 session->remote_jid, video_codecs); | |
658 | |
659 purple_media_stream_info(session->media, PURPLE_MEDIA_INFO_ACCEPT, | |
660 NULL, NULL, FALSE); | |
661 | |
662 result = jabber_iq_new(js, JABBER_IQ_RESULT); | |
663 jabber_iq_set_id(result, iq_id); | |
664 xmlnode_set_attrib(result->node, "to", session->remote_jid); | |
665 jabber_iq_send(result); | |
666 } | |
667 | |
668 static void | |
669 google_session_handle_reject(JabberStream *js, GoogleSession *session, xmlnode *sess) | |
670 { | |
671 purple_media_end(session->media, NULL, NULL); | |
672 } | |
673 | |
674 static void | |
675 google_session_handle_terminate(JabberStream *js, GoogleSession *session, xmlnode *sess) | |
676 { | |
677 purple_media_end(session->media, NULL, NULL); | |
678 } | |
679 | |
680 static void | |
681 google_session_parse_iq(JabberStream *js, GoogleSession *session, xmlnode *sess, const char *iq_id) | |
682 { | |
683 const char *type = xmlnode_get_attrib(sess, "type"); | |
684 | |
685 if (!strcmp(type, "initiate")) { | |
686 google_session_handle_initiate(js, session, sess, iq_id); | |
687 } else if (!strcmp(type, "accept")) { | |
688 google_session_handle_accept(js, session, sess, iq_id); | |
689 } else if (!strcmp(type, "reject")) { | |
690 google_session_handle_reject(js, session, sess); | |
691 } else if (!strcmp(type, "terminate")) { | |
692 google_session_handle_terminate(js, session, sess); | |
693 } else if (!strcmp(type, "candidates")) { | |
694 google_session_handle_candidates(js, session, sess, iq_id); | |
695 } | |
696 } | |
697 | |
698 void | |
699 jabber_google_session_parse(JabberStream *js, const char *from, | |
700 JabberIqType type, const char *iq_id, | |
701 xmlnode *session_node) | |
702 { | |
703 GoogleSession *session = NULL; | |
704 GoogleSessionId id; | |
705 | |
706 xmlnode *desc_node; | |
707 | |
708 GList *iter = NULL; | |
709 | |
710 if (type != JABBER_IQ_SET) | |
711 return; | |
712 | |
713 id.id = (gchar*)xmlnode_get_attrib(session_node, "id"); | |
714 if (!id.id) | |
715 return; | |
716 | |
717 id.initiator = (gchar*)xmlnode_get_attrib(session_node, "initiator"); | |
718 if (!id.initiator) | |
719 return; | |
720 | |
721 iter = purple_media_manager_get_media_by_account( | |
722 purple_media_manager_get(), | |
723 purple_connection_get_account(js->gc)); | |
724 for (; iter; iter = g_list_delete_link(iter, iter)) { | |
725 GoogleSession *gsession = | |
726 purple_media_get_prpl_data(iter->data); | |
727 if (google_session_id_equal(&(gsession->id), &id)) { | |
728 session = gsession; | |
729 break; | |
730 } | |
731 } | |
732 if (iter != NULL) { | |
733 g_list_free(iter); | |
734 } | |
735 | |
736 if (session) { | |
737 google_session_parse_iq(js, session, session_node, iq_id); | |
738 return; | |
739 } | |
740 | |
741 /* If the session doesn't exist, this has to be an initiate message */ | |
742 if (strcmp(xmlnode_get_attrib(session_node, "type"), "initiate")) | |
743 return; | |
744 desc_node = xmlnode_get_child(session_node, "description"); | |
745 if (!desc_node) | |
746 return; | |
747 session = g_new0(GoogleSession, 1); | |
748 session->id.id = g_strdup(id.id); | |
749 session->id.initiator = g_strdup(id.initiator); | |
750 session->state = UNINIT; | |
751 session->js = js; | |
752 session->remote_jid = g_strdup(session->id.initiator); | |
753 | |
754 google_session_handle_initiate(js, session, session_node, iq_id); | |
755 } | |
756 #endif /* USE_VV */ | |
757 | |
758 |