]> sjero.net Git - linphone/blobdiff - coreapi/sal_eXosip2_presence.c
fix publish request, whose request uri must identify the resources for which the...
[linphone] / coreapi / sal_eXosip2_presence.c
index 8b17dfea670f0e39f58aff1f695c69c29a746b1a..c8191098649343f2bb6ebf8689d06e881efd557f 100644 (file)
@@ -20,14 +20,26 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 
 #include "sal_eXosip2.h"
 
+typedef enum {
+       PIDF = 0,
+       RFCxxxx = 1,
+       MSOLDPRES = 2
+} presence_type_t;
 
-static SalOp * sal_find_out_subscribe(Sal *sal, int sid){
+/*
+ * REVISIT: this static variable forces every dialog to use the same presence description type depending 
+ * on what is received on a single dialog...
+ */
+static presence_type_t presence_style = PIDF;
+
+SalOp * sal_find_out_subscribe(Sal *sal, int sid){
        const MSList *elem;
        SalOp *op;
        for(elem=sal->out_subscribes;elem!=NULL;elem=elem->next){
                op=(SalOp*)elem->data;
                if (op->sid==sid) return op;
        }
+       ms_message("No op for sid %i",sid);
        return NULL;
 }
 
@@ -39,7 +51,7 @@ void sal_remove_out_subscribe(Sal *sal, SalOp *op){
        sal->out_subscribes=ms_list_remove(sal->out_subscribes,op);
 }
 
-static SalOp * sal_find_in_subscribe(Sal *sal, int nid){
+SalOp * sal_find_in_subscribe(Sal *sal, int nid){
        const MSList *elem;
        SalOp *op;
        for(elem=sal->in_subscribes;elem!=NULL;elem=elem->next){
@@ -49,7 +61,19 @@ static SalOp * sal_find_in_subscribe(Sal *sal, int nid){
        return NULL;
 }
 
-static void sal_add_in_subscribe(Sal *sal, SalOp *op){
+static SalOp * sal_find_in_subscribe_by_call_id(Sal *sal, osip_call_id_t *call_id){
+       const MSList *elem;
+       SalOp *op;
+       for(elem=sal->in_subscribes;elem!=NULL;elem=elem->next){
+               op=(SalOp*)elem->data;
+               if (op->call_id && osip_call_id_match(op->call_id,call_id)==0)
+                       return op;
+       }
+       return NULL;
+}
+
+static void sal_add_in_subscribe(Sal *sal, SalOp *op, osip_message_t *subs){
+       osip_call_id_clone(subs->call_id,&op->call_id);
        sal->in_subscribes=ms_list_append(sal->in_subscribes,op);
 }
 
@@ -57,26 +81,81 @@ void sal_remove_in_subscribe(Sal *sal, SalOp *op){
        sal->in_subscribes=ms_list_remove(sal->in_subscribes,op);
 }
 
-int sal_text_send(SalOp *op, const char *from, const char *to, const char *msg){
+static const char *days[]={"Sun","Mon","Tue","Wed","Thu","Fri","Sat"};
+static const char *months[]={"Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"};
+
+static void msg_add_current_date(osip_message_t *msg){
+        char tmp[64]={0};
+        time_t curtime=time(NULL);
+        struct tm *ret;
+#ifndef WIN32
+        struct tm gmt;
+        ret=gmtime_r(&curtime,&gmt);
+#else
+        ret=gmtime(&curtime);
+#endif
+        /*cannot use strftime because it is locale dependant*/
+        snprintf(tmp,sizeof(tmp)-1,"%s, %i %s %i %02i:%02i:%02i GMT",
+                 days[ret->tm_wday],ret->tm_mday,months[ret->tm_mon],1900+ret->tm_year,ret->tm_hour,ret->tm_min,ret->tm_sec);
+        osip_message_replace_header(msg,"Date",tmp);
+}
+
+
+int sal_message_send(SalOp *op, const char *from, const char *to, const char* content_type, const char *msg){
        osip_message_t *sip=NULL;
-       if (from)
-               sal_op_set_from(op,from);
-       if (to)
-               sal_op_set_to(op,to);
 
-       eXosip_lock();
-       eXosip_message_build_request(&sip,"MESSAGE",sal_op_get_to(op),
-           sal_op_get_from(op),sal_op_get_route(op));
-       osip_message_set_content_type(sip,"text/plain");
-       osip_message_set_body(sip,msg,strlen(msg));
-       eXosip_message_send_request(sip);
-       eXosip_unlock();
+       if(op->cid == -1)
+       {
+               /* we are not currently in communication with the destination */
+               if (from)
+                       sal_op_set_from(op,from);
+               if (to)
+                       sal_op_set_to(op,to);
+
+               sal_exosip_fix_route(op);
+               eXosip_lock();
+               eXosip_message_build_request(&sip,"MESSAGE",sal_op_get_to(op),
+                       sal_op_get_from(op),sal_op_get_route(op));
+               if (sip!=NULL){
+                       sal_exosip_add_custom_headers(sip,op->base.custom_headers);
+                       msg_add_current_date(sip);
+                       osip_message_set_content_type(sip,content_type);
+                       if (msg) osip_message_set_body(sip,msg,strlen(msg));
+                       sal_add_other(op->base.root,op,sip);
+                       eXosip_message_send_request(sip);
+               }else{
+                       ms_error("Could not build MESSAGE request !");
+               }
+               eXosip_unlock();
+       }
+       else
+       {
+               /* we are currently in communication with the destination */
+               eXosip_lock();
+               //First we generate an INFO message to get the current call_id and a good cseq
+               eXosip_call_build_request(op->did,"MESSAGE",&sip);
+               if(sip == NULL)
+               {
+                       ms_warning("could not get a build info to send MESSAGE, maybe no previous call established ?");
+                       eXosip_unlock();
+                       return -1;
+               }
+               sal_exosip_add_custom_headers(sip,op->base.custom_headers);
+               msg_add_current_date(sip);
+               osip_message_set_content_type(sip,content_type);
+               if (msg) osip_message_set_body(sip,msg,strlen(msg));
+               eXosip_call_send_request(op->did,sip);
+               eXosip_unlock();
+       }
        return 0;
 }
 
+int sal_text_send(SalOp *op, const char *from, const char *to, const char *msg) {
+       return sal_message_send(op,from,to,"text/plain",msg);
+}
 /*presence Subscribe/notify*/
 int sal_subscribe_presence(SalOp *op, const char *from, const char *to){
-       osip_message_t *msg;
+       osip_message_t *msg=NULL;
        if (from)
                sal_op_set_from(op,from);
        if (to)
@@ -85,6 +164,15 @@ int sal_subscribe_presence(SalOp *op, const char *from, const char *to){
        eXosip_lock();
        eXosip_subscribe_build_initial_request(&msg,sal_op_get_to(op),sal_op_get_from(op),
                sal_op_get_route(op),"presence",600);
+       if (msg==NULL){
+               ms_error("Could not build subscribe request to %s",to);
+               eXosip_unlock();
+               return -1;
+       }
+       if (op->base.contact){
+               _osip_list_set_empty(&msg->contacts,(void (*)(void*))osip_contact_free);
+               osip_message_set_contact(msg,op->base.contact);
+       }
        op->sid=eXosip_subscribe_send_initial_request(msg);
        eXosip_unlock();
        if (op->sid==-1){
@@ -113,10 +201,19 @@ int sal_unsubscribe(SalOp *op){
 }
 
 int sal_subscribe_accept(SalOp *op){
-       osip_message_t *msg;
+       osip_message_t *msg=NULL;
        eXosip_lock();
-       eXosip_insubscription_build_answer(op->tid,200,&msg);
-       eXosip_insubscription_send_answer(op->tid,200,msg);
+       eXosip_insubscription_build_answer(op->tid,202,&msg);
+       if (msg==NULL){
+               ms_error("Fail to build answer to subscribe.");
+               eXosip_unlock();
+               return -1;
+       }
+       if (op->base.contact){
+               _osip_list_set_empty(&msg->contacts,(void (*)(void*))osip_contact_free);
+               osip_message_set_contact(msg,op->base.contact);
+       }
+       eXosip_insubscription_send_answer(op->tid,202,msg);
        eXosip_unlock();
        return 0;
 }
@@ -128,274 +225,378 @@ int sal_subscribe_decline(SalOp *op){
        return 0;
 }
 
-static void add_presence_body(osip_message_t *notify, SalPresenceStatus 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==SalPresenceOnline)
-    {
-      sprintf(buf, "<?xml version=\"1.0\"?>\n\
-<!DOCTYPE presence\n\
-PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n\
-<presence>\n\
-<presentity uri=\"%s;method=SUBSCRIBE\" />\n\
-<atom id=\"%i\">\n\
-<address uri=\"%s;user=ip\" priority=\"0.800000\">\n\
-<status status=\"open\" />\n\
-<msnsubstatus substatus=\"online\" />\n\
-</address>\n\
-</atom>\n\
-</presence>", contact_info, atom_id, contact_info);
-
-    }
-  else if (online_status==SalPresenceBusy)
-    {
-      sprintf(buf, "<?xml version=\"1.0\"?>\n\
-<!DOCTYPE presence\n\
-PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n\
-<presence>\n\
-<presentity uri=\"%s;method=SUBSCRIBE\" />\n\
-<atom id=\"%i\">\n\
-<address uri=\"%s;user=ip\" priority=\"0.800000\">\n\
-<status status=\"inuse\" />\n\
-<msnsubstatus substatus=\"busy\" />\n\
-</address>\n\
-</atom>\n\
-</presence>", contact_info, atom_id, contact_info);
-
-    }
-  else if (online_status==SalPresenceBerightback)
-    {
-      sprintf(buf, "<?xml version=\"1.0\"?>\n\
-<!DOCTYPE presence\n\
-PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n\
-<presence>\n\
-<presentity uri=\"%s;method=SUBSCRIBE\" />\n\
-<atom id=\"%i\">\n\
-<address uri=\"%s;user=ip\" priority=\"0.800000\">\n\
-<status status=\"inactive\" />\n\
-<msnsubstatus substatus=\"berightback\" />\n\
-</address>\n\
-</atom>\n\
-</presence>", contact_info, atom_id, contact_info);
+static void mk_presence_body (const SalPresenceStatus online_status, const char *contact_info,
+               char *buf, size_t buflen, presence_type_t ptype) {
+  switch (ptype) {
+    case RFCxxxx: {
+         /* definition from http://msdn.microsoft.com/en-us/library/cc246202%28PROT.10%29.aspx */
+         int atom_id = 1000;
+
+         if (online_status==SalPresenceOnline)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\" priority=\"0.800000\">\n"
+"<status status=\"open\" />\n"
+"<msnsubstatus substatus=\"online\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status == SalPresenceBusy ||
+                         online_status == SalPresenceDonotdisturb)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\" priority=\"0.800000\">\n"
+"<status status=\"inuse\" />\n"
+"<msnsubstatus substatus=\"busy\" />\n"
+"</address>\n"
+"</atom>\n</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status==SalPresenceBerightback)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\" priority=\"0.800000\">\n"
+"<status status=\"open\" />\n"
+"<msnsubstatus substatus=\"berightback\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status == SalPresenceAway ||
+                         online_status == SalPresenceMoved)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\" priority=\"0.800000\">\n"
+"<status status=\"open\" />\n"
+"<msnsubstatus substatus=\"away\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status==SalPresenceOnthephone)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\" priority=\"0.800000\">\n"
+"<status status=\"inuse\" />\n"
+"<msnsubstatus substatus=\"onthephone\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status==SalPresenceOuttolunch)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\" priority=\"0.800000\">\n"
+"<status status=\"open\" />\n"
+"<msnsubstatus substatus=\"outtolunch\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\" priority=\"0.800000\">\n"
+"<status status=\"closed\" />\n"
+"<msnsubstatus substatus=\"away\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+         }
+         break;
+    } 
+    case MSOLDPRES: {
+       /* Couldn't find schema http://schemas.microsoft.com/2002/09/sip/presence
+       *  so messages format has been taken from Communigate that can send notify
+       *  requests with this schema
+       */
+         int atom_id = 1000;
+
+         if (online_status==SalPresenceOnline)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence SYSTEM \"http://schemas.microsoft.com/2002/09/sip/presence\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\">\n"
+"<status status=\"open\" />\n"
+"<msnsubstatus substatus=\"online\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status == SalPresenceBusy ||
+                         online_status == SalPresenceDonotdisturb)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence SYSTEM \"http://schemas.microsoft.com/2002/09/sip/presence\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\">\n"
+"<status status=\"inuse\" />\n"
+"<msnsubstatus substatus=\"busy\" />\n"
+"</address>\n"
+"</atom>\n</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status==SalPresenceBerightback)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence SYSTEM \"http://schemas.microsoft.com/2002/09/sip/presence\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\">\n"
+"<status status=\"inactive\" />\n"
+"<msnsubstatus substatus=\"berightback\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status == SalPresenceAway ||
+                         online_status == SalPresenceMoved)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence SYSTEM \"http://schemas.microsoft.com/2002/09/sip/presence\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\">\n"
+"<status status=\"inactive\" />\n"
+"<msnsubstatus substatus=\"idle\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status==SalPresenceOnthephone)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence SYSTEM \"http://schemas.microsoft.com/2002/09/sip/presence\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\">\n"
+"<status status=\"inuse\" />\n"
+"<msnsubstatus substatus=\"onthephone\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else if (online_status==SalPresenceOuttolunch)
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence SYSTEM \"http://schemas.microsoft.com/2002/09/sip/presence\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\">\n"
+"<status status=\"inactive\" />\n"
+"<msnsubstatus substatus=\"outtolunch\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+
+         }
+         else
+         {
+                 snprintf(buf, buflen, "<?xml version=\"1.0\"?>\n"
+"<!DOCTYPE presence SYSTEM \"http://schemas.microsoft.com/2002/09/sip/presence\">\n"
+"<presence>\n"
+"<presentity uri=\"%s;method=SUBSCRIBE\" />\n"
+"<atom id=\"%i\">\n"
+"<address uri=\"%s\">\n"
+"<status status=\"closed\" />\n"
+"<msnsubstatus substatus=\"offline\" />\n"
+"</address>\n"
+"</atom>\n"
+"</presence>", contact_info, atom_id, contact_info);
+         }
+       break;
+       }
+    default: { /* use pidf+xml as default format, rfc4479, rfc4480, rfc3863 */
 
+       if (online_status==SalPresenceOnline)
+       {
+               snprintf(buf, buflen, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+"<presence xmlns=\"urn:ietf:params:xml:ns:pidf\" "
+"xmlns:dm=\"urn:ietf:params:xml:ns:pidf:data-model\" "
+"xmlns:rpid=\"urn:ietf:params:xml:ns:pidf:rpid\" "
+"entity=\"%s\">\n"
+"<tuple id=\"sg89ae\">\n"
+"<status><basic>open</basic></status>\n"
+"<contact priority=\"0.8\">%s</contact>\n"
+"</tuple>\n"
+"</presence>",
+contact_info, contact_info);
+       }
+       else if (online_status == SalPresenceBusy ||
+                       online_status == SalPresenceDonotdisturb)
+       {
+               snprintf(buf, buflen, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+"<presence xmlns=\"urn:ietf:params:xml:ns:pidf\" "
+"xmlns:dm=\"urn:ietf:params:xml:ns:pidf:data-model\" "
+"xmlns:rpid=\"urn:ietf:params:xml:ns:pidf:rpid\" "
+"entity=\"%s\">\n"
+"<tuple id=\"sg89ae\">\n"
+"<status><basic>open</basic></status>\n"
+"<contact priority=\"0.8\">%s</contact>\n"
+"</tuple>\n"
+"<dm:person id=\"sg89aep\">\n"
+"<rpid:activities><rpid:busy/></rpid:activities>\n"
+"</dm:person>\n"
+"</presence>",
+contact_info, contact_info);
+       }
+       else if (online_status==SalPresenceBerightback)
+       {
+               snprintf(buf, buflen, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+"<presence xmlns=\"urn:ietf:params:xml:ns:pidf\" "
+"xmlns:dm=\"urn:ietf:params:xml:ns:pidf:data-model\" "
+"xmlns:rpid=\"urn:ietf:params:xml:ns:pidf:rpid\" "
+"entity=\"%s\">\n"
+"<tuple id=\"sg89ae\">\n"
+"<status><basic>open</basic></status>\n"
+"<contact priority=\"0.8\">%s</contact>\n"
+"</tuple>\n"
+"<dm:person id=\"sg89aep\">\n"
+"<rpid:activities><rpid:in-transit/></rpid:activities>\n"
+"</dm:person>\n"
+"</presence>",
+contact_info, contact_info);
+       }
+       else if (online_status == SalPresenceAway ||
+                       online_status == SalPresenceMoved)
+       {
+               snprintf(buf, buflen, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+"<presence xmlns=\"urn:ietf:params:xml:ns:pidf\" "
+"xmlns:dm=\"urn:ietf:params:xml:ns:pidf:data-model\" "
+"xmlns:rpid=\"urn:ietf:params:xml:ns:pidf:rpid\" "
+"entity=\"%s\">\n"
+"<tuple id=\"sg89ae\">\n"
+"<status><basic>open</basic></status>\n"
+"<contact priority=\"0.8\">%s</contact>\n"
+"</tuple>\n"
+"<dm:person id=\"sg89aep\">\n"
+"<rpid:activities><rpid:away/></rpid:activities>\n"
+"</dm:person>\n"
+"</presence>",
+contact_info, contact_info);
+       }
+       else if (online_status==SalPresenceOnthephone)
+       {
+               snprintf(buf, buflen, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+"<presence xmlns=\"urn:ietf:params:xml:ns:pidf\" "
+"xmlns:dm=\"urn:ietf:params:xml:ns:pidf:data-model\" "
+"xmlns:rpid=\"urn:ietf:params:xml:ns:pidf:rpid\" "
+"entity=\"%s\">\n"
+"<tuple id=\"sg89ae\">\n"
+"<status><basic>open</basic></status>\n"
+"<contact priority=\"0.8\">%s</contact>\n"
+"</tuple>\n"
+"<dm:person id=\"sg89aep\">\n"
+"<rpid:activities><rpid:on-the-phone/></rpid:activities>\n"
+"</dm:person>\n"
+"</presence>",
+contact_info, contact_info);
+       }
+       else if (online_status==SalPresenceOuttolunch)
+       {
+               snprintf(buf, buflen, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+"<presence xmlns=\"urn:ietf:params:xml:ns:pidf\" "
+"xmlns:dm=\"urn:ietf:params:xml:ns:pidf:data-model\" "
+"xmlns:rpid=\"urn:ietf:params:xml:ns:pidf:rpid\" "
+"entity=\"%s\">\n"
+"<tuple id=\"7777\">\n"
+"<status><basic>open</basic></status>\n"
+"<contact priority=\"0.8\">%s</contact>\n"
+"</tuple>\n"
+"<dm:person id=\"78787878\">\n"
+"<rpid:activities><rpid:meal/></rpid:activities>\n"
+"<rpid:note>Out to lunch</rpid:note> \n"
+"</dm:person>\n"
+"</presence>",
+contact_info, contact_info);
+       }
+       else
+       {
+               snprintf(buf, buflen, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+"<presence xmlns=\"urn:ietf:params:xml:ns:pidf\" "
+"xmlns:dm=\"urn:ietf:params:xml:ns:pidf:data-model\" "
+"xmlns:rpid=\"urn:ietf:params:xml:ns:pidf:rpid\" "
+"entity=\"%s\">\n"
+"<tuple id=\"sg89ae\">\n"
+"<status><basic>closed</basic></status>\n"
+"<contact priority=\"0.8\">%s</contact>\n"
+"</tuple>\n"
+"</presence>\n", contact_info, contact_info);
+       }
+       break;
     }
-  else if (online_status==SalPresenceAway)
-    {
-      sprintf(buf, "<?xml version=\"1.0\"?>\n\
-<!DOCTYPE presence\n\
-PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n\
-<presence>\n\
-<presentity uri=\"%s;method=SUBSCRIBE\" />\n\
-<atom id=\"%i\">\n\
-<address uri=\"%s;user=ip\" priority=\"0.800000\">\n\
-<status status=\"inactive\" />\n\
-<msnsubstatus substatus=\"away\" />\n\
-</address>\n\
-</atom>\n\
-</presence>", contact_info, atom_id, contact_info);
+ } // switch
 
-    }
-  else if (online_status==SalPresenceOnthephone)
-    {
-      sprintf(buf, "<?xml version=\"1.0\"?>\n\
-<!DOCTYPE presence\n\
-PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n\
-<presence>\n\
-<presentity uri=\"%s;method=SUBSCRIBE\" />\n\
-<atom id=\"%i\">\n\
-<address uri=\"%s;user=ip\" priority=\"0.800000\">\n\
-<status status=\"inuse\" />\n\
-<msnsubstatus substatus=\"onthephone\" />\n\
-</address>\n\
-</atom>\n\
-</presence>", contact_info, atom_id, contact_info);
+}
 
-    }
-  else if (online_status==SalPresenceOuttolunch)
-    {
-      sprintf(buf, "<?xml version=\"1.0\"?>\n\
-<!DOCTYPE presence\n\
-PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n\
-<presence>\n\
-<presentity uri=\"%s;method=SUBSCRIBE\" />\n\
-<atom id=\"%i\">\n\
-<address uri=\"%s;user=ip\" priority=\"0.800000\">\n\
-<status status=\"inactive\" />\n\
-<msnsubstatus substatus=\"outtolunch\" />\n\
-</address>\n\
-</atom>\n\
-</presence>", contact_info, atom_id, contact_info);
+static void add_presence_body(osip_message_t *notify, SalPresenceStatus online_status)
+{
+       char buf[1000];
+       char *contact_info;
 
-    }
-  else
-    {
-      sprintf(buf, "<?xml version=\"1.0\"?>\n\
-<!DOCTYPE presence\n\
-PUBLIC \"-//IETF//DTD RFCxxxx XPIDF 1.0//EN\" \"xpidf.dtd\">\n\
-<presence>\n\
-<presentity uri=\"%s;method=SUBSCRIBE\" />\n\
-<atom id=\"%i\">\n\
-<address uri=\"%s;user=ip\" priority=\"0.800000\">\n\
-<status status=\"inactive\" />\n\
-<msnsubstatus substatus=\"away\" />\n\
-</address>\n\
-</atom>\n\
-</presence>", contact_info, atom_id, contact_info);
-    }
+       osip_from_t *from=NULL;
+       from=osip_message_get_from(notify);
+       osip_uri_to_str(from->url,&contact_info);
 
-  osip_message_set_body(notify, buf, strlen(buf));
-  osip_message_set_content_type(notify, "application/xpidf+xml");
-#else
+       mk_presence_body (online_status, contact_info, buf, sizeof (buf), presence_style);
 
-  if (online_status==SalPresenceOnline)
-    {
-      sprintf(buf, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-<presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-          entity=\"%s\">\n\
-<tuple id=\"sg89ae\">\n\
-<status>\n\
-<basic>open</basic>\n\
-</status>\n\
-<contact priority=\"0.8\">%s</contact>\n\
-<note>online</note>\n\
-</tuple>\n\
-</presence>",
-             contact_info, contact_info);
-    }
-  else if (online_status==SalPresenceBusy)
-    {
-      sprintf(buf, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-<presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-          xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-          entity=\"%s\">\n\
-<tuple id=\"sg89ae\">\n\
-<status>\n\
-<basic>open</basic>\n\
-<es:activities>\n\
-  <es:activity>busy</es:activity>\n\
-</es:activities>\n\
-</status>\n\
-<contact priority=\"0.8\">%s</contact>\n\
-<note>busy</note>\n\
-</tuple>\n\
-</presence>",
-             contact_info, contact_info);
-    }
-  else if (online_status==SalPresenceBerightback)
-    {
-      sprintf(buf, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-<presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-          xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-          entity=\"%s\">\n\
-<tuple id=\"sg89ae\">\n\
-<status>\n\
-<basic>open</basic>\n\
-<es:activities>\n\
-  <es:activity>in-transit</es:activity>\n\
-</es:activities>\n\
-</status>\n\
-<contact priority=\"0.8\">%s</contact>\n\
-<note>be right back</note>\n\
-</tuple>\n\
-</presence>",
-             contact_info, contact_info);
-    }
-  else if (online_status==SalPresenceAway)
-    {
-      sprintf(buf, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-<presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-          xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-          entity=\"%s\">\n\
-<tuple id=\"sg89ae\">\n\
-<status>\n\
-<basic>open</basic>\n\
-<es:activities>\n\
-  <es:activity>away</es:activity>\n\
-</es:activities>\n\
-</status>\n\
-<contact priority=\"0.8\">%s</contact>\n\
-<note>away</note>\n\
-</tuple>\n\
-</presence>",
-             contact_info, contact_info);
-    }
-  else if (online_status==SalPresenceOnthephone)
-    {
-      sprintf(buf, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-<presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-          xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-          entity=\"%s\">\n\
-<tuple id=\"sg89ae\">\n\
-<status>\n\
-<basic>open</basic>\n\
-<es:activities>\n\
-  <es:activity>on-the-phone</es:activity>\n\
-</es:activities>\n\
-</status>\n\
-<contact priority=\"0.8\">%s</contact>\n\
-<note>on the phone</note>\n\
-</tuple>\n\
-</presence>",
-             contact_info, contact_info);
-    }
-  else if (online_status==SalPresenceOuttolunch)
-    {
-      sprintf(buf, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-<presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-          xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-          entity=\"%s\">\n\
-<tuple id=\"sg89ae\">\n\
-<status>\n\
-<basic>open</basic>\n\
-<es:activities>\n\
-  <es:activity>meal</es:activity>\n\
-</es:activities>\n\
-</status>\n\
-<contact priority=\"0.8\">%s</contact>\n\
-<note>out to lunch</note>\n\
-</tuple>\n\
-</presence>",
-             contact_info, contact_info);
-    }
-  else
-    {
-      /* */
-      sprintf(buf, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-<presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-entity=\"%s\">\n%s",
-             contact_info,
-"<tuple id=\"sg89ae\">\n\
-<status>\n\
-<basic>closed</basic>\n\
-<es:activities>\n\
-  <es:activity>permanent-absence</es:activity>\n\
-</es:activities>\n\
-</status>\n\
-</tuple>\n\
-\n</presence>\n");
-    }
-  osip_message_set_body(notify, buf, strlen(buf));
-  osip_message_set_content_type(notify, "application/pidf+xml");
+       osip_message_set_body(notify, buf, strlen(buf));
+       osip_message_set_content_type(notify,
+               presence_style ? "application/xpidf+xml" : "application/pidf+xml");
 
-#endif
        osip_free(contact_info);
 }
 
 
 int sal_notify_presence(SalOp *op, SalPresenceStatus status, const char *status_message){
-       osip_message_t *msg;
+       osip_message_t *msg=NULL;
        eXosip_ss_t ss=EXOSIP_SUBCRSTATE_ACTIVE;
        if (op->nid==-1){
                ms_warning("Cannot notify, subscription was closed.");
@@ -407,6 +608,7 @@ int sal_notify_presence(SalOp *op, SalPresenceStatus status, const char *status_
        if (msg!=NULL){
                const char *identity=sal_op_get_contact(op);
                if (identity==NULL) identity=sal_op_get_to(op);
+               _osip_list_set_empty(&msg->contacts,(void (*)(void*))osip_contact_free);
                osip_message_set_contact(msg,identity);
                add_presence_body(msg,status);
                eXosip_insubscription_send_request(op->did,msg);
@@ -423,6 +625,7 @@ int sal_notify_close(SalOp *op){
                const char *identity=sal_op_get_contact(op);
                if (identity==NULL) identity=sal_op_get_to(op);
                osip_message_set_contact(msg,identity);
+               add_presence_body(msg,SalPresenceOffline);
                eXosip_insubscription_send_request(op->did,msg);
        }else ms_error("sal_notify_close(): could not create notify for incoming subscription"
            " did=%i, nid=%i",op->did,op->nid);
@@ -434,155 +637,32 @@ int sal_publish(SalOp *op, const char *from, const char *to, SalPresenceStatus p
        osip_message_t *pub;
        int i;
        char buf[1024];
+       const char *route=sal_op_get_route(op);
 
-       if (presence_mode==SalPresenceOnline)
-       {
-         snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-       <presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-                 entity=\"%s\">\n\
-       <tuple id=\"sg89ae\">\n\
-       <status>\n\
-       <basic>open</basic>\n\
-       </status>\n\
-       <contact priority=\"0.8\">%s</contact>\n\
-       <note>online</note>\n\
-       </tuple>\n\
-       </presence>",
-                  from, from);
-       }
-       else if (presence_mode==SalPresenceBusy
-          ||presence_mode==SalPresenceDonotdisturb)
-       {
-         snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-       <presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-                 xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-                 entity=\"%s\">\n\
-       <tuple id=\"sg89ae\">\n\
-       <status>\n\
-       <basic>open</basic>\n\
-       <es:activities>\n\
-       <es:activity>busy</es:activity>\n\
-       </es:activities>\n\
-       </status>\n\
-       <contact priority=\"0.8\">%s</contact>\n\
-       <note>busy</note>\n\
-       </tuple>\n\
-       </presence>",
-                 from, from);
-       }
-       else if (presence_mode==SalPresenceBerightback)
-       {
-               snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-       <presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-                 xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-                 entity=\"%s\">\n\
-       <tuple id=\"sg89ae\">\n\
-       <status>\n\
-       <basic>open</basic>\n\
-       <es:activities>\n\
-       <es:activity>in-transit</es:activity>\n\
-       </es:activities>\n\
-       </status>\n\
-       <contact priority=\"0.8\">%s</contact>\n\
-       <note>be right back</note>\n\
-       </tuple>\n\
-       </presence>",
-                 from,from);
-       }
-       else if (presence_mode==SalPresenceAway
-          ||presence_mode==SalPresenceMoved)
-       {
-               snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-       <presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-                 xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-                 entity=\"%s\">\n\
-       <tuple id=\"sg89ae\">\n\
-       <status>\n\
-       <basic>open</basic>\n\
-       <es:activities>\n\
-       <es:activity>away</es:activity>\n\
-       </es:activities>\n\
-       </status>\n\
-       <contact priority=\"0.8\">%s</contact>\n\
-       <note>away</note>\n\
-       </tuple>\n\
-       </presence>",
-                 from, from);
-       }
-       else if (presence_mode==SalPresenceOnthephone)
-       {
-         snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-       <presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-                 xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-                 entity=\"%s\">\n\
-       <tuple id=\"sg89ae\">\n\
-       <status>\n\
-       <basic>open</basic>\n\
-       <es:activities>\n\
-       <es:activity>on-the-phone</es:activity>\n\
-       </es:activities>\n\
-       </status>\n\
-       <contact priority=\"0.8\">%s</contact>\n\
-       <note>on the phone</note>\n\
-       </tuple>\n\
-       </presence>",
-                 from, from);
-       }
-       else if (presence_mode==SalPresenceOuttolunch)
-       {
-         snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-       <presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-                 xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-                 entity=\"%s\">\n\
-       <tuple id=\"sg89ae\">\n\
-       <status>\n\
-       <basic>open</basic>\n\
-       <es:activities>\n\
-       <es:activity>meal</es:activity>\n\
-       </es:activities>\n\
-       </status>\n\
-       <contact priority=\"0.8\">%s</contact>\n\
-       <note>out to lunch</note>\n\
-       </tuple>\n\
-       </presence>",
-                 from, from);
-       }
-       else{ 
-         /* offline */
-         snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\
-       <presence xmlns=\"urn:ietf:params:xml:ns:pidf\"\n\
-       xmlns:es=\"urn:ietf:params:xml:ns:pidf:status:rpid-status\"\n\
-       entity=\"%s\">\n%s",
-                 from,
-       "<tuple id=\"sg89ae\">\n\
-       <status>\n\
-       <basic>closed</basic>\n\
-       <es:activities>\n\
-       <es:activity>permanent-absence</e:activity>\n\
-       </es:activities>\n\
-       </status>\n\
-       </tuple>\n\
-       \n</presence>\n");
-       }
+       mk_presence_body (presence_mode, from, buf, sizeof (buf), presence_style);
 
-       i = eXosip_build_publish(&pub,from, to, NULL, "presence", "1800", "application/pidf+xml", buf);
+       i = eXosip_build_publish(&pub,to, from, NULL, "presence", "600", 
+               presence_style ? "application/xpidf+xml" : "application/pidf+xml", buf);
        if (i<0){
                ms_warning("Failed to build publish request.");
                return -1;
        }
-
+       if (route)
+               sal_message_add_route(pub,route);
+       
        eXosip_lock();
        i = eXosip_publish(pub, to); /* should update the sip-if-match parameter
                                    from sip-etag  from last 200ok of PUBLISH */
        eXosip_unlock();
        if (i<0){
-         ms_message("Failed to send publish request.");
-         return -1;
+               ms_message("Failed to send publish request.");
+               return -1;
        }
+       sal_add_other(sal_op_get_sal(op),op,pub);
        return 0;
 }
 
-void sal_exosip_subscription_recv(Sal *sal, eXosip_event_t *ev){
+static void _sal_exosip_subscription_recv(Sal *sal, eXosip_event_t *ev){       
        SalOp *op=sal_op_new(sal);
        char *tmp;
        op->did=ev->did;
@@ -594,10 +674,33 @@ void sal_exosip_subscription_recv(Sal *sal, eXosip_event_t *ev){
        osip_from_to_str(ev->request->to,&tmp);
        sal_op_set_to(op,tmp);
        ms_free(tmp);
-       sal_add_in_subscribe(sal,op);
+       sal_add_in_subscribe(sal,op,ev->request);
        sal->callbacks.subscribe_received(op,sal_op_get_from(op));
 }
 
+void sal_exosip_subscription_recv(Sal *sal, eXosip_event_t *ev){       
+       /*workaround a bug in eXosip: incoming SUBSCRIBES within dialog with expires: 0 are
+        recognized as new incoming subscribes*/
+       SalOp *op=sal_find_in_subscribe_by_call_id(sal,ev->request->call_id);
+       if (op){
+               osip_header_t *h;
+               osip_message_header_get_byname(ev->request,"expires",0,&h);
+               if (h && h->hvalue && atoi(h->hvalue)==0){
+                       ms_warning("This susbscribe is not a new one but terminates an old one.");
+                       ev->did=op->did;
+                       ev->nid=op->nid;
+                       sal_exosip_subscription_closed(sal,ev);
+               }else {
+                       osip_message_t *msg=NULL;
+                       ms_warning("Probably a refresh subscribe");
+                       eXosip_lock();
+                       eXosip_insubscription_build_answer(ev->tid,202,&msg);
+                       eXosip_insubscription_send_answer(ev->tid,202,msg);
+                       eXosip_unlock();
+               }
+       }else _sal_exosip_subscription_recv(sal,ev);
+}
+
 void sal_exosip_notify_recv(Sal *sal, eXosip_event_t *ev){
        SalOp *op=sal_find_out_subscribe(sal,ev->sid);
        char *tmp;
@@ -627,7 +730,8 @@ void sal_exosip_notify_recv(Sal *sal, eXosip_event_t *ev){
        }else if (strstr(body->body,"berightback")!=NULL
                        || strstr(body->body,"in-transit")!=NULL ){
                estatus=SalPresenceBerightback;
-       }else if (strstr(body->body,"away")!=NULL){
+       }else if (strstr(body->body,"away")!=NULL
+                       || strstr(body->body,"idle")){
                estatus=SalPresenceAway;
        }else if (strstr(body->body,"onthephone")!=NULL
                || strstr(body->body,"on-the-phone")!=NULL){
@@ -649,7 +753,16 @@ void sal_exosip_notify_recv(Sal *sal, eXosip_event_t *ev){
                op->did=-1;
                ms_message("And outgoing subscription terminated by remote.");
        }
-       sal->callbacks.notify(op,op->sid!=-1 ? SalSubscribeActive : SalSubscribeTerminated, estatus,NULL);
+       sal->callbacks.notify_presence(op,op->sid!=-1 ? SalSubscribeActive : SalSubscribeTerminated, estatus,NULL);
+
+       /* try to detect presence message style used by server,
+        * and switch our presence messages to servers style */
+       if (strstr (body->body, "//IETF//DTD RFCxxxx XPIDF 1.0//EN") != NULL) {
+               presence_style = RFCxxxx;
+       } else if (strstr(body->body,"http://schemas.microsoft.com/2002/09/sip/presence")!=NULL) {
+               presence_style = MSOLDPRES;
+       }
+       
        osip_free(tmp);
 }
 
@@ -690,7 +803,7 @@ void sal_exosip_subscription_closed(Sal *sal,eXosip_event_t *ev){
        sal_remove_out_subscribe(sal,op);
        op->sid=-1;
        op->did=-1;
-       sal->callbacks.notify(op,SalSubscribeTerminated, SalPresenceOffline,NULL);
+       sal->callbacks.notify_presence(op,SalSubscribeTerminated, SalPresenceOffline,NULL);
 }