123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427 |
- /*
- * $Id$
- *
- * Copyright (C) 2001-2003 FhG Fokus
- *
- * This file is part of Kamailio, a free SIP server.
- *
- * Kamailio 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
- *
- * Kamailio 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- */
- #include "msfuncs.h"
- #include <stdio.h>
- #include <stdlib.h>
- #include <sys/types.h>
- #include <unistd.h>
- #include <time.h>
- #include "../../dprint.h"
- #include "../../config.h"
- #include "../../ut.h"
- #include "../../forward.h"
- #include "../../resolve.h"
- #include "../../globals.h"
- #include "../../udp_server.h"
- #include "../../pt.h"
- #define CONTACT_PREFIX "Contact: <"
- #define CONTACT_SUFFIX ">;msilo=yes"CRLF
- #define CONTACT_PREFIX_LEN (sizeof(CONTACT_PREFIX)-1)
- #define CONTACT_SUFFIX_LEN (sizeof(CONTACT_SUFFIX)-1)
- extern int ms_add_date;
- extern int ms_add_contact;
- /**
- * apostrophes escaping
- * - src: source buffer
- * - slen: length of source buffer
- * - dst: destination buffer
- * - dlen: max length of destination buffer
- * return: destination length => OK; -1 => error
- */
- int m_apo_escape(char* src, int slen, char* dst, int dlen)
- {
- int i, j;
- if(!src || !dst || dlen <= 0)
- return -1;
- if(slen == -1)
- slen = strlen(src);
- for(i=j=0; i<slen; i++)
- {
- switch(src[i])
- {
- case '\'':
- if(j+2>=dlen)
- return -2;
- memcpy(&dst[j], "\\'", 2);
- j += 2;
- break;
- default:
- if(j+1>=dlen)
- return -2;
- dst[j] = src[i];
- j++;
- }
- }
- dst[j] = '\0';
- return j;
- }
- /**
- * Build a RFC 3261 compliant Date string from a time_t value
- * - date: input of time_t to build the string from
- * - buf: pointer to string for output
- * - bufLen: length of buf param
- *
- * return: >0 length of data copied to buf ; <0 error occured
- */
- int timetToSipDateStr(time_t date, char* buf, int bufLen)
- {
- struct tm *gmt;
- char* dayArray[7] = {"Sun","Mon","Tue","Wed","Thu","Fri","Sat"};
- char* monthArray[12] = {"Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"};
- int len = 0;
- gmt = gmtime(&date);
- /* In RFC 3261 the format is always GMT and in the string form like
- * "Wkday, Day Month Year HOUR:MIN:SEC GMT"
- * "Mon, 19 Feb 2007 18:42:27 GMT"
- */
- len = snprintf(buf,bufLen,"Date: %s, %02d %s %d %02d:%02d:%02d GMT\r\n",
- dayArray[gmt->tm_wday],
- gmt->tm_mday,
- monthArray[gmt->tm_mon],
- 1900 + gmt->tm_year,
- gmt->tm_hour,
- gmt->tm_min,
- gmt->tm_sec
- );
- /* snprintf returns number of chars it should have printed, so you
- * need to bounds check against input*/
- return (len > bufLen) ? bufLen : len;
- }
- /**
- * extract the value of Content-Type header
- * - src: pointer to C-T content
- * - len: length of src
- * - ctype: parsed C-T
- * - flag: what to parse - bit mask of CT_TYPE, CT_CHARSET, CT_MSGR
- *
- * return: 0 OK ; -1 error
- */
- int m_extract_content_type(char* src, int len, content_type_t* ctype, int flag)
- {
- char *p, *end;
- int f = 0;
- if( !src || len <=0 )
- goto error;
- p = src;
- end = p + len;
- while((p < end) && (f != flag))
- {
- while((p < end) && (*p==' ' || *p=='\t'))
- p++;
- if(p >= end)
- goto done;
- if((flag & CT_TYPE) && !(f & CT_TYPE))
- {
- ctype->type.s = p;
- while(p < end && *p!=' ' && *p!='\t' && *p!='\0'
- && *p!=';' && *p!='\r' && *p!='\n')
- p++;
-
- LM_DBG("content-type found\n");
- f |= CT_TYPE;
- ctype->type.len = p - ctype->type.s;
- if(f == flag) {
- return 0;
- } else {
- p++;
- continue;
- }
- } else {
- if((flag & CT_CHARSET) && !(f & CT_CHARSET))
- {
- return -1;
- } else {
- if((flag & CT_MSGR) && !(f & CT_MSGR))
- {
- return -1;
- } else {
- return 0;
- }
- }
- }
- }
- done:
- if(f==flag)
- return 0;
- else
- return -1;
- error:
- LM_DBG("error\n");
- return -1;
- }
- /** build MESSAGE headers
- *
- * Add Content-Type, Contact, Date, and extra headers if they exist
- * expects - max buf len of the resulted body in body->len
- * - body->s MUST be allocated
- * return: 0 OK ; -1 error
- * */
- int m_build_headers(str *buf, str ctype, str contact, time_t date, str extra)
- {
- char *p;
- char strDate[48];
- int lenDate = 0;
- int newLen = 0;
- if(!buf || !buf->s || buf->len <= 0 || ctype.len < 0)
- goto error;
- newLen = 14 + ctype.len + CRLF_LEN + extra.len;
- if(contact.len > 0 && ms_add_contact)
- newLen += CONTACT_PREFIX_LEN + contact.len + CONTACT_SUFFIX_LEN;
- if(buf->len <= newLen)
- goto error;
- p = buf->s;
- if(date > 0)
- {
- lenDate = timetToSipDateStr(date,strDate,48);
- strncpy(p, strDate, lenDate);
- p += lenDate;
- }
- if(ctype.len > 0)
- {
- strncpy(p, "Content-Type: ", 14);
- p += 14;
- strncpy(p, ctype.s, ctype.len);
- p += ctype.len;
- strncpy(p, CRLF, CRLF_LEN);
- p += CRLF_LEN;
-
- }
- if(contact.len > 0 && ms_add_contact)
- {
- strncpy(p, CONTACT_PREFIX, CONTACT_PREFIX_LEN);
- p += CONTACT_PREFIX_LEN;
- strncpy(p, contact.s, contact.len);
- p += contact.len;
- strncpy(p, CONTACT_SUFFIX, CONTACT_SUFFIX_LEN);
- p += CONTACT_SUFFIX_LEN;
- }
- if (extra.len > 0) {
- strncpy(p, extra.s, extra.len);
- p += extra.len;
- }
- buf->len = p - buf->s;
- return 0;
- error:
- return -1;
- }
- /** build MESSAGE body --- add incoming time and 'from'
- *
- * expects - max buf len of the resulted body in body->len
- * - body->s MUST be allocated
- * return: 0 OK ; -1 error
- * */
- int m_build_body(str *body, time_t date, str msg, time_t sdate)
- {
- char *p;
-
- if(!body || !(body->s) || body->len <= 0 || msg.len <= 0
- || date < 0 || msg.len < 0 || (46+msg.len > body->len) )
- goto error;
-
- p = body->s;
- if(ms_add_date!=0)
- {
- if(sdate!=0)
- {
- strncpy(p, "[Reminder message - ", 20);
- p += 20;
-
- strncpy(p, ctime(&sdate), 24);
- p += 24;
- *p++ = ']';
- } else {
- strncpy(p, "[Offline message - ", 19);
- p += 19;
-
- strncpy(p, ctime(&date), 24);
- p += 24;
- *p++ = ']';
- }
- *p++ = ' ';
- }
-
- memcpy(p, msg.s, msg.len);
- p += msg.len;
- body->len = p - body->s;
-
- return 0;
- error:
- return -1;
- }
- /* return time stamp of YYYYMMDDHHMMSS */
- int ms_extract_time(str *time_str, int *time_val)
- {
- struct tm stm;
- int i;
- if(time_str==NULL || time_str->s==NULL
- || time_str->len<=0 || time_val==NULL)
- {
- LM_ERR("bad parameters\n");
- return -1;
- }
-
- memset(&stm, 0, sizeof(struct tm));
- for(i=0; i<time_str->len; i++)
- {
- if(time_str->s[i]<'0' || time_str->s[i]>'9')
- {
- LM_ERR("bad time [%.*s]\n", time_str->len, time_str->s);
- return -1;
- }
- switch(i)
- {
- case 0:
- if(time_str->s[i]<'2')
- {
- LM_ERR("bad year in time [%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- stm.tm_year += 1000*(time_str->s[i]-'0') - 1900;
- break;
- case 1:
- stm.tm_year += 100*(time_str->s[i]-'0');
- break;
- case 2:
- stm.tm_year += 10*(time_str->s[i]-'0');
- break;
- case 3:
- stm.tm_year += (time_str->s[i]-'0');
- break;
- case 4:
- if(time_str->s[i]>'1')
- {
- LM_ERR("bad month in time[%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- stm.tm_mon += 10*(time_str->s[i]-'0') - 1;
- break;
- case 5:
- if((time_str->s[i-1]=='0' && time_str->s[i]=='0')
- || (time_str->s[i-1]=='1' && time_str->s[i]>'2'))
- {
- LM_ERR("bad month in time[%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- stm.tm_mon += (time_str->s[i]-'0');
- break;
- case 6:
- if(time_str->s[i]>'3')
- {
- LM_ERR("bad day in time [%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- stm.tm_mday += 10*(time_str->s[i]-'0');
- break;
- case 7:
- if((time_str->s[i-1]=='0' && time_str->s[i]=='0')
- || (time_str->s[i-1]=='3' && time_str->s[i]>'1'))
- {
- LM_ERR("bad day in time [%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- stm.tm_mday += (time_str->s[i]-'0');
- break;
- case 8:
- if(time_str->s[i]>'2')
- {
- LM_ERR("bad hour in time [%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- stm.tm_hour += 10*(time_str->s[i]-'0');
- break;
- case 9:
- if(time_str->s[i-1]=='2' && time_str->s[i]>'3')
- {
- LM_ERR("bad hour in time [%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- stm.tm_hour += (time_str->s[i]-'0');
- break;
- case 10:
- if(time_str->s[i]>'5')
- {
- LM_ERR("bad min in time [%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- stm.tm_min += 10*(time_str->s[i]-'0');
- break;
- case 11:
- stm.tm_min += (time_str->s[i]-'0');
- break;
- case 12:
- if(time_str->s[i]>'5')
- {
- LM_ERR("bad sec in time [%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- stm.tm_sec += 10*(time_str->s[i]-'0');
- break;
- case 13:
- stm.tm_sec += (time_str->s[i]-'0');
- break;
- default:
- LM_ERR("time spec too long [%.*s]\n",
- time_str->len, time_str->s);
- return -1;
- }
- }
- *time_val = (int)mktime(&stm);
- return 0;
- }
|