summaryrefslogtreecommitdiff
path: root/pjsip/src/pjsip-simple
diff options
context:
space:
mode:
authorBenny Prijono <bennylp@teluu.com>2006-03-02 21:18:58 +0000
committerBenny Prijono <bennylp@teluu.com>2006-03-02 21:18:58 +0000
commitaad76a0bb1ea62caaf67e6fecaea5786b5f2811a (patch)
tree8183f5a20882579ad310e060115a832a2e6eb444 /pjsip/src/pjsip-simple
parentcf626e119a5f0b8b424c7cf473f514f1710ce209 (diff)
Added IM and composition indication, and tested
git-svn-id: http://svn.pjsip.org/repos/pjproject/trunk@268 74dad513-b988-da41-8d7b-12977e46ad98
Diffstat (limited to 'pjsip/src/pjsip-simple')
-rw-r--r--pjsip/src/pjsip-simple/errno.c5
-rw-r--r--pjsip/src/pjsip-simple/evsub.c21
-rw-r--r--pjsip/src/pjsip-simple/iscomposing.c215
-rw-r--r--pjsip/src/pjsip-simple/presence.c63
4 files changed, 285 insertions, 19 deletions
diff --git a/pjsip/src/pjsip-simple/errno.c b/pjsip/src/pjsip-simple/errno.c
index e26b5a7b..66d86767 100644
--- a/pjsip/src/pjsip-simple/errno.c
+++ b/pjsip/src/pjsip-simple/errno.c
@@ -29,15 +29,20 @@ static const struct
const char *msg;
} err_str[] =
{
+ /* Event errors */
{ PJSIP_SIMPLE_ENOPKG, "No SIP event package with the specified name" },
{ PJSIP_SIMPLE_EPKGEXISTS, "SIP event package already exist" },
+ /* Presence errors */
{ PJSIP_SIMPLE_ENOTSUBSCRIBE, "Expecting SUBSCRIBE request" },
{ PJSIP_SIMPLE_ENOPRESENCE, "No presence associated with the subscription" },
{ PJSIP_SIMPLE_ENOPRESENCEINFO, "No presence info in the server subscription" },
{ PJSIP_SIMPLE_EBADCONTENT, "Bad Content-Type for presence" },
{ PJSIP_SIMPLE_EBADPIDF, "Bad PIDF content for presence" },
{ PJSIP_SIMPLE_EBADXPIDF, "Bad XPIDF content for presence" },
+
+ /* isComposing errors. */
+ { PJSIP_SIMPLE_EBADISCOMPOSE, "Bad isComposing indication/XML message" },
};
diff --git a/pjsip/src/pjsip-simple/evsub.c b/pjsip/src/pjsip-simple/evsub.c
index 874f9204..0f0b974c 100644
--- a/pjsip/src/pjsip-simple/evsub.c
+++ b/pjsip/src/pjsip-simple/evsub.c
@@ -956,6 +956,15 @@ PJ_DEF(pj_status_t) pjsip_evsub_accept( pjsip_evsub *sub,
pjsip_msg_add_hdr( tdata->msg,
pjsip_hdr_shallow_clone(tdata->pool, sub->expires));
+ /* Add additional header, if any. */
+ if (hdr_list) {
+ const pjsip_hdr *hdr = hdr_list->next;
+ while (hdr != hdr_list) {
+ pjsip_msg_add_hdr( tdata->msg,
+ pjsip_hdr_clone(tdata->pool, hdr));
+ hdr = hdr->next;
+ }
+ }
/* Send the response: */
status = pjsip_dlg_send_response( sub->dlg, tsx, tdata );
@@ -1324,13 +1333,11 @@ static pj_status_t create_response( pjsip_evsub *sub,
/* Add msg body, if any */
if (body) {
- tdata->msg->body = pj_pool_zalloc(tdata->pool,
- sizeof(pjsip_msg_body));
- status = pjsip_msg_body_clone(tdata->pool,
- tdata->msg->body,
- body);
- if (status != PJ_SUCCESS) {
- tdata->msg->body = NULL;
+ tdata->msg->body = pjsip_msg_body_clone(tdata->pool, body);
+ if (tdata->msg->body == NULL) {
+
+ PJ_LOG(4,(THIS_FILE, "Error: unable to clone msg body"));
+
/* Ignore */
return PJ_SUCCESS;
}
diff --git a/pjsip/src/pjsip-simple/iscomposing.c b/pjsip/src/pjsip-simple/iscomposing.c
new file mode 100644
index 00000000..369281af
--- /dev/null
+++ b/pjsip/src/pjsip-simple/iscomposing.c
@@ -0,0 +1,215 @@
+/* $Id$ */
+/*
+ * Copyright (C) 2003-2006 Benny Prijono <benny@prijono.org>
+ *
+ * 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 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 <pjsip-simple/iscomposing.h>
+#include <pjsip-simple/errno.h>
+#include <pjsip/sip_msg.h>
+#include <pjlib-util/errno.h>
+#include <pj/pool.h>
+#include <pj/string.h>
+
+
+/* MIME */
+static const pj_str_t STR_MIME_TYPE = { "application", 11 };
+static const pj_str_t STR_MIME_SUBTYPE = { "im-iscomposing+xml", 18 };
+
+
+/* XML node constants. */
+static const pj_str_t STR_ISCOMPOSING = { "isComposing", 11 };
+static const pj_str_t STR_STATE = { "state", 5 };
+static const pj_str_t STR_ACTIVE = { "active", 6 };
+static const pj_str_t STR_IDLE = { "idle", 4 };
+static const pj_str_t STR_LASTACTIVE = { "lastactive", 10 };
+static const pj_str_t STR_CONTENTTYPE = { "contenttype", 11 };
+static const pj_str_t STR_REFRESH = { "refresh", 7 };
+
+
+/* XML attributes constants */
+static const pj_str_t STR_XMLNS_NAME = { "xmlns", 5 };
+static const pj_str_t STR_XMLNS_VAL = { "urn:ietf:params:xml:ns:im-iscomposing", 37 };
+static const pj_str_t STR_XMLNS_XSI_NAME = { "xmlns:xsi", 9 };
+static const pj_str_t STR_XMLNS_XSI_VAL = { "http://www.w3.org/2001/XMLSchema-instance", 41 };
+static const pj_str_t STR_XSI_SLOC_NAME = { "xsi:schemaLocation", 18 };
+static const pj_str_t STR_XSI_SLOC_VAL = { "urn:ietf:params:xml:ns:im-composing iscomposing.xsd", 51 };
+
+
+PJ_DEF(pj_xml_node*) pjsip_iscomposing_create_xml( pj_pool_t *pool,
+ pj_bool_t is_composing,
+ const pj_time_val *lst_actv,
+ const pj_str_t *content_tp,
+ int refresh)
+{
+ pj_xml_node *doc, *node;
+ pj_xml_attr *attr;
+
+ /* Root document. */
+ doc = pj_xml_node_new(pool, &STR_ISCOMPOSING);
+
+ /* Add attributes */
+ attr = pj_xml_attr_new(pool, &STR_XMLNS_NAME, &STR_XMLNS_VAL);
+ pj_xml_add_attr(doc, attr);
+
+ attr = pj_xml_attr_new(pool, &STR_XMLNS_XSI_NAME, &STR_XMLNS_XSI_VAL);
+ pj_xml_add_attr(doc, attr);
+
+ attr = pj_xml_attr_new(pool, &STR_XSI_SLOC_NAME, &STR_XSI_SLOC_VAL);
+ pj_xml_add_attr(doc, attr);
+
+
+ /* Add state. */
+ node = pj_xml_node_new(pool, &STR_STATE);
+ if (is_composing)
+ node->content = STR_ACTIVE;
+ else
+ node->content = STR_IDLE;
+ pj_xml_add_node(doc, node);
+
+ /* Add lastactive, if any. */
+ if (!is_composing && lst_actv) {
+ PJ_TODO(IMPLEMENT_LAST_ACTIVE_ATTRIBUTE);
+ }
+
+ /* Add contenttype, if any. */
+ if (content_tp) {
+ node = pj_xml_node_new(pool, &STR_CONTENTTYPE);
+ pj_strdup(pool, &node->content, content_tp);
+ pj_xml_add_node(doc, node);
+ }
+
+ /* Add refresh, if any. */
+ if (is_composing && refresh > 1 && refresh < 3601) {
+ node = pj_xml_node_new(pool, &STR_REFRESH);
+ node->content.ptr = pj_pool_alloc(pool, 10);
+ node->content.slen = pj_utoa(refresh, node->content.ptr);
+ pj_xml_add_node(doc, node);
+ }
+
+ /* Done! */
+
+ return doc;
+}
+
+
+
+/*
+ * Function to print XML message body.
+ */
+static int xml_print_body( struct pjsip_msg_body *msg_body,
+ char *buf, pj_size_t size)
+{
+ return pj_xml_print(msg_body->data, buf, size, PJ_TRUE);
+}
+
+
+/*
+ * Function to clone XML document.
+ */
+static void* xml_clone_data(pj_pool_t *pool, const void *data, unsigned len)
+{
+ PJ_UNUSED_ARG(len);
+ return pj_xml_clone( pool, data);
+}
+
+
+
+PJ_DEF(pjsip_msg_body*) pjsip_iscomposing_create_body( pj_pool_t *pool,
+ pj_bool_t is_composing,
+ const pj_time_val *lst_actv,
+ const pj_str_t *content_tp,
+ int refresh)
+{
+ pj_xml_node *doc;
+ pjsip_msg_body *body;
+
+ doc = pjsip_iscomposing_create_xml( pool, is_composing, lst_actv,
+ content_tp, refresh);
+ if (doc == NULL)
+ return NULL;
+
+
+ body = pj_pool_zalloc(pool, sizeof(pjsip_msg_body));
+ body->content_type.type = STR_MIME_TYPE;
+ body->content_type.subtype = STR_MIME_SUBTYPE;
+
+ body->data = doc;
+ body->len = 0;
+
+ body->print_body = &xml_print_body;
+ body->clone_data = &xml_clone_data;
+
+ return body;
+}
+
+
+PJ_DEF(pj_status_t) pjsip_iscomposing_parse( pj_pool_t *pool,
+ char *msg,
+ pj_size_t len,
+ pj_bool_t *p_is_composing,
+ pj_str_t **p_last_active,
+ pj_str_t **p_content_type,
+ int *p_refresh )
+{
+ pj_xml_node *doc, *node;
+
+ /* Set defaults: */
+ if (p_is_composing) *p_is_composing = PJ_FALSE;
+ if (p_last_active) *p_last_active = NULL;
+ if (p_content_type) *p_content_type = NULL;
+
+ /* Parse XML */
+ doc = pj_xml_parse( pool, msg, len);
+ if (!doc)
+ return PJLIB_UTIL_EINXML;
+
+ /* Root document must be "isComposing" */
+ if (pj_stricmp(&doc->name, &STR_ISCOMPOSING) != 0)
+ return PJSIP_SIMPLE_EBADISCOMPOSE;
+
+ /* Get the status. */
+ if (p_is_composing) {
+ node = pj_xml_find_node(doc, &STR_STATE);
+ if (node == NULL)
+ return PJSIP_SIMPLE_EBADISCOMPOSE;
+ *p_is_composing = (pj_stricmp(&node->content, &STR_ACTIVE)==0);
+ }
+
+ /* Get last active. */
+ if (p_last_active) {
+ node = pj_xml_find_node(doc, &STR_LASTACTIVE);
+ if (node)
+ *p_last_active = &node->content;
+ }
+
+ /* Get content type */
+ if (p_content_type) {
+ node = pj_xml_find_node(doc, &STR_CONTENTTYPE);
+ if (node)
+ *p_content_type = &node->content;
+ }
+
+ /* Get refresh */
+ if (p_refresh) {
+ node = pj_xml_find_node(doc, &STR_REFRESH);
+ if (node)
+ *p_refresh = pj_strtoul(&node->content);
+ }
+
+ return PJ_SUCCESS;
+}
+
+
diff --git a/pjsip/src/pjsip-simple/presence.c b/pjsip/src/pjsip-simple/presence.c
index b2cdcf34..f8014fce 100644
--- a/pjsip/src/pjsip-simple/presence.c
+++ b/pjsip/src/pjsip-simple/presence.c
@@ -786,21 +786,22 @@ static pj_status_t pres_parse_xpidf( pjsip_pres *pres,
/*
- * Called when NOTIFY is received.
+ * Process the content of incoming NOTIFY request and update temporary
+ * status.
+ *
+ * return PJ_SUCCESS if incoming request is acceptable. If return value
+ * is not PJ_SUCCESS, res_hdr may be added with Warning header.
*/
-static void pres_on_evsub_rx_notify( pjsip_evsub *sub,
- pjsip_rx_data *rdata,
- int *p_st_code,
- pj_str_t **p_st_text,
- pjsip_hdr *res_hdr,
- pjsip_msg_body **p_body)
+static pj_status_t pres_process_rx_notify( pjsip_pres *pres,
+ pjsip_rx_data *rdata,
+ int *p_st_code,
+ pj_str_t **p_st_text,
+ pjsip_hdr *res_hdr)
{
pjsip_ctype_hdr *ctype_hdr;
- pjsip_pres *pres;
pj_status_t status;
- pres = pjsip_evsub_get_mod_data(sub, mod_presence.id);
- PJ_ASSERT_ON_FAIL(pres!=NULL, {return;});
+ *p_st_text = NULL;
/* Check Content-Type and msg body are present. */
ctype_hdr = rdata->msg_info.ctype;
@@ -818,7 +819,7 @@ static void pres_on_evsub_rx_notify( pjsip_evsub *sub,
&warn_text);
pj_list_push_back(res_hdr, warn_hdr);
- return;
+ return PJSIP_ERRNO_FROM_SIP_STATUS(PJSIP_SC_BAD_REQUEST);
}
/* Parse content. */
@@ -859,7 +860,7 @@ static void pres_on_evsub_rx_notify( pjsip_evsub *sub,
status);
pj_list_push_back(res_hdr, warn_hdr);
- return;
+ return status;
}
/* If application calls pres_get_status(), redirect the call to
@@ -867,6 +868,44 @@ static void pres_on_evsub_rx_notify( pjsip_evsub *sub,
*/
pres->tmp_status._is_valid = PJ_TRUE;
+ return PJ_SUCCESS;
+}
+
+
+/*
+ * Called when NOTIFY is received.
+ */
+static void pres_on_evsub_rx_notify( pjsip_evsub *sub,
+ pjsip_rx_data *rdata,
+ int *p_st_code,
+ pj_str_t **p_st_text,
+ pjsip_hdr *res_hdr,
+ pjsip_msg_body **p_body)
+{
+ pjsip_pres *pres;
+ pj_status_t status;
+
+ pres = pjsip_evsub_get_mod_data(sub, mod_presence.id);
+ PJ_ASSERT_ON_FAIL(pres!=NULL, {return;});
+
+ /* Only process the message body if it exists, otherwise treat as
+ * presence status is closed.
+ */
+ if (rdata->msg_info.msg->body) {
+ status = pres_process_rx_notify( pres, rdata, p_st_code, p_st_text,
+ res_hdr );
+ if (status != PJ_SUCCESS)
+ return;
+
+ } else {
+ unsigned i;
+
+ /* Subscription is terminated. Consider contact is offline */
+ pres->tmp_status._is_valid = PJ_TRUE;
+ for (i=0; i<pres->tmp_status.info_cnt; ++i)
+ pres->tmp_status.info[i].basic_open = PJ_FALSE;
+ }
+
/* Notify application. */
if (pres->user_cb.on_rx_notify) {
(*pres->user_cb.on_rx_notify)(sub, rdata, p_st_code, p_st_text,