/*************************************************************************** * friend.c * * Sat May 15 15:25:16 2004 * Copyright 2004 Simon Morlat * Email ****************************************************************************/ /* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Library General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "linphonecore.h" #include "private.h" #include #include #include "lpconfig.h" const char *linphone_online_status_to_string(LinphoneOnlineStatus ss){ const char *str=NULL; switch(ss){ case LINPHONE_STATUS_UNKNOWN: str=_("Unknown"); break; case LINPHONE_STATUS_ONLINE: str=_("Online"); break; case LINPHONE_STATUS_BUSY: str=_("Busy"); break; case LINPHONE_STATUS_BERIGHTBACK: str=_("Be right back"); break; case LINPHONE_STATUS_AWAY: str=_("Away"); break; case LINPHONE_STATUS_ONTHEPHONE: str=_("On the phone"); break; case LINPHONE_STATUS_OUTTOLUNCH: str=_("Out to lunch"); break; case LINPHONE_STATUS_NOT_DISTURB: str=_("Do not disturb"); break; case LINPHONE_STATUS_MOVED: str=_("Moved"); break; case LINPHONE_STATUS_ALT_SERVICE: str=_("Using another messaging service"); break; case LINPHONE_STATUS_OFFLINE: str=_("Offline"); break; case LINPHONE_STATUS_PENDING: str=_("Pending"); break; case LINPHONE_STATUS_CLOSED: str=_("Closed"); break; default: str=_("Unknown-bug"); } return str; } static int friend_data_compare(const void * a, const void * b, void * data){ osip_from_t *fa=((LinphoneFriend*)a)->url; osip_from_t *fb=((LinphoneFriend*)b)->url; char *ua,*ub; ua=fa->url->username; ub=fb->url->username; if (ua!=NULL && ub!=NULL) { //printf("Comparing usernames %s,%s\n",ua,ub); return strcasecmp(ua,ub); } else { /* compare hosts*/ ua=fa->url->host; ub=fb->url->host; if (ua!=NULL && ub!=NULL){ int ret=strcasecmp(ua,ub); //printf("Comparing hostnames %s,%s,res=%i\n",ua,ub,ret); return ret; } else return -1; } } static int friend_compare(const void * a, const void * b){ return friend_data_compare(a,b,NULL); } MSList *find_friend(MSList *fl, const osip_from_t *friend, LinphoneFriend **lf){ MSList *res=NULL; LinphoneFriend dummy; if (lf!=NULL) *lf=NULL; dummy.url=(osip_from_t*)friend; res=ms_list_find_custom(fl,friend_compare,&dummy); if (lf!=NULL && res!=NULL) *lf=(LinphoneFriend*)res->data; return res; } LinphoneFriend *linphone_find_friend_by_nid(MSList *l, int nid){ MSList *elem; for (elem=l;elem!=NULL;elem=elem->next){ LinphoneFriend *lf=(LinphoneFriend*)elem->data; if (lf->nid==nid) return lf; } return NULL; } LinphoneFriend *linphone_find_friend_by_sid(MSList *l, int sid){ MSList *elem; for (elem=l;elem!=NULL;elem=elem->next){ LinphoneFriend *lf=(LinphoneFriend*)elem->data; if (lf->sid==sid) return lf; } return NULL; } void __linphone_friend_do_subscribe(LinphoneFriend *fr){ char *friend=NULL; const char *route=NULL; const char *from=NULL; osip_message_t *msg=NULL; osip_from_to_str(fr->url,&friend); if (fr->proxy!=NULL){ route=fr->proxy->reg_route; from=fr->proxy->reg_identity; }else from=linphone_core_get_primary_contact(fr->lc); if (fr->sid<0){ /* people for which we don't have yet an answer should appear as offline */ fr->lc->vtable.notify_recv(fr->lc,(LinphoneFriend*)fr,friend,_("Gone"),"sip-closed.png"); } eXosip_lock(); eXosip_subscribe_build_initial_request(&msg,friend,from,route,"presence",600); eXosip_subscribe_send_initial_request(msg); eXosip_unlock(); osip_free(friend); } LinphoneFriend * linphone_friend_new(){ LinphoneFriend *obj=ms_new0(LinphoneFriend,1); obj->out_did=-1; obj->in_did=-1; obj->nid=-1; obj->sid=-1; obj->pol=LinphoneSPAccept; obj->status=LINPHONE_STATUS_OFFLINE; obj->subscribe=TRUE; return obj; } LinphoneFriend *linphone_friend_new_with_addr(const char *addr){ LinphoneFriend *fr=linphone_friend_new(); if (linphone_friend_set_sip_addr(fr,addr)<0){ linphone_friend_destroy(fr); return NULL; } return fr; } void linphone_core_interpret_friend_uri(LinphoneCore *lc, const char *uri, char **result){ int err; osip_from_t *fr=NULL; osip_from_init(&fr); err=osip_from_parse(fr,uri); if (err<0){ char *tmp=NULL; if (strchr(uri,'@')!=NULL){ /*try adding sip:*/ tmp=ms_strdup_printf("sip:%s",uri); }else if (lc->default_proxy!=NULL){ /*try adding domain part from default current proxy*/ osip_from_t *id=NULL; osip_from_init(&id); if (osip_from_parse(id,linphone_core_get_identity(lc))==0){ if (id->url->port!=NULL && strlen(id->url->port)>0) tmp=ms_strdup_printf("sip:%s@%s:%s",uri,id->url->host,id->url->port); else tmp=ms_strdup_printf("sip:%s@%s",uri,id->url->host); } osip_from_free(id); } if (osip_from_parse(fr,tmp)==0){ /*looks good */ ms_message("%s interpreted as %s",uri,tmp); *result=tmp; }else *result=NULL; }else *result=ms_strdup(uri); osip_from_free(fr); } int linphone_friend_set_sip_addr(LinphoneFriend *lf, const char *addr){ int err; osip_from_t *fr=NULL; osip_from_init(&fr); err=osip_from_parse(fr,addr); if (err<0) { ms_warning("Invalid friend sip uri: %s",addr); osip_from_free(fr); return -1; } if (lf->url!=NULL) osip_from_free(lf->url); lf->url=fr; return 0; } int linphone_friend_set_name(LinphoneFriend *lf, const char *name){ osip_from_t *fr=lf->url; if (fr==NULL){ ms_error("linphone_friend_set_sip_addr() must be called before linphone_friend_set_name()."); return -1; } if (fr->displayname!=NULL){ osip_free(fr->displayname); fr->displayname=NULL; } if (name && name[0]!='\0'){ fr->displayname=osip_strdup(name); } return 0; } int linphone_friend_send_subscribe(LinphoneFriend *fr, bool_t val){ fr->subscribe=val; return 0; } int linphone_friend_set_inc_subscribe_policy(LinphoneFriend *fr, LinphoneSubscribePolicy pol) { fr->pol=pol; return 0; } int linphone_friend_set_proxy(LinphoneFriend *fr, struct _LinphoneProxyConfig *cfg){ fr->proxy=cfg; return 0; } void linphone_friend_set_sid(LinphoneFriend *lf, int sid){ lf->sid=sid; } void linphone_friend_set_nid(LinphoneFriend *lf, int nid){ lf->nid=nid; lf->inc_subscribe_pending=TRUE; } void add_presence_body(osip_message_t *notify, LinphoneOnlineStatus online_status) { char buf[1000]; #ifdef SUPPORT_MSN int atom_id = 1000; #endif char *contact_info; osip_contact_t *ct=NULL; osip_message_get_contact(notify,0,&ct); osip_contact_to_str(ct,&contact_info); #ifdef SUPPORT_MSN if (online_status==LINPHONE_STATUS_ONLINE) { sprintf(buf, "\n\ \n\ \n\ \n\ \n\
\n\ \n\ \n\
\n\
\n\
", contact_info, atom_id, contact_info); } else if (online_status==LINPHONE_STATUS_BUSY) { sprintf(buf, "\n\ \n\ \n\ \n\ \n\
\n\ \n\ \n\
\n\
\n\
", contact_info, atom_id, contact_info); } else if (online_status==LINPHONE_STATUS_BERIGHTBACK) { sprintf(buf, "\n\ \n\ \n\ \n\ \n\
\n\ \n\ \n\
\n\
\n\
", contact_info, atom_id, contact_info); } else if (online_status==LINPHONE_STATUS_AWAY) { sprintf(buf, "\n\ \n\ \n\ \n\ \n\
\n\ \n\ \n\
\n\
\n\
", contact_info, atom_id, contact_info); } else if (online_status==LINPHONE_STATUS_ONTHEPHONE) { sprintf(buf, "\n\ \n\ \n\ \n\ \n\
\n\ \n\ \n\
\n\
\n\
", contact_info, atom_id, contact_info); } else if (online_status==LINPHONE_STATUS_OUTTOLUNCH) { sprintf(buf, "\n\ \n\ \n\ \n\ \n\
\n\ \n\ \n\
\n\
\n\
", contact_info, atom_id, contact_info); } else { sprintf(buf, "\n\ \n\ \n\ \n\ \n\
\n\ \n\ \n\
\n\
\n\
", contact_info, atom_id, contact_info); } osip_message_set_body(notify, buf, strlen(buf)); osip_message_set_content_type(notify, "application/xpidf+xml"); #else if (online_status==LINPHONE_STATUS_ONLINE) { sprintf(buf, "\n\ \n\ \n\ \n\ open\n\ \n\ %s\n\ online\n\ \n\ ", contact_info, contact_info); } else if (online_status==LINPHONE_STATUS_BUSY) { sprintf(buf, "\n\ \n\ \n\ \n\ open\n\ \n\ busy\n\ \n\ \n\ %s\n\ busy\n\ \n\ ", contact_info, contact_info); } else if (online_status==LINPHONE_STATUS_BERIGHTBACK) { sprintf(buf, "\n\ \n\ \n\ \n\ open\n\ \n\ in-transit\n\ \n\ \n\ %s\n\ be right back\n\ \n\ ", contact_info, contact_info); } else if (online_status==LINPHONE_STATUS_AWAY) { sprintf(buf, "\n\ \n\ \n\ \n\ open\n\ \n\ away\n\ \n\ \n\ %s\n\ away\n\ \n\ ", contact_info, contact_info); } else if (online_status==LINPHONE_STATUS_ONTHEPHONE) { sprintf(buf, "\n\ \n\ \n\ \n\ open\n\ \n\ on-the-phone\n\ \n\ \n\ %s\n\ on the phone\n\ \n\ ", contact_info, contact_info); } else if (online_status==LINPHONE_STATUS_OUTTOLUNCH) { sprintf(buf, "\n\ \n\ \n\ \n\ open\n\ \n\ meal\n\ \n\ \n\ %s\n\ out to lunch\n\ \n\ ", contact_info, contact_info); } else { /* */ sprintf(buf, "\n\ \n%s", contact_info, "\n\ \n\ closed\n\ \n\ permanent-absence\n\ \n\ \n\ \n\ \n\n"); } osip_message_set_body(notify, buf, strlen(buf)); osip_message_set_content_type(notify, "application/pidf+xml"); #endif osip_free(contact_info); } void linphone_friend_notify(LinphoneFriend *lf, int ss, LinphoneOnlineStatus os){ //printf("Wish to notify %p, lf->nid=%i\n",lf,lf->nid); if (lf->in_did!=-1){ osip_message_t *msg=NULL; const char *identity; if (lf->proxy!=NULL) identity=lf->proxy->reg_identity; else identity=linphone_core_get_primary_contact(lf->lc); eXosip_lock(); eXosip_insubscription_build_notify(lf->in_did,ss,0,&msg); if (msg!=NULL){ osip_message_set_contact(msg,identity); add_presence_body(msg,os); eXosip_insubscription_send_request(lf->in_did,msg); }else ms_error("could not create notify for incoming subscription."); eXosip_unlock(); } } static void linphone_friend_unsubscribe(LinphoneFriend *lf){ if (lf->out_did!=-1) { osip_message_t *msg=NULL; eXosip_lock(); eXosip_subscribe_build_refresh_request(lf->out_did,&msg); if (msg){ osip_message_set_expires(msg,"0"); eXosip_subscribe_send_refresh_request(lf->out_did,msg); }else ms_error("Could not build subscribe refresh request !"); eXosip_unlock(); } } void linphone_friend_destroy(LinphoneFriend *lf){ linphone_friend_notify(lf,EXOSIP_SUBCRSTATE_TERMINATED,LINPHONE_STATUS_CLOSED); linphone_friend_unsubscribe(lf); if (lf->url!=NULL) osip_from_free(lf->url); ms_free(lf); } void linphone_friend_check_for_removed_proxy(LinphoneFriend *lf, LinphoneProxyConfig *cfg){ if (lf->proxy==cfg){ lf->proxy=NULL; } } char *linphone_friend_get_addr(LinphoneFriend *lf){ char *ret,*tmp; if (lf->url==NULL) return NULL; osip_uri_to_str(lf->url->url,&tmp); ret=ms_strdup(tmp); osip_free(tmp); return ret; } char *linphone_friend_get_name(LinphoneFriend *lf){ if (lf->url==NULL) return NULL; if (lf->url->displayname==NULL) return NULL; return ms_strdup(lf->url->displayname); } char * linphone_friend_get_url(LinphoneFriend *lf){ char *tmp,*ret; if (lf->url==NULL) return NULL; osip_from_to_str(lf->url,&tmp); ret=ms_strdup(tmp); ms_free(tmp); return ret; } bool_t linphone_friend_get_send_subscribe(const LinphoneFriend *lf){ return lf->subscribe; } LinphoneSubscribePolicy linphone_friend_get_inc_subscribe_policy(const LinphoneFriend *lf){ return lf->pol; } LinphoneOnlineStatus linphone_friend_get_status(const LinphoneFriend *lf){ return lf->status; } void linphone_friend_apply(LinphoneFriend *fr, LinphoneCore *lc){ if (fr->url==NULL) { ms_warning("No sip url defined."); return; } fr->lc=lc; linphone_core_write_friends_config(lc); if (fr->inc_subscribe_pending){ switch(fr->pol){ case LinphoneSPWait: linphone_friend_notify(fr,EXOSIP_SUBCRSTATE_PENDING,LINPHONE_STATUS_PENDING); break; case LinphoneSPAccept: if (fr->lc!=NULL) { linphone_friend_notify(fr,EXOSIP_SUBCRSTATE_ACTIVE,fr->lc->presence_mode); } break; case LinphoneSPDeny: linphone_friend_notify(fr,EXOSIP_SUBCRSTATE_TERMINATED,LINPHONE_STATUS_CLOSED); break; } fr->inc_subscribe_pending=FALSE; } if (fr->subscribe && fr->out_did==-1){ __linphone_friend_do_subscribe(fr); } ms_message("linphone_friend_apply() done."); } void linphone_friend_edit(LinphoneFriend *fr){ } void linphone_friend_done(LinphoneFriend *fr){ ms_return_if_fail(fr!=NULL); if (fr->lc==NULL) return; linphone_friend_apply(fr,fr->lc); } void linphone_core_add_friend(LinphoneCore *lc, LinphoneFriend *lf) { ms_return_if_fail(lf->lc==NULL); ms_return_if_fail(lf->url!=NULL); lc->friends=ms_list_append(lc->friends,lf); linphone_friend_apply(lf,lc); return ; } void linphone_core_remove_friend(LinphoneCore *lc, LinphoneFriend* fl){ MSList *el=ms_list_find(lc->friends,(void *)fl); if (el!=NULL){ lc->friends=ms_list_remove_link(lc->friends,el); linphone_friend_destroy((LinphoneFriend*)el->data); linphone_core_write_friends_config(lc); } } #define key_compare(key, word) strncasecmp((key),(word),strlen(key)) LinphoneSubscribePolicy __policy_str_to_enum(const char* pol){ if (key_compare("accept",pol)==0){ return LinphoneSPAccept; } if (key_compare("deny",pol)==0){ return LinphoneSPDeny; } if (key_compare("wait",pol)==0){ return LinphoneSPWait; } ms_warning("Unrecognized subscribe policy: %s",pol); return LinphoneSPWait; } LinphoneProxyConfig *__index_to_proxy(LinphoneCore *lc, int index){ if (index>=0) return (LinphoneProxyConfig*)ms_list_nth_data(lc->sip_conf.proxies,index); else return NULL; } LinphoneFriend * linphone_friend_new_from_config_file(LinphoneCore *lc, int index){ const char *tmp; char item[50]; int a; LinphoneFriend *lf; LpConfig *config=lc->config; sprintf(item,"friend_%i",index); if (!lp_config_has_section(config,item)){ return NULL; } tmp=lp_config_get_string(config,item,"url",NULL); if (tmp==NULL) { return NULL; } lf=linphone_friend_new_with_addr(tmp); if (lf==NULL) { return NULL; } tmp=lp_config_get_string(config,item,"pol",NULL); if (tmp==NULL) linphone_friend_set_inc_subscribe_policy(lf,LinphoneSPWait); else{ linphone_friend_set_inc_subscribe_policy(lf,__policy_str_to_enum(tmp)); } a=lp_config_get_int(config,item,"subscribe",0); linphone_friend_send_subscribe(lf,a); a=lp_config_get_int(config,item,"proxy",-1); if (a!=-1) { linphone_friend_set_proxy(lf,__index_to_proxy(lc,a)); } return lf; } const char *__policy_enum_to_str(LinphoneSubscribePolicy pol){ switch(pol){ case LinphoneSPAccept: return "accept"; break; case LinphoneSPDeny: return "deny"; break; case LinphoneSPWait: return "wait"; break; } ms_warning("Invalid policy enum value."); return "wait"; } void linphone_friend_write_to_config_file(LpConfig *config, LinphoneFriend *lf, int index){ char key[50]; char *tmp; int a; sprintf(key,"friend_%i",index); if (lf==NULL){ lp_config_clean_section(config,key); return; } if (lf->url!=NULL){ osip_from_to_str(lf->url,&tmp); if (tmp==NULL) { return; } lp_config_set_string(config,key,"url",tmp); osip_free(tmp); } lp_config_set_string(config,key,"pol",__policy_enum_to_str(lf->pol)); lp_config_set_int(config,key,"subscribe",lf->subscribe); if (lf->proxy!=NULL){ a=ms_list_index(lf->lc->sip_conf.proxies,lf->proxy); lp_config_set_int(config,key,"proxy",a); }else lp_config_set_int(config,key,"proxy",-1); } void linphone_core_write_friends_config(LinphoneCore* lc) { MSList *elem; int i; if (!lc->ready) return; /*dont write config when reading it !*/ for (elem=lc->friends,i=0; elem!=NULL; elem=ms_list_next(elem),i++){ linphone_friend_write_to_config_file(lc->config,(LinphoneFriend*)elem->data,i); } linphone_friend_write_to_config_file(lc->config,NULL,i); /* set the end */ }