2006-12-21 06:30:28 +00:00
|
|
|
|
/*
|
|
|
|
|
* This file is part of the Sofia-SIP package
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 2005 Nokia Corporation.
|
|
|
|
|
*
|
|
|
|
|
* Contact: Pekka Pessi <pekka.pessi@nokia.com>
|
|
|
|
|
*
|
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
|
* modify it under the terms of the GNU Lesser General Public License
|
|
|
|
|
* as published by the Free Software Foundation; either version 2.1 of
|
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This library 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
|
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
|
|
|
* 02110-1301 USA
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**@ingroup sl_utils
|
|
|
|
|
*
|
2008-12-16 18:05:22 +00:00
|
|
|
|
* @CFILE sl_utils_print.c
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @brief Implementation of SIP library utility print functions.
|
|
|
|
|
*
|
|
|
|
|
* @author Pekka Pessi <Pekka.Pessi@nokia.com>
|
|
|
|
|
*
|
|
|
|
|
* @date Created: Thu Oct 5 15:38:39 2000 ppessi
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
|
|
#include <sofia-sip/sip_header.h>
|
|
|
|
|
#include "sofia-sip/sl_utils.h"
|
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
|
/**Print a SIP message.
|
2006-12-21 06:30:28 +00:00
|
|
|
|
*
|
|
|
|
|
* The function sl_message_log() prints shorthand information identifying
|
|
|
|
|
* the SIP message to the given output @a stream. The shorthand information
|
|
|
|
|
* include the method and URL by default. If @a details is nonzero, topmost
|
|
|
|
|
* @Via, @CSeq, @To and @From is included, too.
|
|
|
|
|
*
|
|
|
|
|
* @param stream output stream (if @c NULL, @c stdout is used).
|
|
|
|
|
* @param prefix string printed before the first line.
|
|
|
|
|
*<EFBFBD>@param sip message to be logged.
|
|
|
|
|
* @param details flag specifying if detailed output is desired.
|
|
|
|
|
*/
|
2008-12-16 18:05:22 +00:00
|
|
|
|
void sl_message_log(FILE *stream,
|
2006-12-21 06:30:28 +00:00
|
|
|
|
char const *prefix, sip_t const *sip, int details)
|
|
|
|
|
{
|
|
|
|
|
sip_cseq_t const *cs = sip->sip_cseq;
|
|
|
|
|
|
|
|
|
|
if (stream == NULL)
|
|
|
|
|
stream = stdout;
|
|
|
|
|
|
|
|
|
|
assert(cs);
|
2008-12-16 18:05:22 +00:00
|
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
|
if (sip->sip_request) {
|
|
|
|
|
fprintf(stream,
|
|
|
|
|
"%s%s "URL_FORMAT_STRING" (CSeq %d %s)\n",
|
|
|
|
|
prefix,
|
|
|
|
|
sip->sip_request->rq_method_name,
|
|
|
|
|
URL_PRINT_ARGS(sip->sip_request->rq_url),
|
|
|
|
|
cs->cs_seq,
|
|
|
|
|
cs->cs_method_name);
|
|
|
|
|
|
|
|
|
|
if (!details)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (sip->sip_via) {
|
|
|
|
|
fputs(prefix, stream);
|
|
|
|
|
sl_via_print(stream, "Via: %s\n", sip->sip_via);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
fprintf(stream,
|
|
|
|
|
"%s%03u %s (CSeq %d %s)\n",
|
|
|
|
|
prefix,
|
|
|
|
|
sip->sip_status->st_status,
|
|
|
|
|
sip->sip_status->st_phrase,
|
|
|
|
|
cs->cs_seq,
|
|
|
|
|
cs->cs_method_name);
|
|
|
|
|
if (!details)
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (sip->sip_from)
|
|
|
|
|
sl_from_print(stream, "\tFrom: %s\n", sip->sip_from);
|
|
|
|
|
|
|
|
|
|
if (sip->sip_to)
|
|
|
|
|
sl_to_print(stream, "\tTo: %s\n", sip->sip_to);
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
|
/** Print @From header.
|
2006-12-21 06:30:28 +00:00
|
|
|
|
*
|
|
|
|
|
* The function sl_from_print() prints the contents of @a from header to
|
|
|
|
|
* the output @a stream. The @a fmt specifies the output format, where %s
|
|
|
|
|
* is replaced with header contents. If @a fmt is @c NULL, only the header
|
|
|
|
|
* contents are printed.
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param stream output stream
|
2008-12-16 18:05:22 +00:00
|
|
|
|
* @param fmt output format
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param from header object
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
|
|
|
|
* @return
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* The function sl_from_print() returns number of bytes printed,
|
|
|
|
|
* or -1 upon an error.
|
|
|
|
|
*/
|
|
|
|
|
issize_t sl_from_print(FILE *stream, char const *fmt, sip_from_t const *from)
|
|
|
|
|
{
|
|
|
|
|
sip_addr_t a[1];
|
|
|
|
|
|
|
|
|
|
if (from == NULL)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
memcpy(a, from, sizeof a);
|
|
|
|
|
a->a_params = NULL;
|
|
|
|
|
if (!a->a_display) a->a_display = "";
|
2008-12-16 18:05:22 +00:00
|
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
|
return sl_header_print(stream, fmt, (sip_header_t *)a);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Print @To header.
|
|
|
|
|
*
|
|
|
|
|
* The function sl_to_print() prints the contents of @a to header to
|
|
|
|
|
* the output @a stream. The @a fmt specifies the output format, where %s
|
|
|
|
|
* is replaced with header contents. If @a fmt is @c NULL, only the header
|
|
|
|
|
* contents are printed.
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param stream output stream
|
2008-12-16 18:05:22 +00:00
|
|
|
|
* @param fmt output format
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param to header object
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
|
|
|
|
* @return
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* The function sl_to_print() returns number of bytes printed,
|
|
|
|
|
* or -1 upon an error.
|
|
|
|
|
*/
|
|
|
|
|
issize_t sl_to_print(FILE *stream, char const *fmt, sip_to_t const *to)
|
|
|
|
|
{
|
|
|
|
|
return sl_from_print(stream, fmt, (sip_from_t const *)to);
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
|
/** Print @Contact header.
|
2006-12-21 06:30:28 +00:00
|
|
|
|
*
|
|
|
|
|
* The function sl_contact_print() prints the contents of @a contact
|
|
|
|
|
* header to the output @a stream. The @a fmt specifies the output format,
|
|
|
|
|
* where %s is replaced with header contents. If @a fmt is @c NULL, only the
|
|
|
|
|
* header contents are printed.
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param stream output stream
|
2008-12-16 18:05:22 +00:00
|
|
|
|
* @param fmt output format
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param contact header object
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
|
|
|
|
* @return
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* The function sl_contact_print() returns number of bytes printed,
|
|
|
|
|
* or -1 upon an error.
|
|
|
|
|
*/
|
|
|
|
|
issize_t sl_contact_print(FILE *stream, char const *fmt,
|
|
|
|
|
sip_contact_t const *m)
|
|
|
|
|
{
|
|
|
|
|
return sl_from_print(stream, fmt, (sip_from_t const *)m);
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
|
/** Print @Allow header(s).
|
2006-12-21 06:30:28 +00:00
|
|
|
|
*
|
|
|
|
|
* The function sl_allow_print() prints the contents of @a allow header to
|
|
|
|
|
* the output @a stream. The @a fmt specifies the output format, where %s
|
|
|
|
|
* is replaced with header contents. If @a fmt is @c NULL, only the header
|
|
|
|
|
* contents are printed.
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param stream output stream
|
2008-12-16 18:05:22 +00:00
|
|
|
|
* @param fmt output format
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param allow header object
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
|
|
|
|
* @return
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* The function sl_allow_print() returns number of bytes printed,
|
|
|
|
|
* or -1 upon an error.
|
|
|
|
|
*/
|
|
|
|
|
issize_t sl_allow_print(FILE *stream,
|
|
|
|
|
char const *fmt,
|
|
|
|
|
sip_allow_t const *allow)
|
|
|
|
|
{
|
|
|
|
|
return sl_header_print(stream, fmt, (sip_header_t *)allow);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
|
/** Print message payload.
|
2006-12-21 06:30:28 +00:00
|
|
|
|
*
|
|
|
|
|
* The function sl_payload_print() prints the contents of @a payload
|
|
|
|
|
* object to the output @a stream. The @a fmt specifies the output format,
|
|
|
|
|
* where %s is replaced with header contents. If @a fmt is @c NULL, only the
|
|
|
|
|
* header contents are printed.
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param stream output stream
|
2008-12-16 18:05:22 +00:00
|
|
|
|
* @param prefix prefix appended to each payload line
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param pl payload object
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
|
|
|
|
* @return
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* The function sl_payload_print() returns number of bytes printed,
|
|
|
|
|
* or -1 upon an error.
|
|
|
|
|
*/
|
|
|
|
|
issize_t sl_payload_print(FILE *stream, char const *prefix, sip_payload_t const *pl)
|
|
|
|
|
{
|
|
|
|
|
char *s = pl->pl_data, *end = pl->pl_data + pl->pl_len;
|
2009-01-27 22:59:00 +00:00
|
|
|
|
size_t n, total = 0, crlf = 1, actual;
|
2006-12-21 06:30:28 +00:00
|
|
|
|
|
|
|
|
|
while (s < end && *s != '\0') {
|
|
|
|
|
n = strncspn(s, end - s, "\r\n");
|
|
|
|
|
crlf = strnspn(s + n, end - s - n, "\r\n");
|
|
|
|
|
if (prefix)
|
|
|
|
|
fputs(prefix, stream), total += strlen(prefix);
|
2009-01-27 22:59:00 +00:00
|
|
|
|
actual = fwrite(s, 1, n + crlf, stream) ;
|
|
|
|
|
if (actual == 0)
|
|
|
|
|
return -1;
|
|
|
|
|
s += actual;
|
|
|
|
|
total += actual;
|
2006-12-21 06:30:28 +00:00
|
|
|
|
}
|
|
|
|
|
if (crlf == 0)
|
|
|
|
|
fputs("\n", stream), total++;
|
|
|
|
|
|
|
|
|
|
return (issize_t)total;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
|
/** Print @Via header.
|
2006-12-21 06:30:28 +00:00
|
|
|
|
*
|
|
|
|
|
* The function sl_via_print() prints the contents of @a via header to
|
|
|
|
|
* the output @a stream. The @a fmt specifies the output format, where %s
|
|
|
|
|
* is replaced with header contents. If @a fmt is @c NULL, only the header
|
|
|
|
|
* contents are printed.
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param stream output stream
|
2008-12-16 18:05:22 +00:00
|
|
|
|
* @param fmt output format
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param v header object
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
|
|
|
|
* @return
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* The function sl_via_print() returns number of bytes printed,
|
|
|
|
|
* or -1 upon an error.
|
|
|
|
|
*/
|
|
|
|
|
issize_t sl_via_print(FILE *stream, char const *fmt, sip_via_t const *v)
|
|
|
|
|
{
|
|
|
|
|
char s[1024];
|
|
|
|
|
|
|
|
|
|
sip_header_field_e(s, sizeof(s), (sip_header_t const *)v, 0);
|
|
|
|
|
s[sizeof(s) - 1] = '\0';
|
|
|
|
|
|
|
|
|
|
if (fmt && strcmp(fmt, "%s"))
|
|
|
|
|
return fprintf(stream, fmt, s);
|
|
|
|
|
if (fputs(s, stream) >= 0)
|
|
|
|
|
return (issize_t)strlen(s);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
|
/** Print a header.
|
2006-12-21 06:30:28 +00:00
|
|
|
|
*
|
|
|
|
|
* Prints the contents of an header to the output @a stream. The @a fmt
|
2008-12-16 18:05:22 +00:00
|
|
|
|
* specifies the output format, where %s is replaced with header contents.
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* If @a fmt is @c NULL, only the header contents are printed.
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param stream output stream
|
2008-12-16 18:05:22 +00:00
|
|
|
|
* @param fmt output format
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* @param v header object
|
2008-12-16 18:05:22 +00:00
|
|
|
|
*
|
|
|
|
|
* @return
|
2006-12-21 06:30:28 +00:00
|
|
|
|
* Number of bytes logged, or -1 upon an error.
|
|
|
|
|
*/
|
|
|
|
|
issize_t sl_header_print(FILE *stream, char const *fmt, sip_header_t const *h)
|
|
|
|
|
{
|
|
|
|
|
char *s, b[1024];
|
|
|
|
|
issize_t len;
|
|
|
|
|
|
|
|
|
|
len = sip_header_field_e(s = b, sizeof b, h, 0);
|
|
|
|
|
if (len == -1)
|
|
|
|
|
return len;
|
|
|
|
|
|
|
|
|
|
if ((size_t)len >= sizeof b) {
|
|
|
|
|
s = malloc(len + 1); if (!s) return -1;
|
|
|
|
|
sip_header_field_e(s, len + 1, h, 0);
|
2008-12-16 18:05:22 +00:00
|
|
|
|
}
|
2006-12-21 06:30:28 +00:00
|
|
|
|
s[len] = '\0';
|
|
|
|
|
|
|
|
|
|
if (fmt != NULL && strcmp(fmt, "%s") != 0)
|
|
|
|
|
len = fprintf(stream, fmt, s);
|
|
|
|
|
else if (fputs(s, stream) < 0)
|
|
|
|
|
len = -1;
|
|
|
|
|
|
|
|
|
|
if (s != b)
|
|
|
|
|
free(s);
|
2008-12-16 18:05:22 +00:00
|
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
|
return len;
|
|
|
|
|
}
|