2005-11-19 20:07:43 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
2010-02-06 03:38:24 +00:00
|
|
|
* Copyright (C) 2005-2010, Anthony Minessale II <anthm@freeswitch.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
*
|
|
|
|
* Version: MPL 1.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
2009-02-04 21:20:54 +00:00
|
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
2009-02-04 21:20:54 +00:00
|
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
2008-01-16 20:18:09 +00:00
|
|
|
* Juan Jose Comellas <juanjo@comellas.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
*
|
|
|
|
*
|
2008-10-06 23:05:55 +00:00
|
|
|
* switch_utils.c -- Compatibility and Helper Code
|
2005-11-19 20:07:43 +00:00
|
|
|
*
|
|
|
|
*/
|
2008-01-27 17:42:51 +00:00
|
|
|
|
2006-02-26 00:12:17 +00:00
|
|
|
#include <switch.h>
|
2007-01-19 19:11:44 +00:00
|
|
|
#ifndef WIN32
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#endif
|
2007-11-05 18:45:26 +00:00
|
|
|
#include "private/switch_core_pvt.h"
|
2008-01-19 21:54:11 +00:00
|
|
|
#define ESCAPE_META '\\'
|
2007-11-26 23:41:00 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
struct switch_network_node {
|
|
|
|
uint32_t ip;
|
|
|
|
uint32_t mask;
|
|
|
|
uint32_t bits;
|
|
|
|
switch_bool_t ok;
|
2008-07-16 17:44:54 +00:00
|
|
|
char *token;
|
2009-06-02 16:55:10 +00:00
|
|
|
char *str;
|
2008-03-26 22:14:09 +00:00
|
|
|
struct switch_network_node *next;
|
|
|
|
};
|
|
|
|
typedef struct switch_network_node switch_network_node_t;
|
|
|
|
|
|
|
|
struct switch_network_list {
|
|
|
|
struct switch_network_node *node_head;
|
|
|
|
switch_bool_t default_type;
|
|
|
|
switch_memory_pool_t *pool;
|
2009-06-02 16:55:10 +00:00
|
|
|
char *name;
|
2008-03-26 22:14:09 +00:00
|
|
|
};
|
|
|
|
|
2008-04-21 14:35:14 +00:00
|
|
|
#ifndef WIN32
|
2009-06-10 00:48:35 +00:00
|
|
|
SWITCH_DECLARE(int) switch_inet_pton(int af, const char *src, void *dst)
|
2008-04-21 14:35:14 +00:00
|
|
|
{
|
|
|
|
return inet_pton(af, src, dst);
|
|
|
|
}
|
|
|
|
#endif
|
2008-03-26 22:14:09 +00:00
|
|
|
|
2009-04-13 18:35:26 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_frame_alloc(switch_frame_t **frame, switch_size_t size)
|
|
|
|
{
|
|
|
|
switch_frame_t *new_frame;
|
|
|
|
|
|
|
|
switch_zmalloc(new_frame, sizeof(*new_frame));
|
|
|
|
|
|
|
|
switch_set_flag(new_frame, SFF_DYNAMIC);
|
|
|
|
new_frame->buflen = size;
|
|
|
|
new_frame->data = malloc(size);
|
|
|
|
switch_assert(new_frame->data);
|
|
|
|
|
|
|
|
*frame = new_frame;
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_frame_dup(switch_frame_t *orig, switch_frame_t **clone)
|
|
|
|
{
|
|
|
|
switch_frame_t *new_frame;
|
|
|
|
|
2009-05-29 18:48:54 +00:00
|
|
|
if (!orig) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_assert(orig->buflen);
|
|
|
|
|
2009-04-13 18:35:26 +00:00
|
|
|
new_frame = malloc(sizeof(*new_frame));
|
2009-05-15 21:07:18 +00:00
|
|
|
|
|
|
|
switch_assert(new_frame);
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-04-13 18:35:26 +00:00
|
|
|
*new_frame = *orig;
|
|
|
|
switch_set_flag(new_frame, SFF_DYNAMIC);
|
2009-05-29 18:48:54 +00:00
|
|
|
|
2009-04-13 18:35:26 +00:00
|
|
|
new_frame->data = malloc(new_frame->buflen);
|
|
|
|
switch_assert(new_frame->data);
|
|
|
|
|
|
|
|
memcpy(new_frame->data, orig->data, orig->datalen);
|
|
|
|
new_frame->codec = NULL;
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-04-13 18:35:26 +00:00
|
|
|
*clone = new_frame;
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_frame_free(switch_frame_t **frame)
|
|
|
|
{
|
|
|
|
if (!frame || !*frame || !switch_test_flag((*frame), SFF_DYNAMIC)) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-05-29 18:48:54 +00:00
|
|
|
switch_safe_free((*frame)->data);
|
2009-04-13 18:35:26 +00:00
|
|
|
free(*frame);
|
|
|
|
*frame = NULL;
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_network_list_create(switch_network_list_t **list, const char *name, switch_bool_t default_type,
|
|
|
|
switch_memory_pool_t *pool)
|
2008-03-26 22:14:09 +00:00
|
|
|
{
|
|
|
|
switch_network_list_t *new_list;
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
if (!pool) {
|
|
|
|
switch_core_new_memory_pool(&pool);
|
|
|
|
}
|
|
|
|
|
|
|
|
new_list = switch_core_alloc(pool, sizeof(**list));
|
|
|
|
new_list->pool = pool;
|
|
|
|
new_list->default_type = default_type;
|
2009-06-02 16:55:10 +00:00
|
|
|
new_list->name = switch_core_strdup(new_list->pool, name);
|
2008-03-26 22:14:09 +00:00
|
|
|
|
|
|
|
*list = new_list;
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2008-07-16 17:44:54 +00:00
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_network_list_validate_ip_token(switch_network_list_t *list, uint32_t ip, const char **token)
|
2008-03-26 22:14:09 +00:00
|
|
|
{
|
|
|
|
switch_network_node_t *node;
|
|
|
|
switch_bool_t ok = list->default_type;
|
|
|
|
uint32_t bits = 0;
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
for (node = list->node_head; node; node = node->next) {
|
|
|
|
if (node->bits > bits && switch_test_subnet(ip, node->ip, node->mask)) {
|
|
|
|
if (node->ok) {
|
|
|
|
ok = SWITCH_TRUE;
|
|
|
|
} else {
|
|
|
|
ok = SWITCH_FALSE;
|
|
|
|
}
|
2008-07-16 17:44:54 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
bits = node->bits;
|
2008-07-16 17:44:54 +00:00
|
|
|
|
|
|
|
if (token) {
|
|
|
|
*token = node->token;
|
|
|
|
}
|
2008-03-26 22:14:09 +00:00
|
|
|
}
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_network_list_perform_add_cidr_token(switch_network_list_t *list, const char *cidr_str, switch_bool_t ok,
|
|
|
|
const char *token)
|
2008-03-26 22:14:09 +00:00
|
|
|
{
|
|
|
|
uint32_t ip, mask, bits;
|
|
|
|
switch_network_node_t *node;
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
if (switch_parse_cidr(cidr_str, &ip, &mask, &bits)) {
|
2009-12-02 18:51:28 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error Adding %s (%s) [%s] to list %s\n",
|
2010-02-06 03:38:24 +00:00
|
|
|
cidr_str, ok ? "allow" : "deny", switch_str_nil(token), list->name);
|
2008-03-26 22:14:09 +00:00
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
node = switch_core_alloc(list->pool, sizeof(*node));
|
|
|
|
|
|
|
|
node->ip = ip;
|
|
|
|
node->mask = mask;
|
|
|
|
node->ok = ok;
|
|
|
|
node->bits = bits;
|
2009-06-02 16:55:10 +00:00
|
|
|
node->str = switch_core_strdup(list->pool, cidr_str);
|
2008-03-26 22:14:09 +00:00
|
|
|
|
2009-10-23 16:03:42 +00:00
|
|
|
if (!zstr(token)) {
|
2008-07-16 17:44:54 +00:00
|
|
|
node->token = switch_core_strdup(list->pool, token);
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
node->next = list->node_head;
|
|
|
|
list->node_head = node;
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding %s (%s) [%s] to list %s\n",
|
2009-12-02 18:51:28 +00:00
|
|
|
cidr_str, ok ? "allow" : "deny", switch_str_nil(token), list->name);
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_network_list_add_cidr_token(switch_network_list_t *list, const char *cidr_str, switch_bool_t ok, const char *token)
|
|
|
|
{
|
|
|
|
char *cidr_str_dup = NULL;
|
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-12-02 18:51:28 +00:00
|
|
|
if (strchr(cidr_str, ',')) {
|
|
|
|
char *argv[32] = { 0 };
|
2010-02-06 03:38:24 +00:00
|
|
|
int i, argc;
|
2009-12-02 19:00:25 +00:00
|
|
|
cidr_str_dup = strdup(cidr_str);
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-12-02 18:51:28 +00:00
|
|
|
switch_assert(cidr_str_dup);
|
2010-02-06 03:38:24 +00:00
|
|
|
if ((argc = switch_separate_string(cidr_str_dup, ',', argv, (sizeof(argv) / sizeof(argv[0]))))) {
|
2009-12-02 18:51:28 +00:00
|
|
|
for (i = 0; i < argc; i++) {
|
|
|
|
switch_status_t this_status;
|
|
|
|
if ((this_status = switch_network_list_perform_add_cidr_token(list, argv[i], ok, token)) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
status = this_status;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
status = switch_network_list_perform_add_cidr_token(list, cidr_str, ok, token);
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-12-02 18:51:28 +00:00
|
|
|
switch_safe_free(cidr_str_dup);
|
|
|
|
return status;
|
2008-03-26 22:14:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_network_list_add_host_mask(switch_network_list_t *list, const char *host, const char *mask_str, switch_bool_t ok)
|
|
|
|
{
|
|
|
|
int ip, mask;
|
|
|
|
switch_network_node_t *node;
|
|
|
|
|
2008-04-21 14:35:14 +00:00
|
|
|
switch_inet_pton(AF_INET, host, &ip);
|
|
|
|
switch_inet_pton(AF_INET, mask_str, &mask);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
node = switch_core_alloc(list->pool, sizeof(*node));
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2009-06-02 16:55:10 +00:00
|
|
|
node->ip = ntohl(ip);
|
|
|
|
node->mask = ntohl(mask);
|
2008-03-26 22:14:09 +00:00
|
|
|
node->ok = ok;
|
|
|
|
|
|
|
|
/* http://graphics.stanford.edu/~seander/bithacks.html */
|
|
|
|
mask = mask - ((mask >> 1) & 0x55555555);
|
2008-05-27 04:30:03 +00:00
|
|
|
mask = (mask & 0x33333333) + ((mask >> 2) & 0x33333333);
|
2008-03-26 22:14:09 +00:00
|
|
|
node->bits = (((mask + (mask >> 4)) & 0xF0F0F0F) * 0x1010101) >> 24;
|
2009-06-02 16:55:10 +00:00
|
|
|
node->str = switch_core_sprintf(list->pool, "%s:%s", host, mask_str);
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
node->next = list->node_head;
|
|
|
|
list->node_head = node;
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_parse_cidr(const char *string, uint32_t *ip, uint32_t *mask, uint32_t *bitp)
|
|
|
|
{
|
2008-03-31 16:06:08 +00:00
|
|
|
char host[128];
|
2008-03-26 22:14:09 +00:00
|
|
|
char *bit_str;
|
|
|
|
int32_t bits;
|
|
|
|
|
2008-03-31 16:06:08 +00:00
|
|
|
switch_copy_string(host, string, sizeof(host));
|
2008-03-26 22:14:09 +00:00
|
|
|
bit_str = strchr(host, '/');
|
|
|
|
|
|
|
|
if (!bit_str) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*bit_str++ = '\0';
|
|
|
|
bits = atoi(bit_str);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
if (bits < 0 || bits > 32) {
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
bits = atoi(bit_str);
|
2008-04-21 14:35:14 +00:00
|
|
|
switch_inet_pton(AF_INET, host, ip);
|
2008-07-14 16:32:56 +00:00
|
|
|
*ip = htonl(*ip);
|
|
|
|
|
|
|
|
*mask = 0xFFFFFFFF & ~(0xFFFFFFFF >> bits);
|
|
|
|
|
2008-03-26 22:14:09 +00:00
|
|
|
*bitp = bits;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-12-10 21:58:40 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_find_end_paren(const char *s, char open, char close)
|
|
|
|
{
|
|
|
|
const char *e = NULL;
|
|
|
|
int depth = 0;
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-12-10 21:58:40 +00:00
|
|
|
while (s && *s && *s == ' ') {
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
|
2007-12-11 21:31:57 +00:00
|
|
|
if (s && *s == open) {
|
2007-12-10 21:58:40 +00:00
|
|
|
depth++;
|
|
|
|
for (e = s + 1; e && *e; e++) {
|
|
|
|
if (*e == open) {
|
|
|
|
depth++;
|
|
|
|
} else if (*e == close) {
|
|
|
|
depth--;
|
|
|
|
if (!depth) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-26 18:04:22 +00:00
|
|
|
return (e && *e == close) ? (char *) e : NULL;
|
2007-12-10 21:58:40 +00:00
|
|
|
}
|
|
|
|
|
2007-11-28 22:27:33 +00:00
|
|
|
SWITCH_DECLARE(switch_size_t) switch_fd_read_line(int fd, char *buf, switch_size_t len)
|
2007-11-26 23:41:00 +00:00
|
|
|
{
|
|
|
|
char c, *p;
|
|
|
|
int cur;
|
|
|
|
switch_size_t total = 0;
|
|
|
|
|
|
|
|
p = buf;
|
2008-01-05 21:47:24 +00:00
|
|
|
while (total + 2 < len && (cur = read(fd, &c, 1)) == 1) {
|
2007-11-26 23:41:00 +00:00
|
|
|
total += cur;
|
|
|
|
*p++ = c;
|
|
|
|
if (c == '\r' || c == '\n') {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*p++ = '\0';
|
2008-01-05 21:47:24 +00:00
|
|
|
assert(total < len);
|
2007-11-26 23:41:00 +00:00
|
|
|
return total;
|
|
|
|
}
|
|
|
|
|
2007-12-04 16:22:02 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_amp_encode(char *s, char *buf, switch_size_t len)
|
|
|
|
{
|
|
|
|
char *p, *q;
|
2007-12-04 16:52:24 +00:00
|
|
|
switch_size_t x = 0;
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(s);
|
2007-12-04 16:22:02 +00:00
|
|
|
|
|
|
|
q = buf;
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
for (p = s; x < len; p++) {
|
|
|
|
switch (*p) {
|
2007-12-04 16:22:02 +00:00
|
|
|
case '<':
|
2008-05-27 04:30:03 +00:00
|
|
|
if (x + 4 > len - 1) {
|
2007-12-04 16:22:02 +00:00
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
*q++ = '&';
|
|
|
|
*q++ = 'l';
|
|
|
|
*q++ = 't';
|
|
|
|
*q++ = ';';
|
|
|
|
x += 4;
|
|
|
|
break;
|
|
|
|
case '>':
|
2008-05-27 04:30:03 +00:00
|
|
|
if (x + 4 > len - 1) {
|
2007-12-04 16:22:02 +00:00
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
*q++ = '&';
|
|
|
|
*q++ = 'g';
|
|
|
|
*q++ = 't';
|
|
|
|
*q++ = ';';
|
|
|
|
x += 4;
|
|
|
|
break;
|
|
|
|
default:
|
2008-05-27 04:30:03 +00:00
|
|
|
if (x + 1 > len - 1) {
|
2007-12-04 16:22:02 +00:00
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
*q++ = *p;
|
|
|
|
x++;
|
|
|
|
if (*p == '\0') {
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
end:
|
2007-12-04 16:22:02 +00:00
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
2007-11-26 23:41:00 +00:00
|
|
|
|
2007-10-06 00:30:04 +00:00
|
|
|
static const char switch_b64_table[65] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
|
|
|
#define B64BUFFLEN 1024
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_b64_encode(unsigned char *in, switch_size_t ilen, unsigned char *out, switch_size_t olen)
|
|
|
|
{
|
2008-05-27 04:30:03 +00:00
|
|
|
int y = 0, bytes = 0;
|
|
|
|
size_t x = 0;
|
|
|
|
unsigned int b = 0, l = 0;
|
|
|
|
|
|
|
|
for (x = 0; x < ilen; x++) {
|
|
|
|
b = (b << 8) + in[x];
|
|
|
|
l += 8;
|
|
|
|
while (l >= 6) {
|
|
|
|
out[bytes++] = switch_b64_table[(b >> (l -= 6)) % 64];
|
|
|
|
if (++y != 72) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
//out[bytes++] = '\n';
|
|
|
|
y = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (l > 0) {
|
|
|
|
out[bytes++] = switch_b64_table[((b % 16) << (6 - l)) % 64];
|
|
|
|
}
|
|
|
|
if (l != 0) {
|
2007-10-06 00:30:04 +00:00
|
|
|
while (l < 6) {
|
|
|
|
out[bytes++] = '=', l += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2007-10-06 00:30:04 +00:00
|
|
|
}
|
|
|
|
|
2008-01-16 06:01:53 +00:00
|
|
|
SWITCH_DECLARE(switch_size_t) switch_b64_decode(char *in, char *out, switch_size_t olen)
|
2007-11-27 19:25:16 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
char l64[256];
|
|
|
|
int b = 0, c, l = 0, i;
|
|
|
|
char *ip, *op = out;
|
|
|
|
size_t ol = 0;
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
for (i = 0; i < 256; i++) {
|
2007-11-27 19:25:16 +00:00
|
|
|
l64[i] = -1;
|
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
for (i = 0; i < 64; i++) {
|
|
|
|
l64[(int) switch_b64_table[i]] = (char) i;
|
2007-11-27 19:25:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (ip = in; ip && *ip; ip++) {
|
2008-05-27 04:30:03 +00:00
|
|
|
c = l64[(int) *ip];
|
2007-11-27 19:25:16 +00:00
|
|
|
if (c == -1) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
b = (b << 6) + c;
|
|
|
|
l += 6;
|
|
|
|
|
|
|
|
while (l >= 8) {
|
2008-05-27 04:30:03 +00:00
|
|
|
op[ol++] = (char) ((b >> (l -= 8)) % 256);
|
|
|
|
if (ol >= olen - 2) {
|
2007-11-27 19:25:16 +00:00
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
end:
|
2007-11-27 19:25:16 +00:00
|
|
|
|
|
|
|
op[ol++] = '\0';
|
|
|
|
|
2008-01-16 06:01:53 +00:00
|
|
|
return ol;
|
2007-11-27 19:25:16 +00:00
|
|
|
}
|
|
|
|
|
2007-11-26 17:43:42 +00:00
|
|
|
static int write_buf(int fd, const char *buf)
|
2007-10-12 17:12:31 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
int len = (int) strlen(buf);
|
|
|
|
if (fd && write(fd, buf, len) != len) {
|
|
|
|
close(fd);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_simple_email(const char *to,
|
|
|
|
const char *from,
|
|
|
|
const char *headers,
|
|
|
|
const char *body, const char *file, const char *convert_cmd, const char *convert_ext)
|
2007-10-12 17:12:31 +00:00
|
|
|
{
|
|
|
|
char *bound = "XXXX_boundary_XXXX";
|
2007-11-26 23:41:00 +00:00
|
|
|
const char *mime_type = "audio/inline";
|
2007-10-12 17:12:31 +00:00
|
|
|
char filename[80], buf[B64BUFFLEN];
|
|
|
|
int fd = 0, ifd = 0;
|
|
|
|
int x = 0, y = 0, bytes = 0, ilen = 0;
|
|
|
|
unsigned int b = 0, l = 0;
|
|
|
|
unsigned char in[B64BUFFLEN];
|
|
|
|
unsigned char out[B64BUFFLEN + 512];
|
2009-10-01 22:43:44 +00:00
|
|
|
char *dupfile = NULL, *ext = NULL;
|
|
|
|
char *newfile = NULL;
|
|
|
|
switch_bool_t rval = SWITCH_FALSE;
|
|
|
|
|
2009-10-23 16:03:42 +00:00
|
|
|
if (!zstr(file) && !zstr(convert_cmd) && !zstr(convert_ext)) {
|
2009-10-01 22:43:44 +00:00
|
|
|
if ((ext = strrchr(file, '.'))) {
|
|
|
|
dupfile = strdup(file);
|
|
|
|
if ((ext = strrchr(dupfile, '.'))) {
|
|
|
|
*ext++ = '\0';
|
|
|
|
newfile = switch_mprintf("%s.%s", dupfile, convert_ext);
|
|
|
|
}
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-10-01 22:43:44 +00:00
|
|
|
if (newfile) {
|
|
|
|
char cmd[1024] = "";
|
|
|
|
switch_snprintf(cmd, sizeof(cmd), "%s %s %s", convert_cmd, file, newfile);
|
|
|
|
switch_system(cmd, SWITCH_TRUE);
|
|
|
|
file = newfile;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_safe_free(dupfile);
|
|
|
|
}
|
2007-10-12 17:12:31 +00:00
|
|
|
|
2009-01-25 21:23:07 +00:00
|
|
|
switch_snprintf(filename, 80, "%smail.%d%04x", SWITCH_GLOBAL_dirs.temp_dir, (int) switch_epoch_time_now(NULL), rand() & 0xffff);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
|
|
|
if ((fd = open(filename, O_WRONLY | O_CREAT | O_TRUNC, 0644))) {
|
|
|
|
if (file) {
|
2008-07-12 18:44:05 +00:00
|
|
|
if ((ifd = open(file, O_RDONLY | O_BINARY)) < 1) {
|
2009-10-05 17:38:46 +00:00
|
|
|
rval = SWITCH_FALSE;
|
|
|
|
goto end;
|
2008-05-27 04:30:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
switch_snprintf(buf, B64BUFFLEN, "MIME-Version: 1.0\nContent-Type: multipart/mixed; boundary=\"%s\"\n", bound);
|
|
|
|
if (!write_buf(fd, buf)) {
|
2009-10-05 17:38:46 +00:00
|
|
|
rval = SWITCH_FALSE;
|
|
|
|
goto end;
|
2008-05-27 04:30:03 +00:00
|
|
|
}
|
|
|
|
|
2009-10-05 17:38:46 +00:00
|
|
|
if (headers && !write_buf(fd, headers)) {
|
|
|
|
rval = SWITCH_FALSE;
|
|
|
|
goto end;
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2009-10-05 17:38:46 +00:00
|
|
|
if (!write_buf(fd, "\n\n")) {
|
|
|
|
rval = SWITCH_FALSE;
|
|
|
|
goto end;
|
|
|
|
}
|
2007-10-12 17:12:31 +00:00
|
|
|
|
2008-05-26 02:51:34 +00:00
|
|
|
if (body && switch_stristr("content-type", body)) {
|
|
|
|
switch_snprintf(buf, B64BUFFLEN, "--%s\n", bound);
|
|
|
|
} else {
|
|
|
|
switch_snprintf(buf, B64BUFFLEN, "--%s\nContent-Type: text/plain\n\n", bound);
|
|
|
|
}
|
2009-10-05 17:38:46 +00:00
|
|
|
if (!write_buf(fd, buf)) {
|
2010-02-06 03:38:24 +00:00
|
|
|
rval = SWITCH_FALSE;
|
2009-10-05 17:38:46 +00:00
|
|
|
goto end;
|
|
|
|
}
|
2007-10-12 17:12:31 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
if (body) {
|
|
|
|
if (!write_buf(fd, body)) {
|
2009-10-05 17:38:46 +00:00
|
|
|
rval = SWITCH_FALSE;
|
|
|
|
goto end;
|
2008-05-27 04:30:03 +00:00
|
|
|
}
|
|
|
|
}
|
2007-10-12 17:12:31 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
if (file) {
|
2007-12-11 21:31:57 +00:00
|
|
|
const char *stipped_file = switch_cut_path(file);
|
2007-11-26 23:41:00 +00:00
|
|
|
const char *new_type;
|
|
|
|
char *ext;
|
|
|
|
|
2007-12-11 21:31:57 +00:00
|
|
|
if ((ext = strrchr(stipped_file, '.'))) {
|
2007-11-26 23:41:00 +00:00
|
|
|
ext++;
|
|
|
|
if ((new_type = switch_core_mime_ext2type(ext))) {
|
|
|
|
mime_type = new_type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-12-12 21:53:32 +00:00
|
|
|
switch_snprintf(buf, B64BUFFLEN,
|
2008-05-27 04:30:03 +00:00
|
|
|
"\n\n--%s\nContent-Type: %s; name=\"%s\"\n"
|
|
|
|
"Content-ID: <ATTACHED@freeswitch.org>\n"
|
|
|
|
"Content-Transfer-Encoding: base64\n"
|
|
|
|
"Content-Description: Sound attachment.\n"
|
|
|
|
"Content-Disposition: attachment; filename=\"%s\"\n\n", bound, mime_type, stipped_file, stipped_file);
|
2009-10-05 17:38:46 +00:00
|
|
|
if (!write_buf(fd, buf)) {
|
|
|
|
rval = SWITCH_FALSE;
|
|
|
|
goto end;
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
|
|
|
while ((ilen = read(ifd, in, B64BUFFLEN))) {
|
|
|
|
for (x = 0; x < ilen; x++) {
|
|
|
|
b = (b << 8) + in[x];
|
|
|
|
l += 8;
|
|
|
|
while (l >= 6) {
|
|
|
|
out[bytes++] = switch_b64_table[(b >> (l -= 6)) % 64];
|
|
|
|
if (++y != 72)
|
|
|
|
continue;
|
|
|
|
out[bytes++] = '\n';
|
|
|
|
y = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (write(fd, &out, bytes) != bytes) {
|
2009-10-01 22:43:44 +00:00
|
|
|
rval = -1;
|
2008-05-27 04:30:03 +00:00
|
|
|
} else
|
|
|
|
bytes = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (l > 0) {
|
|
|
|
out[bytes++] = switch_b64_table[((b % 16) << (6 - l)) % 64];
|
|
|
|
}
|
|
|
|
if (l != 0)
|
|
|
|
while (l < 6) {
|
|
|
|
out[bytes++] = '=', l += 2;
|
|
|
|
}
|
|
|
|
if (write(fd, &out, bytes) != bytes) {
|
2009-10-01 22:43:44 +00:00
|
|
|
rval = -1;
|
2008-05-27 04:30:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_snprintf(buf, B64BUFFLEN, "\n\n--%s--\n.\n", bound);
|
2009-10-05 17:38:46 +00:00
|
|
|
|
|
|
|
if (!write_buf(fd, buf)) {
|
|
|
|
rval = SWITCH_FALSE;
|
|
|
|
goto end;
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (fd) {
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
if (ifd) {
|
|
|
|
close(ifd);
|
|
|
|
}
|
2009-10-20 18:07:24 +00:00
|
|
|
|
2009-10-23 16:03:42 +00:00
|
|
|
if (zstr(from)) {
|
2009-10-20 18:07:24 +00:00
|
|
|
from = "freeswitch";
|
|
|
|
}
|
2008-07-10 15:35:35 +00:00
|
|
|
#ifdef WIN32
|
2009-10-27 19:43:12 +00:00
|
|
|
switch_snprintf(buf, B64BUFFLEN, "type %s | %s -f %s %s %s", filename, runtime.mailer_app, from, runtime.mailer_app_args, to);
|
2008-07-10 15:35:35 +00:00
|
|
|
#else
|
2009-10-27 19:43:12 +00:00
|
|
|
switch_snprintf(buf, B64BUFFLEN, "/bin/cat %s | %s -f %s %s %s", filename, runtime.mailer_app, from, runtime.mailer_app_args, to);
|
2008-07-10 15:35:35 +00:00
|
|
|
#endif
|
2008-09-26 15:50:12 +00:00
|
|
|
if (switch_system(buf, SWITCH_TRUE) < 0) {
|
2008-05-27 04:30:03 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Unable to execute command: %s\n", buf);
|
|
|
|
}
|
2007-10-12 17:12:31 +00:00
|
|
|
|
2007-12-17 21:08:24 +00:00
|
|
|
if (unlink(filename) != 0) {
|
2008-10-11 05:44:11 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Failed to delete file [%s]\n", filename);
|
2007-12-17 21:08:24 +00:00
|
|
|
}
|
2007-10-12 17:12:31 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
if (file) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Emailed file [%s] to [%s]\n", filename, to);
|
|
|
|
} else {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Emailed data to [%s]\n", to);
|
|
|
|
}
|
2007-10-12 17:12:31 +00:00
|
|
|
|
2009-10-01 22:43:44 +00:00
|
|
|
rval = SWITCH_TRUE;
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
end:
|
|
|
|
|
2009-10-01 22:43:44 +00:00
|
|
|
if (newfile) {
|
|
|
|
unlink(newfile);
|
|
|
|
free(newfile);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rval;
|
2007-10-12 17:12:31 +00:00
|
|
|
}
|
|
|
|
|
2007-11-08 23:46:26 +00:00
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_is_lan_addr(const char *ip)
|
|
|
|
{
|
2009-10-23 16:03:42 +00:00
|
|
|
if (zstr(ip))
|
2008-05-27 04:30:03 +00:00
|
|
|
return SWITCH_FALSE;
|
2007-11-08 23:46:26 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
return (strncmp(ip, "10.", 3) &&
|
2007-11-08 23:46:26 +00:00
|
|
|
strncmp(ip, "192.168.", 8) &&
|
|
|
|
strncmp(ip, "127.", 4) &&
|
|
|
|
strncmp(ip, "255.", 4) &&
|
|
|
|
strncmp(ip, "0.", 2) &&
|
|
|
|
strncmp(ip, "1.", 2) &&
|
|
|
|
strncmp(ip, "2.", 2) &&
|
|
|
|
strncmp(ip, "172.16.", 7) &&
|
|
|
|
strncmp(ip, "172.17.", 7) &&
|
|
|
|
strncmp(ip, "172.18.", 7) &&
|
|
|
|
strncmp(ip, "172.19.", 7) &&
|
2008-05-27 04:30:03 +00:00
|
|
|
strncmp(ip, "172.2", 5) && strncmp(ip, "172.30.", 7) && strncmp(ip, "172.31.", 7) && strncmp(ip, "192.0.2.", 8) && strncmp(ip, "169.254.", 8)
|
|
|
|
)? SWITCH_FALSE : SWITCH_TRUE;
|
2007-11-08 23:46:26 +00:00
|
|
|
}
|
|
|
|
|
2009-05-12 17:10:57 +00:00
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_ast2regex(const char *pat, char *rbuf, size_t len)
|
2007-11-09 19:25:46 +00:00
|
|
|
{
|
2009-05-12 17:10:57 +00:00
|
|
|
const char *p = pat;
|
2007-12-11 21:31:57 +00:00
|
|
|
|
|
|
|
if (!pat) {
|
|
|
|
return SWITCH_FALSE;
|
|
|
|
}
|
|
|
|
|
2007-11-09 19:25:46 +00:00
|
|
|
memset(rbuf, 0, len);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-11-09 21:38:51 +00:00
|
|
|
*(rbuf + strlen(rbuf)) = '^';
|
2007-11-09 19:25:46 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
while (p && *p) {
|
2007-11-09 19:25:46 +00:00
|
|
|
if (*p == 'N') {
|
|
|
|
strncat(rbuf, "[2-9]", len - strlen(rbuf));
|
|
|
|
} else if (*p == 'X') {
|
|
|
|
strncat(rbuf, "[0-9]", len - strlen(rbuf));
|
|
|
|
} else if (*p == 'Z') {
|
|
|
|
strncat(rbuf, "[1-9]", len - strlen(rbuf));
|
|
|
|
} else if (*p == '.') {
|
|
|
|
strncat(rbuf, ".*", len - strlen(rbuf));
|
|
|
|
} else if (strlen(rbuf) < len - 1) {
|
|
|
|
*(rbuf + strlen(rbuf)) = *p;
|
|
|
|
}
|
|
|
|
p++;
|
|
|
|
}
|
2007-11-09 21:38:51 +00:00
|
|
|
*(rbuf + strlen(rbuf)) = '$';
|
2007-11-09 19:25:46 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
return strcmp(pat, rbuf) ? SWITCH_TRUE : SWITCH_FALSE;
|
2007-11-09 19:25:46 +00:00
|
|
|
}
|
2007-11-08 23:46:26 +00:00
|
|
|
|
2007-11-09 15:26:32 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_replace_char(char *str, char from, char to, switch_bool_t dup)
|
|
|
|
{
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
if (dup) {
|
|
|
|
p = strdup(str);
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(p);
|
2007-11-09 15:26:32 +00:00
|
|
|
} else {
|
|
|
|
p = str;
|
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
for (; p && *p; p++) {
|
2007-11-09 15:26:32 +00:00
|
|
|
if (*p == from) {
|
|
|
|
*p = to;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2007-11-08 23:46:26 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_strip_spaces(const char *str)
|
|
|
|
{
|
|
|
|
const char *sp = str;
|
|
|
|
char *p, *s = NULL;
|
2008-05-19 18:43:01 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
if (!sp)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
while (*sp == ' ') {
|
2007-11-08 23:46:26 +00:00
|
|
|
sp++;
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-11-08 23:46:26 +00:00
|
|
|
s = strdup(sp);
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
if (!s)
|
|
|
|
return NULL;
|
2008-05-19 18:43:01 +00:00
|
|
|
|
2007-11-08 23:46:26 +00:00
|
|
|
p = s + (strlen(s) - 1);
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
while (*p == ' ') {
|
2007-11-08 23:46:26 +00:00
|
|
|
*p-- = '\0';
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-11-08 23:46:26 +00:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2010-03-08 08:12:12 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_strip_commas(char *in, char *out, switch_size_t len)
|
|
|
|
{
|
|
|
|
char *p = in, *q = out;
|
|
|
|
char *ret = out;
|
|
|
|
switch_size_t x = 0;
|
|
|
|
|
|
|
|
for (; p && *p; p++) {
|
|
|
|
if ((*p > 47 && *p < 58)) {
|
|
|
|
*q++ = *p;
|
|
|
|
} else if (*p != ',') {
|
|
|
|
ret = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (++x > len) {
|
|
|
|
ret = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(char *) switch_strip_nonnumerics(char *in, char *out, switch_size_t len)
|
|
|
|
{
|
|
|
|
char *p = in, *q = out;
|
|
|
|
char *ret = out;
|
|
|
|
switch_size_t x = 0;
|
|
|
|
/* valid are 0 - 9, period (.), minus (-), and plus (+) - remove all others */
|
|
|
|
for (; p && *p; p++) {
|
|
|
|
if ((*p > 47 && *p < 58) || *p == '.' || *p == '-' || *p == '+') {
|
|
|
|
*q++ = *p;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (++x > len) {
|
|
|
|
ret = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-11-13 03:47:07 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_separate_paren_args(char *str)
|
|
|
|
{
|
|
|
|
char *e, *args;
|
|
|
|
switch_size_t br;
|
|
|
|
|
|
|
|
if ((args = strchr(str, '('))) {
|
|
|
|
e = args - 1;
|
|
|
|
*args++ = '\0';
|
2008-05-27 04:30:03 +00:00
|
|
|
while (*e == ' ') {
|
2007-11-13 03:47:07 +00:00
|
|
|
*e-- = '\0';
|
|
|
|
}
|
|
|
|
e = args;
|
|
|
|
br = 1;
|
2008-05-27 04:30:03 +00:00
|
|
|
while (e && *e) {
|
2007-11-13 03:47:07 +00:00
|
|
|
if (*e == '(') {
|
|
|
|
br++;
|
|
|
|
} else if (br > 1 && *e == ')') {
|
|
|
|
br--;
|
|
|
|
} else if (br == 1 && *e == ')') {
|
|
|
|
*e = '\0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
e++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return args;
|
|
|
|
}
|
|
|
|
|
2007-11-08 23:46:26 +00:00
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_is_number(const char *str)
|
|
|
|
{
|
|
|
|
const char *p;
|
|
|
|
switch_bool_t r = SWITCH_TRUE;
|
|
|
|
|
2009-06-09 19:52:07 +00:00
|
|
|
if (*str == '-' || *str == '+') {
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
|
2007-11-08 23:46:26 +00:00
|
|
|
for (p = str; p && *p; p++) {
|
|
|
|
if (!(*p == '.' || (*p > 47 && *p < 58))) {
|
|
|
|
r = SWITCH_FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2007-11-15 16:22:18 +00:00
|
|
|
SWITCH_DECLARE(const char *) switch_stristr(const char *instr, const char *str)
|
2007-10-16 14:24:02 +00:00
|
|
|
{
|
2007-10-29 06:16:41 +00:00
|
|
|
/*
|
|
|
|
** Rev History: 16/07/97 Greg Thayer Optimized
|
|
|
|
** 07/04/95 Bob Stout ANSI-fy
|
|
|
|
** 02/03/94 Fred Cole Original
|
|
|
|
** 09/01/03 Bob Stout Bug fix (lines 40-41) per Fred Bulback
|
|
|
|
**
|
|
|
|
** Hereby donated to public domain.
|
|
|
|
*/
|
|
|
|
const char *pptr, *sptr, *start;
|
|
|
|
|
|
|
|
if (!str || !instr)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
for (start = str; *start; start++) {
|
|
|
|
/* find start of pattern in string */
|
2008-12-15 02:48:50 +00:00
|
|
|
for (; ((*start) && (switch_toupper(*start) != switch_toupper(*instr))); start++);
|
2007-10-29 06:16:41 +00:00
|
|
|
|
|
|
|
if (!*start)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
pptr = instr;
|
|
|
|
sptr = start;
|
|
|
|
|
2008-12-15 02:48:50 +00:00
|
|
|
while (switch_toupper(*sptr) == switch_toupper(*pptr)) {
|
2007-10-29 06:16:41 +00:00
|
|
|
sptr++;
|
|
|
|
pptr++;
|
|
|
|
|
|
|
|
/* if end of pattern then pattern was found */
|
|
|
|
if (!*pptr)
|
|
|
|
return (start);
|
|
|
|
|
|
|
|
if (!*sptr)
|
|
|
|
return NULL;
|
2007-10-24 23:20:47 +00:00
|
|
|
}
|
|
|
|
}
|
2007-10-16 14:24:02 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2009-06-03 22:30:09 +00:00
|
|
|
#ifdef HAVE_GETIFADDRS
|
2009-06-02 16:55:10 +00:00
|
|
|
#include <ifaddrs.h>
|
|
|
|
static int get_netmask(struct sockaddr_in *me, int *mask)
|
|
|
|
{
|
|
|
|
struct ifaddrs *ifaddrs, *i = NULL;
|
|
|
|
|
2009-06-02 18:30:18 +00:00
|
|
|
if (!me || getifaddrs(&ifaddrs) < 0) {
|
2009-06-02 16:55:10 +00:00
|
|
|
return -1;
|
2010-02-06 03:38:24 +00:00
|
|
|
}
|
2009-06-02 16:55:10 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
for (i = ifaddrs; i; i = i->ifa_next) {
|
|
|
|
struct sockaddr_in *s = (struct sockaddr_in *) i->ifa_addr;
|
|
|
|
struct sockaddr_in *m = (struct sockaddr_in *) i->ifa_netmask;
|
2009-06-02 16:55:10 +00:00
|
|
|
|
2009-06-02 18:30:18 +00:00
|
|
|
if (s && m && s->sin_addr.s_addr == me->sin_addr.s_addr) {
|
2009-06-02 16:55:10 +00:00
|
|
|
*mask = m->sin_addr.s_addr;
|
2009-06-09 17:25:20 +00:00
|
|
|
freeifaddrs(ifaddrs);
|
2009-06-02 16:55:10 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-06-03 22:30:09 +00:00
|
|
|
freeifaddrs(ifaddrs);
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-06-02 16:55:10 +00:00
|
|
|
return -2;
|
|
|
|
}
|
2009-06-03 22:30:09 +00:00
|
|
|
#elif defined(__linux__)
|
|
|
|
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <net/if.h>
|
|
|
|
static int get_netmask(struct sockaddr_in *me, int *mask)
|
|
|
|
{
|
|
|
|
|
|
|
|
static struct ifreq ifreqs[20] = { {{{0}}} };
|
|
|
|
struct ifconf ifconf;
|
2010-02-06 03:38:24 +00:00
|
|
|
int nifaces, i;
|
2009-06-03 22:30:09 +00:00
|
|
|
int sock;
|
|
|
|
int r = -1;
|
2010-02-06 03:38:24 +00:00
|
|
|
|
|
|
|
memset(&ifconf, 0, sizeof(ifconf));
|
|
|
|
ifconf.ifc_buf = (char *) (ifreqs);
|
2009-06-03 22:30:09 +00:00
|
|
|
ifconf.ifc_len = sizeof(ifreqs);
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
|
|
|
|
if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
|
2009-06-03 22:30:09 +00:00
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ioctl(sock, SIOCGIFCONF, (char *) &ifconf) < 0) {
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
nifaces = ifconf.ifc_len / sizeof(struct ifreq);
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
for (i = 0; i < nifaces; i++) {
|
2009-06-03 22:30:09 +00:00
|
|
|
struct sockaddr_in *sin = NULL;
|
|
|
|
struct in_addr ip;
|
|
|
|
|
|
|
|
ioctl(sock, SIOCGIFADDR, &ifreqs[i]);
|
2010-02-06 03:38:24 +00:00
|
|
|
sin = (struct sockaddr_in *) &ifreqs[i].ifr_addr;
|
2009-06-03 22:30:09 +00:00
|
|
|
ip = sin->sin_addr;
|
|
|
|
|
|
|
|
if (ip.s_addr == me->sin_addr.s_addr) {
|
|
|
|
ioctl(sock, SIOCGIFNETMASK, &ifreqs[i]);
|
2010-02-06 03:38:24 +00:00
|
|
|
sin = (struct sockaddr_in *) &ifreqs[i].ifr_addr;
|
2009-06-03 22:30:09 +00:00
|
|
|
//mask = sin->sin_addr;
|
|
|
|
*mask = sin->sin_addr.s_addr;
|
|
|
|
r = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
|
|
|
end:
|
|
|
|
|
2009-06-03 22:30:09 +00:00
|
|
|
close(sock);
|
|
|
|
return r;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#elif defined(WIN32)
|
|
|
|
|
|
|
|
static int get_netmask(struct sockaddr_in *me, int *mask)
|
|
|
|
{
|
2009-06-04 05:23:19 +00:00
|
|
|
SOCKET sock = WSASocket(AF_INET, SOCK_DGRAM, 0, 0, 0, 0);
|
|
|
|
INTERFACE_INFO interfaces[20];
|
|
|
|
unsigned long bytes;
|
|
|
|
int interface_count, x;
|
2009-09-16 13:55:23 +00:00
|
|
|
int r = -1;
|
2009-06-04 05:23:19 +00:00
|
|
|
|
|
|
|
*mask = 0;
|
|
|
|
|
|
|
|
if (sock == SOCKET_ERROR) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (WSAIoctl(sock, SIO_GET_INTERFACE_LIST, 0, 0, &interfaces, sizeof(interfaces), &bytes, 0, 0) == SOCKET_ERROR) {
|
2009-09-16 13:55:23 +00:00
|
|
|
r = -1;
|
|
|
|
goto end;
|
2009-06-04 05:23:19 +00:00
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
interface_count = bytes / sizeof(INTERFACE_INFO);
|
2009-06-04 05:23:19 +00:00
|
|
|
|
|
|
|
for (x = 0; x < interface_count; ++x) {
|
2010-02-06 03:38:24 +00:00
|
|
|
struct sockaddr_in *addr = (struct sockaddr_in *) &(interfaces[x].iiAddress);
|
2009-06-04 05:23:19 +00:00
|
|
|
|
|
|
|
if (addr->sin_addr.s_addr == me->sin_addr.s_addr) {
|
2010-02-06 03:38:24 +00:00
|
|
|
struct sockaddr_in *netmask = (struct sockaddr_in *) &(interfaces[x].iiNetmask);
|
2009-06-04 05:23:19 +00:00
|
|
|
*mask = netmask->sin_addr.s_addr;
|
2009-09-16 13:55:23 +00:00
|
|
|
r = 0;
|
2009-06-04 05:23:19 +00:00
|
|
|
break;
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
}
|
2009-06-04 05:23:19 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
end:
|
2009-09-16 13:55:23 +00:00
|
|
|
closesocket(sock);
|
|
|
|
return r;
|
2009-06-03 22:30:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
static int get_netmask(struct sockaddr_in *me, int *mask)
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-06-02 16:55:10 +00:00
|
|
|
#endif
|
|
|
|
|
2009-08-27 22:58:55 +00:00
|
|
|
|
2009-08-28 20:30:03 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_resolve_host(const char *host, char *buf, size_t buflen)
|
2009-08-27 22:58:55 +00:00
|
|
|
{
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
struct addrinfo *ai;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
if ((err = getaddrinfo(host, 0, 0, &ai))) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
2009-08-27 22:58:55 +00:00
|
|
|
|
|
|
|
get_addr(buf, buflen, ai->ai_addr, sizeof(*ai->ai_addr));
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
freeaddrinfo(ai);
|
2009-08-27 22:58:55 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2009-08-27 22:58:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-06-02 16:55:10 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_find_local_ip(char *buf, int len, int *mask, int family)
|
2007-01-19 19:11:44 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
char *base;
|
2010-02-24 18:13:28 +00:00
|
|
|
const char *force_local_ip_v4 = switch_core_get_variable("force_local_ip_v4");
|
|
|
|
const char *force_local_ip_v6 = switch_core_get_variable("force_local_ip_v6");
|
2007-01-19 19:11:44 +00:00
|
|
|
|
|
|
|
#ifdef WIN32
|
2007-03-29 22:31:56 +00:00
|
|
|
SOCKET tmp_socket;
|
|
|
|
SOCKADDR_STORAGE l_address;
|
|
|
|
int l_address_len;
|
|
|
|
struct addrinfo *address_info;
|
2007-01-19 19:11:44 +00:00
|
|
|
#else
|
2007-01-19 19:20:28 +00:00
|
|
|
#ifdef __Darwin__
|
2007-03-29 22:31:56 +00:00
|
|
|
int ilen;
|
2007-01-19 19:11:44 +00:00
|
|
|
#else
|
2007-03-29 22:31:56 +00:00
|
|
|
unsigned int ilen;
|
2007-01-19 19:11:44 +00:00
|
|
|
#endif
|
2007-03-29 22:31:56 +00:00
|
|
|
int tmp_socket = -1, on = 1;
|
|
|
|
char abuf[25] = "";
|
2007-01-19 19:11:44 +00:00
|
|
|
#endif
|
|
|
|
|
2010-02-24 18:13:28 +00:00
|
|
|
switch (family) {
|
|
|
|
case AF_INET:
|
|
|
|
if (force_local_ip_v4) {
|
|
|
|
switch_copy_string(buf, force_local_ip_v4, len);
|
|
|
|
}
|
|
|
|
case AF_INET6:
|
|
|
|
if (force_local_ip_v6) {
|
|
|
|
switch_copy_string(buf, force_local_ip_v6, len);
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-28 08:47:55 +00:00
|
|
|
if (len < 16) {
|
|
|
|
return status;
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch (family) {
|
|
|
|
case AF_INET:
|
2008-07-22 17:19:26 +00:00
|
|
|
switch_copy_string(buf, "127.0.0.1", len);
|
2007-03-29 22:31:56 +00:00
|
|
|
base = "82.45.148.209";
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
2008-07-22 17:19:26 +00:00
|
|
|
switch_copy_string(buf, "::1", len);
|
2010-02-06 03:38:24 +00:00
|
|
|
base = "2001:503:BA3E::2:30"; // DNS Root server A
|
2007-03-29 22:31:56 +00:00
|
|
|
break;
|
2007-01-19 19:52:19 +00:00
|
|
|
default:
|
|
|
|
base = "127.0.0.1";
|
|
|
|
break;
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
2006-11-15 03:17:28 +00:00
|
|
|
|
2007-01-19 19:11:44 +00:00
|
|
|
#ifdef WIN32
|
2007-03-29 22:31:56 +00:00
|
|
|
tmp_socket = socket(family, SOCK_DGRAM, 0);
|
|
|
|
|
|
|
|
getaddrinfo(base, NULL, NULL, &address_info);
|
|
|
|
|
2007-10-18 20:15:35 +00:00
|
|
|
if (!address_info || WSAIoctl(tmp_socket,
|
2008-05-27 04:30:03 +00:00
|
|
|
SIO_ROUTING_INTERFACE_QUERY,
|
|
|
|
address_info->ai_addr, (DWORD) address_info->ai_addrlen, &l_address, sizeof(l_address), (LPDWORD) & l_address_len, NULL,
|
|
|
|
NULL)) {
|
2007-03-29 22:31:56 +00:00
|
|
|
|
|
|
|
closesocket(tmp_socket);
|
2007-10-18 20:15:35 +00:00
|
|
|
if (address_info)
|
|
|
|
freeaddrinfo(address_info);
|
2007-03-29 22:31:56 +00:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2009-06-02 16:55:10 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
closesocket(tmp_socket);
|
|
|
|
freeaddrinfo(address_info);
|
|
|
|
|
|
|
|
if (!getnameinfo((const struct sockaddr *) &l_address, l_address_len, buf, len, NULL, 0, NI_NUMERICHOST)) {
|
2010-02-06 03:38:24 +00:00
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
2009-06-04 05:23:19 +00:00
|
|
|
if (mask) {
|
|
|
|
get_netmask((struct sockaddr_in *) &l_address, mask);
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
2007-01-19 19:11:44 +00:00
|
|
|
#else
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch (family) {
|
|
|
|
case AF_INET:
|
|
|
|
{
|
|
|
|
struct sockaddr_in iface_out;
|
|
|
|
struct sockaddr_in remote;
|
|
|
|
memset(&remote, 0, sizeof(struct sockaddr_in));
|
|
|
|
|
|
|
|
remote.sin_family = AF_INET;
|
|
|
|
remote.sin_addr.s_addr = inet_addr(base);
|
|
|
|
remote.sin_port = htons(4242);
|
|
|
|
|
|
|
|
memset(&iface_out, 0, sizeof(iface_out));
|
|
|
|
tmp_socket = socket(AF_INET, SOCK_DGRAM, 0);
|
|
|
|
|
|
|
|
if (setsockopt(tmp_socket, SOL_SOCKET, SO_BROADCAST, &on, sizeof(on)) == -1) {
|
|
|
|
goto doh;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (connect(tmp_socket, (struct sockaddr *) &remote, sizeof(struct sockaddr_in)) == -1) {
|
|
|
|
goto doh;
|
|
|
|
}
|
|
|
|
|
|
|
|
ilen = sizeof(iface_out);
|
|
|
|
if (getsockname(tmp_socket, (struct sockaddr *) &iface_out, &ilen) == -1) {
|
|
|
|
goto doh;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iface_out.sin_addr.s_addr == 0) {
|
|
|
|
goto doh;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
switch_copy_string(buf, get_addr(abuf, sizeof(abuf), (struct sockaddr *) &iface_out, sizeof(iface_out)), len);
|
2009-06-02 16:55:10 +00:00
|
|
|
if (mask) {
|
2010-02-06 03:38:24 +00:00
|
|
|
get_netmask((struct sockaddr_in *) &iface_out, mask);
|
2009-06-02 16:55:10 +00:00
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
{
|
|
|
|
struct sockaddr_in6 iface_out;
|
|
|
|
struct sockaddr_in6 remote;
|
|
|
|
memset(&remote, 0, sizeof(struct sockaddr_in6));
|
|
|
|
|
|
|
|
remote.sin6_family = AF_INET6;
|
2008-07-03 18:50:15 +00:00
|
|
|
switch_inet_pton(AF_INET6, base, &remote.sin6_addr);
|
2007-03-29 22:31:56 +00:00
|
|
|
remote.sin6_port = htons(4242);
|
|
|
|
|
|
|
|
memset(&iface_out, 0, sizeof(iface_out));
|
|
|
|
tmp_socket = socket(AF_INET6, SOCK_DGRAM, 0);
|
|
|
|
|
2008-07-03 18:50:15 +00:00
|
|
|
if (connect(tmp_socket, (struct sockaddr *) &remote, sizeof(remote)) == -1) {
|
2007-03-29 22:31:56 +00:00
|
|
|
goto doh;
|
|
|
|
}
|
|
|
|
|
|
|
|
ilen = sizeof(iface_out);
|
|
|
|
if (getsockname(tmp_socket, (struct sockaddr *) &iface_out, &ilen) == -1) {
|
|
|
|
goto doh;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iface_out.sin6_addr.s6_addr == 0) {
|
|
|
|
goto doh;
|
|
|
|
}
|
|
|
|
|
|
|
|
inet_ntop(AF_INET6, (const void *) &iface_out.sin6_addr, buf, len - 1);
|
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
doh:
|
|
|
|
if (tmp_socket > 0) {
|
|
|
|
close(tmp_socket);
|
|
|
|
tmp_socket = -1;
|
|
|
|
}
|
2007-01-19 19:11:44 +00:00
|
|
|
#endif
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
return status;
|
2007-01-19 19:11:44 +00:00
|
|
|
}
|
|
|
|
|
2007-05-12 14:48:14 +00:00
|
|
|
SWITCH_DECLARE(switch_time_t) switch_str_time(const char *in)
|
2007-01-03 00:21:17 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_time_exp_t tm = { 0 };
|
|
|
|
int proceed = 0, ovector[30];
|
|
|
|
switch_regex_t *re = NULL;
|
|
|
|
char replace[1024] = "";
|
|
|
|
switch_time_t ret = 0;
|
|
|
|
char *pattern = "^(\\d+)-(\\d+)-(\\d+)\\s*(\\d*):{0,1}(\\d*):{0,1}(\\d*)";
|
|
|
|
|
2009-01-25 21:23:07 +00:00
|
|
|
switch_time_exp_lt(&tm, switch_micro_time_now());
|
2007-03-29 22:31:56 +00:00
|
|
|
tm.tm_year = tm.tm_mon = tm.tm_mday = tm.tm_hour = tm.tm_min = tm.tm_sec = 0;
|
|
|
|
|
|
|
|
if ((proceed = switch_regex_perform(in, pattern, &re, ovector, sizeof(ovector) / sizeof(ovector[0])))) {
|
|
|
|
|
|
|
|
if (proceed > 1) {
|
|
|
|
switch_regex_copy_substring(in, ovector, proceed, 1, replace, sizeof(replace));
|
|
|
|
tm.tm_year = atoi(replace) - 1900;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (proceed > 2) {
|
|
|
|
switch_regex_copy_substring(in, ovector, proceed, 2, replace, sizeof(replace));
|
|
|
|
tm.tm_mon = atoi(replace) - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (proceed > 3) {
|
|
|
|
switch_regex_copy_substring(in, ovector, proceed, 3, replace, sizeof(replace));
|
|
|
|
tm.tm_mday = atoi(replace);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (proceed > 4) {
|
|
|
|
switch_regex_copy_substring(in, ovector, proceed, 4, replace, sizeof(replace));
|
|
|
|
tm.tm_hour = atoi(replace);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (proceed > 5) {
|
|
|
|
switch_regex_copy_substring(in, ovector, proceed, 5, replace, sizeof(replace));
|
|
|
|
tm.tm_min = atoi(replace);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (proceed > 6) {
|
|
|
|
switch_regex_copy_substring(in, ovector, proceed, 6, replace, sizeof(replace));
|
|
|
|
tm.tm_sec = atoi(replace);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_time_exp_gmt_get(&ret, &tm);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
/* possible else with more patterns later */
|
|
|
|
return ret;
|
2007-01-03 00:21:17 +00:00
|
|
|
}
|
|
|
|
|
2007-05-12 14:48:14 +00:00
|
|
|
SWITCH_DECLARE(const char *) switch_priority_name(switch_priority_t priority)
|
2006-02-26 00:12:17 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
switch (priority) { /*lol */
|
2006-02-26 00:12:17 +00:00
|
|
|
case SWITCH_PRIORITY_NORMAL:
|
|
|
|
return "NORMAL";
|
|
|
|
case SWITCH_PRIORITY_LOW:
|
|
|
|
return "LOW";
|
|
|
|
case SWITCH_PRIORITY_HIGH:
|
|
|
|
return "HIGH";
|
|
|
|
default:
|
|
|
|
return "INVALID";
|
|
|
|
}
|
|
|
|
}
|
2005-12-06 17:18:56 +00:00
|
|
|
|
2006-02-23 22:41:08 +00:00
|
|
|
static char RFC2833_CHARS[] = "0123456789*#ABCDF";
|
|
|
|
|
2009-06-10 00:07:10 +00:00
|
|
|
#ifdef _MSC_VER
|
2007-12-04 03:43:32 +00:00
|
|
|
/* Copyright (c) 1996 by Internet Software Consortium.
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS
|
|
|
|
* ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE
|
|
|
|
* CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
|
|
|
|
* PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
|
|
|
|
* ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
|
|
|
* SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* WARNING: Don't even consider trying to compile this on a system where
|
|
|
|
* sizeof(int) < 4. sizeof(int) > 4 is fine; all the world's not a VAX.
|
|
|
|
*/
|
|
|
|
|
2007-12-11 10:01:55 +00:00
|
|
|
static const char *switch_inet_ntop4(const unsigned char *src, char *dst, size_t size);
|
2007-12-04 03:43:32 +00:00
|
|
|
#if HAVE_SIN6
|
2007-12-11 10:01:55 +00:00
|
|
|
static const char *switch_inet_ntop6(const unsigned char *src, char *dst, size_t size);
|
2007-12-04 03:43:32 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* char *
|
|
|
|
* inet_ntop(af, src, dst, size)
|
|
|
|
* convert a network format address to presentation format.
|
|
|
|
* return:
|
|
|
|
* pointer to presentation format address (`dst'), or NULL (see errno).
|
|
|
|
* author:
|
|
|
|
* Paul Vixie, 1996.
|
|
|
|
*/
|
2009-06-10 00:04:29 +00:00
|
|
|
SWITCH_DECLARE(const char *) switch_inet_ntop(int af, void const *src, char *dst, size_t size)
|
2007-12-04 03:43:32 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
switch (af) {
|
|
|
|
case AF_INET:
|
2007-12-11 10:01:55 +00:00
|
|
|
return switch_inet_ntop4(src, dst, size);
|
2007-12-04 03:43:32 +00:00
|
|
|
#if HAVE_SIN6
|
|
|
|
case AF_INET6:
|
2007-12-11 10:01:55 +00:00
|
|
|
return switch_inet_ntop6(src, dst, size);
|
2007-12-04 03:43:32 +00:00
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
/* NOTREACHED */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* const char *
|
|
|
|
* inet_ntop4(src, dst, size)
|
|
|
|
* format an IPv4 address, more or less like inet_ntoa()
|
|
|
|
* return:
|
|
|
|
* `dst' (as a const)
|
|
|
|
* notes:
|
|
|
|
* (1) uses no statics
|
|
|
|
* (2) takes a unsigned char* not an in_addr as input
|
|
|
|
* author:
|
|
|
|
* Paul Vixie, 1996.
|
|
|
|
*/
|
2008-05-27 04:30:03 +00:00
|
|
|
static const char *switch_inet_ntop4(const unsigned char *src, char *dst, size_t size)
|
2007-12-04 03:43:32 +00:00
|
|
|
{
|
|
|
|
static const char fmt[] = "%u.%u.%u.%u";
|
|
|
|
char tmp[sizeof "255.255.255.255"];
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
if (switch_snprintf(tmp, sizeof tmp, fmt, src[0], src[1], src[2], src[3]) >= (int) size) {
|
2007-12-04 03:43:32 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return strcpy(dst, tmp);
|
|
|
|
}
|
|
|
|
|
2010-05-11 17:07:13 +00:00
|
|
|
#if HAVE_SIN6 || defined(NTDDI_VERSION)
|
2007-12-04 03:43:32 +00:00
|
|
|
/* const char *
|
|
|
|
* inet_ntop6(src, dst, size)
|
|
|
|
* convert IPv6 binary address into presentation (printable) format
|
|
|
|
* author:
|
|
|
|
* Paul Vixie, 1996.
|
|
|
|
*/
|
2008-05-27 04:30:03 +00:00
|
|
|
static const char *switch_inet_ntop6(unsigned char const *src, char *dst, size_t size)
|
2007-12-04 03:43:32 +00:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Note that int32_t and int16_t need only be "at least" large enough
|
|
|
|
* to contain a value of the specified size. On some systems, like
|
|
|
|
* Crays, there is no such thing as an integer variable with 16 bits.
|
|
|
|
* Keep this in mind if you think this function should have been coded
|
|
|
|
* to use pointer overlays. All the world's not a VAX.
|
|
|
|
*/
|
|
|
|
char tmp[sizeof "ffff:ffff:ffff:ffff:ffff:ffff:255.255.255.255"], *tp;
|
2008-05-27 04:30:03 +00:00
|
|
|
struct {
|
|
|
|
int base, len;
|
|
|
|
} best = {
|
|
|
|
-1, 0}, cur = {
|
|
|
|
-1, 0};
|
2007-12-04 03:43:32 +00:00
|
|
|
unsigned int words[8];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Preprocess:
|
2008-05-27 04:30:03 +00:00
|
|
|
* Copy the input (bytewise) array into a wordwise array.
|
|
|
|
* Find the longest run of 0x00's in src[] for :: shorthanding.
|
2007-12-04 03:43:32 +00:00
|
|
|
*/
|
|
|
|
for (i = 0; i < 16; i += 2)
|
|
|
|
words[i / 2] = (src[i] << 8) | (src[i + 1]);
|
|
|
|
best.base = -1;
|
|
|
|
cur.base = -1;
|
|
|
|
for (i = 0; i < 8; i++) {
|
|
|
|
if (words[i] == 0) {
|
|
|
|
if (cur.base == -1)
|
|
|
|
cur.base = i, cur.len = 1;
|
|
|
|
else
|
|
|
|
cur.len++;
|
|
|
|
} else {
|
|
|
|
if (cur.base != -1) {
|
|
|
|
if (best.base == -1 || cur.len > best.len)
|
|
|
|
best = cur;
|
|
|
|
cur.base = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (cur.base != -1) {
|
|
|
|
if (best.base == -1 || cur.len > best.len)
|
|
|
|
best = cur;
|
|
|
|
}
|
|
|
|
if (best.base != -1 && best.len < 2)
|
|
|
|
best.base = -1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Format the result.
|
|
|
|
*/
|
|
|
|
tp = tmp;
|
|
|
|
for (i = 0; i < 8; i++) {
|
|
|
|
/* Are we inside the best run of 0x00's? */
|
2008-05-27 04:30:03 +00:00
|
|
|
if (best.base != -1 && i >= best.base && i < (best.base + best.len)) {
|
2007-12-04 03:43:32 +00:00
|
|
|
if (i == best.base)
|
|
|
|
*tp++ = ':';
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* Are we following an initial run of 0x00s or any real hex? */
|
|
|
|
if (i != 0)
|
|
|
|
*tp++ = ':';
|
|
|
|
/* Is this address an encapsulated IPv4? */
|
2008-05-27 04:30:03 +00:00
|
|
|
if (i == 6 && best.base == 0 && (best.len == 6 || (best.len == 5 && words[5] == 0xffff))) {
|
2010-05-05 16:58:09 +00:00
|
|
|
if (!switch_inet_ntop4(src + 12, tp, sizeof tmp - (tp - tmp)))
|
2007-12-04 03:43:32 +00:00
|
|
|
return (NULL);
|
|
|
|
tp += strlen(tp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
tp += sprintf(tp, "%x", words[i]);
|
|
|
|
}
|
|
|
|
/* Was it a trailing run of 0x00's? */
|
|
|
|
if (best.base != -1 && (best.base + best.len) == 8)
|
|
|
|
*tp++ = ':';
|
|
|
|
*tp++ = '\0';
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check for overflow, copy, and we're done.
|
|
|
|
*/
|
2008-05-27 04:30:03 +00:00
|
|
|
if ((size_t) (tp - tmp) >= size) {
|
2007-12-04 03:43:32 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return strcpy(dst, tmp);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2009-10-30 01:06:51 +00:00
|
|
|
SWITCH_DECLARE(int) get_addr_int(switch_sockaddr_t *sa)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
struct sockaddr_in *s = (struct sockaddr_in *) &sa->sa;
|
2009-10-30 01:06:51 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
return ntohs((unsigned short) s->sin_addr.s_addr);
|
2009-10-30 01:06:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_cmp_addr(switch_sockaddr_t *sa1, switch_sockaddr_t *sa2)
|
|
|
|
{
|
2009-10-30 14:26:17 +00:00
|
|
|
struct sockaddr_in *s1;
|
|
|
|
struct sockaddr_in *s2;
|
2009-10-30 01:06:51 +00:00
|
|
|
|
2009-10-30 14:26:17 +00:00
|
|
|
struct sockaddr_in6 *s16;
|
|
|
|
struct sockaddr_in6 *s26;
|
|
|
|
|
|
|
|
struct sockaddr *ss1;
|
|
|
|
struct sockaddr *ss2;
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
if (!(sa1 && sa2))
|
|
|
|
return 0;
|
2009-10-30 14:26:17 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
s1 = (struct sockaddr_in *) &sa1->sa;
|
|
|
|
s2 = (struct sockaddr_in *) &sa2->sa;
|
2009-10-30 14:26:17 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
s16 = (struct sockaddr_in6 *) &sa1->sa;
|
|
|
|
s26 = (struct sockaddr_in6 *) &sa2->sa;
|
2009-10-30 14:26:17 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
ss1 = (struct sockaddr *) &sa1->sa;
|
|
|
|
ss2 = (struct sockaddr *) &sa2->sa;
|
|
|
|
|
|
|
|
if (ss1->sa_family != ss2->sa_family)
|
|
|
|
return 0;
|
2009-10-30 14:26:17 +00:00
|
|
|
|
|
|
|
switch (ss1->sa_family) {
|
|
|
|
case AF_INET:
|
2010-02-06 03:38:24 +00:00
|
|
|
return (s1->sin_addr.s_addr == s2->sin_addr.s_addr && s1->sin_port == s2->sin_port);
|
2009-10-30 14:26:17 +00:00
|
|
|
case AF_INET6:
|
2009-11-02 16:46:47 +00:00
|
|
|
if (s16->sin6_addr.s6_addr && s26->sin6_addr.s6_addr) {
|
|
|
|
int i;
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
if (s16->sin6_port != s26->sin6_port)
|
|
|
|
return 0;
|
2009-11-02 16:46:47 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
if (*((int32_t *) s16->sin6_addr.s6_addr + i) != *((int32_t *) s26->sin6_addr.s6_addr + i))
|
|
|
|
return 0;
|
2009-11-02 16:46:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
2009-10-30 14:26:17 +00:00
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
|
|
|
return 0;
|
2009-10-30 01:06:51 +00:00
|
|
|
}
|
|
|
|
|
2010-04-30 20:01:32 +00:00
|
|
|
SWITCH_DECLARE(char *) get_addr6(char *buf, switch_size_t len, struct sockaddr_in6 *sa, socklen_t salen)
|
|
|
|
{
|
|
|
|
switch_assert(buf);
|
|
|
|
*buf = '\0';
|
|
|
|
|
|
|
|
if (sa) {
|
2010-05-11 17:07:13 +00:00
|
|
|
#if defined(NTDDI_VERSION)
|
2010-05-05 16:58:09 +00:00
|
|
|
switch_inet_ntop6((unsigned char*)sa, buf, len);
|
|
|
|
#else
|
2010-04-30 20:01:32 +00:00
|
|
|
inet_ntop(AF_INET6, sa, buf, len);
|
2010-05-05 16:58:09 +00:00
|
|
|
#endif
|
2010-04-30 20:01:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
2009-10-30 01:06:51 +00:00
|
|
|
|
2008-07-03 18:50:15 +00:00
|
|
|
SWITCH_DECLARE(char *) get_addr(char *buf, switch_size_t len, struct sockaddr *sa, socklen_t salen)
|
2006-11-15 03:17:28 +00:00
|
|
|
{
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(buf);
|
2007-12-04 00:58:51 +00:00
|
|
|
*buf = '\0';
|
2009-10-30 01:06:51 +00:00
|
|
|
|
2008-07-03 18:50:15 +00:00
|
|
|
if (sa) {
|
2010-02-06 03:38:24 +00:00
|
|
|
getnameinfo(sa, salen, buf, (socklen_t) len, NULL, 0, NI_NUMERICHOST);
|
2007-12-06 22:26:37 +00:00
|
|
|
}
|
2006-11-15 03:17:28 +00:00
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2008-07-03 18:50:15 +00:00
|
|
|
SWITCH_DECLARE(unsigned short) get_port(struct sockaddr *sa)
|
|
|
|
{
|
|
|
|
unsigned short port = 0;
|
|
|
|
if (sa) {
|
|
|
|
switch (sa->sa_family) {
|
2010-02-06 03:38:24 +00:00
|
|
|
case AF_INET:
|
|
|
|
port = ntohs(((struct sockaddr_in *) sa)->sin_port);
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
port = ntohs(((struct sockaddr_in6 *) sa)->sin6_port);
|
|
|
|
break;
|
2008-07-03 18:50:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(int) switch_build_uri(char *uri, switch_size_t size, const char *scheme, const char *user, const switch_sockaddr_t *sa, int flags)
|
2008-07-03 18:50:15 +00:00
|
|
|
{
|
|
|
|
char host[NI_MAXHOST], serv[NI_MAXSERV];
|
2009-05-19 20:25:59 +00:00
|
|
|
struct sockaddr_in6 si6;
|
2008-07-03 18:50:15 +00:00
|
|
|
const struct sockaddr *addr;
|
|
|
|
const char *colon;
|
|
|
|
|
|
|
|
if (flags & SWITCH_URI_NO_SCOPE && sa->family == AF_INET6) {
|
2009-05-19 20:25:59 +00:00
|
|
|
memcpy(&si6, &sa->sa, sa->salen);
|
|
|
|
si6.sin6_scope_id = 0;
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
addr = (const struct sockaddr *) &si6;
|
2008-07-03 18:50:15 +00:00
|
|
|
} else {
|
2010-02-06 03:38:24 +00:00
|
|
|
addr = (const struct sockaddr *) (intptr_t) & sa->sa;
|
2008-07-03 18:50:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (getnameinfo(addr, sa->salen, host, sizeof(host), serv, sizeof(serv),
|
2010-02-06 03:38:24 +00:00
|
|
|
((flags & SWITCH_URI_NUMERIC_HOST) ? NI_NUMERICHOST : 0) | ((flags & SWITCH_URI_NUMERIC_PORT) ? NI_NUMERICSERV : 0)) != 0) {
|
|
|
|
return 0;
|
2008-07-03 18:50:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
colon = strchr(host, ':');
|
|
|
|
|
|
|
|
return switch_snprintf(uri, size, "%s:%s%s%s%s%s%s%s", scheme,
|
2010-02-06 03:38:24 +00:00
|
|
|
user ? user : "", user ? "@" : "", colon ? "[" : "", host, colon ? "]" : "", serv[0] ? ":" : "", serv[0] ? serv : "");
|
2008-07-03 18:50:15 +00:00
|
|
|
}
|
|
|
|
|
2006-02-23 22:41:08 +00:00
|
|
|
SWITCH_DECLARE(char) switch_rfc2833_to_char(int event)
|
|
|
|
{
|
2006-08-22 21:18:36 +00:00
|
|
|
if (event > -1 && event < (int32_t) sizeof(RFC2833_CHARS)) {
|
2006-03-30 23:02:50 +00:00
|
|
|
return RFC2833_CHARS[event];
|
|
|
|
}
|
|
|
|
return '\0';
|
2006-02-23 22:41:08 +00:00
|
|
|
}
|
|
|
|
|
2006-02-24 00:02:02 +00:00
|
|
|
SWITCH_DECLARE(unsigned char) switch_char_to_rfc2833(char key)
|
2006-02-23 22:41:08 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
char *c;
|
2006-03-30 23:02:50 +00:00
|
|
|
unsigned char counter = 0;
|
2006-02-23 22:41:08 +00:00
|
|
|
|
2008-12-15 02:48:50 +00:00
|
|
|
key = (char) switch_toupper(key);
|
2007-03-29 22:31:56 +00:00
|
|
|
for (c = RFC2833_CHARS; *c; c++) {
|
|
|
|
if (*c == key) {
|
|
|
|
return counter;
|
|
|
|
}
|
2006-03-30 23:02:50 +00:00
|
|
|
counter++;
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
return '\0';
|
2006-02-23 22:41:08 +00:00
|
|
|
}
|
|
|
|
|
2007-05-12 14:48:14 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_escape_char(switch_memory_pool_t *pool, char *in, const char *delim, char esc)
|
2006-10-06 22:39:49 +00:00
|
|
|
{
|
2007-05-12 14:48:14 +00:00
|
|
|
char *data;
|
|
|
|
const char *p, *d;
|
2007-03-29 22:31:56 +00:00
|
|
|
int count = 1, i = 0;
|
|
|
|
|
|
|
|
p = in;
|
|
|
|
while (*p) {
|
|
|
|
d = delim;
|
|
|
|
while (*d) {
|
|
|
|
if (*p == *d) {
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
d++;
|
|
|
|
}
|
|
|
|
p++;
|
|
|
|
}
|
2006-10-06 22:39:49 +00:00
|
|
|
|
|
|
|
if (count == 1) {
|
|
|
|
return in;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = switch_core_alloc(pool, strlen(in) + count);
|
2007-03-29 22:31:56 +00:00
|
|
|
|
|
|
|
p = in;
|
|
|
|
while (*p) {
|
|
|
|
d = delim;
|
|
|
|
while (*d) {
|
|
|
|
if (*p == *d) {
|
|
|
|
data[i++] = esc;
|
|
|
|
}
|
|
|
|
d++;
|
|
|
|
}
|
|
|
|
data[i++] = *p;
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
return data;
|
2006-10-06 22:39:49 +00:00
|
|
|
}
|
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
/* Helper function used when separating strings to unescape a character. The
|
|
|
|
supported characters are:
|
2006-10-06 22:39:49 +00:00
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
\n linefeed
|
|
|
|
\r carriage return
|
|
|
|
\t tab
|
|
|
|
\s space
|
|
|
|
|
|
|
|
Any other character is returned as it was received. */
|
|
|
|
static char unescape_char(char escaped)
|
|
|
|
{
|
|
|
|
char unescaped;
|
2008-01-19 21:59:06 +00:00
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
switch (escaped) {
|
2008-05-27 04:30:03 +00:00
|
|
|
case 'n':
|
|
|
|
unescaped = '\n';
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
unescaped = '\r';
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
unescaped = '\t';
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
unescaped = ' ';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
unescaped = escaped;
|
2008-01-16 20:18:09 +00:00
|
|
|
}
|
|
|
|
return unescaped;
|
|
|
|
}
|
|
|
|
|
2009-12-03 02:37:05 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_escape_string(const char *in, char *out, switch_size_t outlen)
|
|
|
|
{
|
|
|
|
const char *p;
|
|
|
|
char *o = out;
|
2010-02-06 03:38:24 +00:00
|
|
|
|
|
|
|
for (p = in; *p; p++) {
|
|
|
|
switch (*p) {
|
|
|
|
case '\n':
|
|
|
|
*o++ = '\\';
|
|
|
|
*o++ = 'n';
|
|
|
|
break;
|
|
|
|
case '\r':
|
|
|
|
*o++ = '\\';
|
|
|
|
*o++ = 'r';
|
|
|
|
break;
|
|
|
|
case '\t':
|
|
|
|
*o++ = '\\';
|
|
|
|
*o++ = 't';
|
|
|
|
break;
|
|
|
|
case ' ':
|
|
|
|
*o++ = '\\';
|
|
|
|
*o++ = 's';
|
|
|
|
break;
|
|
|
|
case '$':
|
|
|
|
*o++ = '\\';
|
|
|
|
*o++ = '$';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*o++ = *p;
|
|
|
|
break;
|
2009-12-03 02:37:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*o++ = '\0';
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
return out;
|
2009-12-03 02:37:05 +00:00
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_escape_string_pool(const char *in, switch_memory_pool_t *pool)
|
2009-12-03 02:37:05 +00:00
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
int len = strlen(in) * 2;
|
2009-12-03 02:37:05 +00:00
|
|
|
char *buf = switch_core_alloc(pool, len);
|
|
|
|
return switch_escape_string(in, buf, len);
|
|
|
|
}
|
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
/* Helper function used when separating strings to remove quotes, leading /
|
|
|
|
trailing spaces, and to convert escaped characters. */
|
2008-01-22 14:16:45 +00:00
|
|
|
static char *cleanup_separated_string(char *str, char delim)
|
2005-12-06 17:18:56 +00:00
|
|
|
{
|
2006-08-17 00:53:09 +00:00
|
|
|
char *ptr;
|
2008-01-16 20:18:09 +00:00
|
|
|
char *dest;
|
|
|
|
char *start;
|
|
|
|
char *end = NULL;
|
|
|
|
int inside_quotes = 0;
|
2005-12-06 17:18:56 +00:00
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
/* Skip initial whitespace */
|
|
|
|
for (ptr = str; *ptr == ' '; ++ptr) {
|
2005-12-06 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
for (start = dest = ptr; *ptr; ++ptr) {
|
2008-01-19 21:54:11 +00:00
|
|
|
char e;
|
|
|
|
int esc = 0;
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2008-01-19 21:54:11 +00:00
|
|
|
if (*ptr == ESCAPE_META) {
|
2008-05-27 04:30:03 +00:00
|
|
|
e = *(ptr + 1);
|
2010-01-26 18:20:09 +00:00
|
|
|
if (e == '\'' || e == '"' || (delim && e == delim) || e == ESCAPE_META || (e = unescape_char(*(ptr + 1))) != *(ptr + 1)) {
|
2008-01-19 21:54:11 +00:00
|
|
|
++ptr;
|
2010-01-26 18:20:09 +00:00
|
|
|
*dest++ = e;
|
2008-01-16 20:18:09 +00:00
|
|
|
end = dest;
|
2008-01-19 21:54:11 +00:00
|
|
|
esc++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!esc) {
|
|
|
|
if (*ptr == '\'') {
|
|
|
|
inside_quotes = (1 - inside_quotes);
|
|
|
|
} else {
|
|
|
|
*dest++ = *ptr;
|
|
|
|
if (*ptr != ' ' || inside_quotes) {
|
|
|
|
end = dest;
|
|
|
|
}
|
2008-01-16 20:18:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (end) {
|
|
|
|
*end = '\0';
|
|
|
|
}
|
2010-01-26 18:20:09 +00:00
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
return start;
|
|
|
|
}
|
2005-12-06 17:18:56 +00:00
|
|
|
|
2009-11-12 22:21:30 +00:00
|
|
|
SWITCH_DECLARE(unsigned int) switch_separate_string_string(char *buf, char *delim, char **array, unsigned int arraylen)
|
|
|
|
{
|
|
|
|
unsigned int count = 0;
|
|
|
|
char *d;
|
|
|
|
size_t dlen = strlen(delim);
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-11-12 22:21:30 +00:00
|
|
|
array[count++] = buf;
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
while (count < arraylen && array[count - 1]) {
|
|
|
|
if ((d = strstr(array[count - 1], delim))) {
|
2009-11-12 22:21:30 +00:00
|
|
|
*d = '\0';
|
|
|
|
d += dlen;
|
|
|
|
array[count++] = d;
|
2010-02-06 03:38:24 +00:00
|
|
|
} else
|
|
|
|
break;
|
2009-11-12 22:21:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
/* Separate a string using a delimiter that is not a space */
|
2008-01-17 05:52:50 +00:00
|
|
|
static unsigned int separate_string_char_delim(char *buf, char delim, char **array, unsigned int arraylen)
|
2008-01-16 20:18:09 +00:00
|
|
|
{
|
|
|
|
enum tokenizer_state {
|
|
|
|
START,
|
|
|
|
FIND_DELIM
|
|
|
|
} state = START;
|
|
|
|
|
|
|
|
unsigned int count = 0;
|
|
|
|
char *ptr = buf;
|
2008-05-27 04:30:03 +00:00
|
|
|
int inside_quotes = 0;
|
2008-01-17 05:52:50 +00:00
|
|
|
unsigned int i;
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
while (*ptr && count < arraylen) {
|
|
|
|
switch (state) {
|
2008-05-27 04:30:03 +00:00
|
|
|
case START:
|
|
|
|
array[count++] = ptr;
|
|
|
|
state = FIND_DELIM;
|
|
|
|
break;
|
2005-12-06 17:18:56 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
case FIND_DELIM:
|
|
|
|
/* escaped characters are copied verbatim to the destination string */
|
|
|
|
if (*ptr == ESCAPE_META) {
|
2008-01-16 20:18:09 +00:00
|
|
|
++ptr;
|
2008-05-27 04:30:03 +00:00
|
|
|
} else if (*ptr == '\'') {
|
|
|
|
inside_quotes = (1 - inside_quotes);
|
|
|
|
} else if (*ptr == delim && !inside_quotes) {
|
|
|
|
*ptr = '\0';
|
|
|
|
state = START;
|
|
|
|
}
|
|
|
|
++ptr;
|
|
|
|
break;
|
2005-12-06 17:18:56 +00:00
|
|
|
}
|
|
|
|
}
|
2008-01-16 20:18:09 +00:00
|
|
|
/* strip quotes, escaped chars and leading / trailing spaces */
|
2010-01-26 18:20:09 +00:00
|
|
|
if (count > 1) {
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
array[i] = cleanup_separated_string(array[i], delim);
|
|
|
|
}
|
2005-12-06 17:18:56 +00:00
|
|
|
}
|
2008-01-16 20:18:09 +00:00
|
|
|
return count;
|
|
|
|
}
|
2005-12-06 17:18:56 +00:00
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
/* Separate a string using a delimiter that is a space */
|
2008-01-17 05:52:50 +00:00
|
|
|
static unsigned int separate_string_blank_delim(char *buf, char **array, unsigned int arraylen)
|
2008-01-16 20:18:09 +00:00
|
|
|
{
|
|
|
|
enum tokenizer_state {
|
|
|
|
START,
|
|
|
|
SKIP_INITIAL_SPACE,
|
|
|
|
FIND_DELIM,
|
|
|
|
SKIP_ENDING_SPACE
|
|
|
|
} state = START;
|
|
|
|
|
|
|
|
unsigned int count = 0;
|
|
|
|
char *ptr = buf;
|
2008-05-27 04:30:03 +00:00
|
|
|
int inside_quotes = 0;
|
2008-01-17 05:52:50 +00:00
|
|
|
unsigned int i;
|
2007-11-08 23:46:26 +00:00
|
|
|
|
2008-01-16 20:18:09 +00:00
|
|
|
while (*ptr && count < arraylen) {
|
|
|
|
switch (state) {
|
2008-05-27 04:30:03 +00:00
|
|
|
case START:
|
|
|
|
array[count++] = ptr;
|
|
|
|
state = SKIP_INITIAL_SPACE;
|
|
|
|
break;
|
2008-01-16 20:18:09 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
case SKIP_INITIAL_SPACE:
|
|
|
|
if (*ptr == ' ') {
|
|
|
|
++ptr;
|
|
|
|
} else {
|
|
|
|
state = FIND_DELIM;
|
|
|
|
}
|
|
|
|
break;
|
2008-01-16 20:18:09 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
case FIND_DELIM:
|
|
|
|
if (*ptr == ESCAPE_META) {
|
2008-01-16 20:18:09 +00:00
|
|
|
++ptr;
|
2008-05-27 04:30:03 +00:00
|
|
|
} else if (*ptr == '\'') {
|
|
|
|
inside_quotes = (1 - inside_quotes);
|
|
|
|
} else if (*ptr == ' ' && !inside_quotes) {
|
|
|
|
*ptr = '\0';
|
|
|
|
state = SKIP_ENDING_SPACE;
|
|
|
|
}
|
|
|
|
++ptr;
|
|
|
|
break;
|
2008-01-16 20:18:09 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
case SKIP_ENDING_SPACE:
|
|
|
|
if (*ptr == ' ') {
|
|
|
|
++ptr;
|
|
|
|
} else {
|
|
|
|
state = START;
|
|
|
|
}
|
|
|
|
break;
|
2006-10-31 21:38:06 +00:00
|
|
|
}
|
|
|
|
}
|
2008-01-16 20:18:09 +00:00
|
|
|
/* strip quotes, escaped chars and leading / trailing spaces */
|
2010-01-26 18:20:09 +00:00
|
|
|
if (count > 1) {
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
array[i] = cleanup_separated_string(array[i], 0);
|
|
|
|
}
|
2008-01-16 20:18:09 +00:00
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2008-01-17 05:52:50 +00:00
|
|
|
SWITCH_DECLARE(unsigned int) switch_separate_string(char *buf, char delim, char **array, unsigned int arraylen)
|
2008-01-16 20:18:09 +00:00
|
|
|
{
|
|
|
|
if (!buf || !array || !arraylen) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
memset(array, 0, arraylen * sizeof(*array));
|
2006-10-31 21:38:06 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
return (delim == ' ' ? separate_string_blank_delim(buf, array, arraylen) : separate_string_char_delim(buf, delim, array, arraylen));
|
2005-12-06 17:18:56 +00:00
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-02-13 21:03:06 +00:00
|
|
|
SWITCH_DECLARE(const char *) switch_cut_path(const char *in)
|
2006-01-06 02:01:11 +00:00
|
|
|
{
|
2007-02-13 21:03:06 +00:00
|
|
|
const char *p, *ret = in;
|
|
|
|
const char delims[] = "/\\";
|
|
|
|
const char *i;
|
2006-01-06 02:01:11 +00:00
|
|
|
|
2006-11-10 16:30:02 +00:00
|
|
|
if (in) {
|
|
|
|
for (i = delims; *i; i++) {
|
|
|
|
p = in;
|
|
|
|
while ((p = strchr(p, *i)) != 0) {
|
|
|
|
ret = ++p;
|
|
|
|
}
|
2006-01-06 02:01:11 +00:00
|
|
|
}
|
2006-11-10 16:30:02 +00:00
|
|
|
return ret;
|
|
|
|
} else {
|
|
|
|
return NULL;
|
2006-01-06 02:01:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-30 00:13:31 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_string_match(const char *string, size_t string_len, const char *search, size_t search_len)
|
2006-04-14 16:45:31 +00:00
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; (i < search_len) && (i < string_len); i++) {
|
|
|
|
if (string[i] != search[i]) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
|
|
|
|
2006-04-14 16:45:31 +00:00
|
|
|
if (i == search_len) {
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(char *) switch_string_replace(const char *string, const char *search, const char *replace)
|
|
|
|
{
|
|
|
|
size_t string_len = strlen(string);
|
|
|
|
size_t search_len = strlen(search);
|
|
|
|
size_t replace_len = strlen(replace);
|
|
|
|
size_t i, n;
|
|
|
|
size_t dest_len = 0;
|
2007-12-11 20:32:40 +00:00
|
|
|
char *dest, *tmp;
|
2007-03-29 22:31:56 +00:00
|
|
|
|
|
|
|
dest = (char *) malloc(sizeof(char));
|
2007-12-11 21:31:57 +00:00
|
|
|
switch_assert(dest);
|
2006-04-14 16:45:31 +00:00
|
|
|
|
|
|
|
for (i = 0; i < string_len; i++) {
|
|
|
|
if (switch_string_match(string + i, string_len - i, search, search_len) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
for (n = 0; n < replace_len; n++) {
|
|
|
|
dest[dest_len] = replace[n];
|
|
|
|
dest_len++;
|
2007-12-11 20:32:40 +00:00
|
|
|
tmp = (char *) realloc(dest, sizeof(char) * (dest_len + 1));
|
|
|
|
switch_assert(tmp);
|
|
|
|
dest = tmp;
|
2006-04-14 16:45:31 +00:00
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
i += search_len - 1;
|
2006-04-14 16:45:31 +00:00
|
|
|
} else {
|
|
|
|
dest[dest_len] = string[i];
|
|
|
|
dest_len++;
|
2007-12-11 20:32:40 +00:00
|
|
|
tmp = (char *) realloc(dest, sizeof(char) * (dest_len + 1));
|
|
|
|
switch_assert(tmp);
|
|
|
|
dest = tmp;
|
2006-04-14 16:45:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dest[dest_len] = 0;
|
|
|
|
return dest;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2009-09-23 17:32:01 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_util_quote_shell_arg(const char *string)
|
|
|
|
{
|
|
|
|
size_t string_len = strlen(string);
|
|
|
|
size_t i;
|
|
|
|
size_t n = 0;
|
2010-02-06 03:38:24 +00:00
|
|
|
size_t dest_len = string_len + 1; /* +1 for the opening quote */
|
2009-09-23 17:32:01 +00:00
|
|
|
char *dest, *tmp;
|
|
|
|
|
|
|
|
dest = (char *) malloc(sizeof(char) * dest_len);
|
|
|
|
switch_assert(dest);
|
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
dest[n++] = '"';
|
|
|
|
#else
|
|
|
|
dest[n++] = '\'';
|
|
|
|
#endif
|
|
|
|
|
|
|
|
for (i = 0; i < string_len; i++) {
|
|
|
|
switch (string[i]) {
|
|
|
|
#ifdef WIN32
|
|
|
|
case '"':
|
|
|
|
case '%':
|
|
|
|
dest[n++] = ' ';
|
|
|
|
break;
|
|
|
|
#else
|
|
|
|
case '\'':
|
|
|
|
/* We replace ' by '\'' */
|
2010-02-06 03:38:24 +00:00
|
|
|
dest_len += 3;
|
2009-09-23 17:32:01 +00:00
|
|
|
tmp = (char *) realloc(dest, sizeof(char) * (dest_len));
|
|
|
|
switch_assert(tmp);
|
|
|
|
dest = tmp;
|
|
|
|
dest[n++] = '\'';
|
|
|
|
dest[n++] = '\\';
|
|
|
|
dest[n++] = '\'';
|
|
|
|
dest[n++] = '\'';
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
dest[n++] = string[i];
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
dest_len += 2; /* +2 for the closing quote and the null character */
|
2009-09-23 17:32:01 +00:00
|
|
|
tmp = (char *) realloc(dest, sizeof(char) * (dest_len));
|
|
|
|
switch_assert(tmp);
|
|
|
|
dest = tmp;
|
|
|
|
#ifdef WIN32
|
|
|
|
dest[n++] = '"';
|
|
|
|
#else
|
|
|
|
dest[n++] = '\'';
|
|
|
|
#endif
|
|
|
|
dest[n++] = 0;
|
|
|
|
switch_assert(n == dest_len);
|
|
|
|
return dest;
|
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
SWITCH_DECLARE(int) switch_socket_waitfor(switch_pollfd_t *poll, int ms)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
|
|
|
int nsds = 0;
|
2006-01-03 01:17:59 +00:00
|
|
|
|
2006-09-15 21:43:18 +00:00
|
|
|
switch_poll(poll, 1, &nsds, ms);
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-01-03 01:17:59 +00:00
|
|
|
return nsds;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2005-12-23 03:39:33 +00:00
|
|
|
|
2010-03-24 19:04:19 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_url_encode(const char *url, char *buf, size_t len)
|
2006-05-10 15:47:54 +00:00
|
|
|
{
|
2007-11-01 11:28:26 +00:00
|
|
|
const char *p;
|
2007-03-29 22:31:56 +00:00
|
|
|
size_t x = 0;
|
|
|
|
const char urlunsafe[] = "\r\n \"#%&+:;<=>?@[\\]^`{|}";
|
|
|
|
const char hex[] = "0123456789ABCDEF";
|
2006-05-10 15:47:54 +00:00
|
|
|
|
2007-02-09 22:56:42 +00:00
|
|
|
if (!buf) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!url) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-05-21 21:31:17 +00:00
|
|
|
len--;
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
for (p = url; *p; p++) {
|
2008-05-21 21:31:17 +00:00
|
|
|
if (x >= len) {
|
|
|
|
break;
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
if (*p < ' ' || *p > '~' || strchr(urlunsafe, *p)) {
|
2009-03-03 14:17:28 +00:00
|
|
|
if ((x + 3) > len) {
|
2007-03-29 22:31:56 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
buf[x++] = '%';
|
2010-05-06 22:36:13 +00:00
|
|
|
buf[x++] = hex[(*p >> 4) & 0x0f];
|
2007-03-29 22:31:56 +00:00
|
|
|
buf[x++] = hex[*p & 0x0f];
|
|
|
|
} else {
|
|
|
|
buf[x++] = *p;
|
|
|
|
}
|
|
|
|
}
|
2008-05-21 21:31:17 +00:00
|
|
|
buf[x] = '\0';
|
|
|
|
|
2010-03-24 19:04:19 +00:00
|
|
|
return buf;
|
2006-05-10 15:47:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(char *) switch_url_decode(char *s)
|
|
|
|
{
|
|
|
|
char *o;
|
|
|
|
unsigned int tmp;
|
|
|
|
|
2009-10-23 16:03:42 +00:00
|
|
|
if (zstr(s)) {
|
2009-06-16 13:41:01 +00:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2006-05-10 15:47:54 +00:00
|
|
|
for (o = s; *s; s++, o++) {
|
|
|
|
if (*s == '%' && strlen(s) > 2 && sscanf(s + 1, "%2x", &tmp) == 1) {
|
2007-03-29 22:31:56 +00:00
|
|
|
*o = (char) tmp;
|
2006-05-10 15:47:54 +00:00
|
|
|
s += 2;
|
|
|
|
} else {
|
|
|
|
*o = *s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*o = '\0';
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2008-12-15 02:48:50 +00:00
|
|
|
|
|
|
|
/* Written by Marc Espie, public domain */
|
|
|
|
#define SWITCH_CTYPE_NUM_CHARS 256
|
|
|
|
|
|
|
|
const short _switch_C_toupper_[1 + SWITCH_CTYPE_NUM_CHARS] = {
|
|
|
|
EOF,
|
2010-02-06 03:38:24 +00:00
|
|
|
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
|
|
|
|
0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
|
|
|
|
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17,
|
|
|
|
0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
|
|
|
|
0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27,
|
|
|
|
0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
|
|
|
|
0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37,
|
|
|
|
0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
|
|
|
|
0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47,
|
|
|
|
0x48, 0x49, 0x4a, 0x4b, 0x4c, 0x4d, 0x4e, 0x4f,
|
|
|
|
0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57,
|
|
|
|
0x58, 0x59, 0x5a, 0x5b, 0x5c, 0x5d, 0x5e, 0x5f,
|
|
|
|
0x60, 'A', 'B', 'C', 'D', 'E', 'F', 'G',
|
|
|
|
'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O',
|
|
|
|
'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W',
|
|
|
|
'X', 'Y', 'Z', 0x7b, 0x7c, 0x7d, 0x7e, 0x7f,
|
|
|
|
0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87,
|
|
|
|
0x88, 0x89, 0x8a, 0x8b, 0x8c, 0x8d, 0x8e, 0x8f,
|
|
|
|
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97,
|
|
|
|
0x98, 0x99, 0x9a, 0x9b, 0x9c, 0x9d, 0x9e, 0x9f,
|
|
|
|
0xa0, 0xa1, 0xa2, 0xa3, 0xa4, 0xa5, 0xa6, 0xa7,
|
|
|
|
0xa8, 0xa9, 0xaa, 0xab, 0xac, 0xad, 0xae, 0xaf,
|
|
|
|
0xb0, 0xb1, 0xb2, 0xb3, 0xb4, 0xb5, 0xb6, 0xb7,
|
|
|
|
0xb8, 0xb9, 0xba, 0xbb, 0xbc, 0xbd, 0xbe, 0xbf,
|
|
|
|
0xc0, 0xc1, 0xc2, 0xc3, 0xc4, 0xc5, 0xc6, 0xc7,
|
|
|
|
0xc8, 0xc9, 0xca, 0xcb, 0xcc, 0xcd, 0xce, 0xcf,
|
|
|
|
0xd0, 0xd1, 0xd2, 0xd3, 0xd4, 0xd5, 0xd6, 0xd7,
|
|
|
|
0xd8, 0xd9, 0xda, 0xdb, 0xdc, 0xdd, 0xde, 0xdf,
|
|
|
|
0xe0, 0xe1, 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7,
|
|
|
|
0xe8, 0xe9, 0xea, 0xeb, 0xec, 0xed, 0xee, 0xef,
|
|
|
|
0xf0, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7,
|
|
|
|
0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff
|
2008-12-15 02:48:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const short *_switch_toupper_tab_ = _switch_C_toupper_;
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_toupper(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
if ((unsigned int) c > 255)
|
|
|
|
return (c);
|
2008-12-15 02:48:50 +00:00
|
|
|
if (c < -1)
|
|
|
|
return EOF;
|
2010-02-06 03:38:24 +00:00
|
|
|
return ((_switch_toupper_tab_ + 1)[c]);
|
2008-12-15 02:48:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const short _switch_C_tolower_[1 + SWITCH_CTYPE_NUM_CHARS] = {
|
|
|
|
EOF,
|
2010-02-06 03:38:24 +00:00
|
|
|
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
|
|
|
|
0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
|
|
|
|
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17,
|
|
|
|
0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
|
|
|
|
0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27,
|
|
|
|
0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
|
|
|
|
0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37,
|
|
|
|
0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
|
|
|
|
0x40, 'a', 'b', 'c', 'd', 'e', 'f', 'g',
|
|
|
|
'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o',
|
|
|
|
'p', 'q', 'r', 's', 't', 'u', 'v', 'w',
|
|
|
|
'x', 'y', 'z', 0x5b, 0x5c, 0x5d, 0x5e, 0x5f,
|
|
|
|
0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67,
|
|
|
|
0x68, 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f,
|
|
|
|
0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77,
|
|
|
|
0x78, 0x79, 0x7a, 0x7b, 0x7c, 0x7d, 0x7e, 0x7f,
|
|
|
|
0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87,
|
|
|
|
0x88, 0x89, 0x8a, 0x8b, 0x8c, 0x8d, 0x8e, 0x8f,
|
|
|
|
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97,
|
|
|
|
0x98, 0x99, 0x9a, 0x9b, 0x9c, 0x9d, 0x9e, 0x9f,
|
|
|
|
0xa0, 0xa1, 0xa2, 0xa3, 0xa4, 0xa5, 0xa6, 0xa7,
|
|
|
|
0xa8, 0xa9, 0xaa, 0xab, 0xac, 0xad, 0xae, 0xaf,
|
|
|
|
0xb0, 0xb1, 0xb2, 0xb3, 0xb4, 0xb5, 0xb6, 0xb7,
|
|
|
|
0xb8, 0xb9, 0xba, 0xbb, 0xbc, 0xbd, 0xbe, 0xbf,
|
|
|
|
0xc0, 0xc1, 0xc2, 0xc3, 0xc4, 0xc5, 0xc6, 0xc7,
|
|
|
|
0xc8, 0xc9, 0xca, 0xcb, 0xcc, 0xcd, 0xce, 0xcf,
|
|
|
|
0xd0, 0xd1, 0xd2, 0xd3, 0xd4, 0xd5, 0xd6, 0xd7,
|
|
|
|
0xd8, 0xd9, 0xda, 0xdb, 0xdc, 0xdd, 0xde, 0xdf,
|
|
|
|
0xe0, 0xe1, 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7,
|
|
|
|
0xe8, 0xe9, 0xea, 0xeb, 0xec, 0xed, 0xee, 0xef,
|
|
|
|
0xf0, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7,
|
|
|
|
0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff
|
2008-12-15 02:48:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const short *_switch_tolower_tab_ = _switch_C_tolower_;
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_tolower(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
if ((unsigned int) c > 255)
|
|
|
|
return (c);
|
2008-12-15 02:48:50 +00:00
|
|
|
if (c < -1)
|
|
|
|
return EOF;
|
2010-02-06 03:38:24 +00:00
|
|
|
return ((_switch_tolower_tab_ + 1)[c]);
|
2008-12-15 02:48:50 +00:00
|
|
|
}
|
|
|
|
|
2008-12-15 03:21:36 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1989 The Regents of the University of California.
|
|
|
|
* All rights reserved.
|
|
|
|
* (c) UNIX System Laboratories, Inc.
|
|
|
|
* All or some portions of this file are derived from material licensed
|
|
|
|
* to the University of California by American Telephone and Telegraph
|
|
|
|
* Co. or Unix System Laboratories, Inc. and are reproduced herein with
|
|
|
|
* the permission of UNIX System Laboratories, Inc.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. Neither the name of the University nor the names of its contributors
|
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#undef _U
|
|
|
|
#undef _L
|
|
|
|
#undef _N
|
|
|
|
#undef _S
|
|
|
|
#undef _P
|
|
|
|
#undef _C
|
|
|
|
#undef _X
|
|
|
|
#undef _B
|
|
|
|
|
|
|
|
#define _U 0x01
|
|
|
|
#define _L 0x02
|
|
|
|
#define _N 0x04
|
|
|
|
#define _S 0x08
|
|
|
|
#define _P 0x10
|
|
|
|
#define _C 0x20
|
|
|
|
#define _X 0x40
|
|
|
|
#define _B 0x80
|
|
|
|
|
2008-12-15 05:12:12 +00:00
|
|
|
const int _switch_C_ctype_[1 + SWITCH_CTYPE_NUM_CHARS] = {
|
2008-12-15 03:21:36 +00:00
|
|
|
0,
|
2010-02-06 03:38:24 +00:00
|
|
|
_C, _C, _C, _C, _C, _C, _C, _C,
|
|
|
|
_C, _C | _S, _C | _S, _C | _S, _C | _S, _C | _S, _C, _C,
|
|
|
|
_C, _C, _C, _C, _C, _C, _C, _C,
|
|
|
|
_C, _C, _C, _C, _C, _C, _C, _C,
|
|
|
|
_S | _B, _P, _P, _P, _P, _P, _P, _P,
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P,
|
|
|
|
_N, _N, _N, _N, _N, _N, _N, _N,
|
|
|
|
_N, _N, _P, _P, _P, _P, _P, _P,
|
|
|
|
_P, _U | _X, _U | _X, _U | _X, _U | _X, _U | _X, _U | _X, _U,
|
|
|
|
_U, _U, _U, _U, _U, _U, _U, _U,
|
|
|
|
_U, _U, _U, _U, _U, _U, _U, _U,
|
|
|
|
_U, _U, _U, _P, _P, _P, _P, _P,
|
|
|
|
_P, _L | _X, _L | _X, _L | _X, _L | _X, _L | _X, _L | _X, _L,
|
|
|
|
_L, _L, _L, _L, _L, _L, _L, _L,
|
|
|
|
_L, _L, _L, _L, _L, _L, _L, _L,
|
2008-12-15 03:21:36 +00:00
|
|
|
/* determine printability based on the IS0 8859 8-bit standard */
|
2010-02-06 03:38:24 +00:00
|
|
|
_L, _L, _L, _P, _P, _P, _P, _C,
|
|
|
|
|
|
|
|
_C, _C, _C, _C, _C, _C, _C, _C, /* 80 */
|
|
|
|
_C, _C, _C, _C, _C, _C, _C, _C, /* 88 */
|
|
|
|
_C, _C, _C, _C, _C, _C, _C, _C, /* 90 */
|
|
|
|
_C, _C, _C, _C, _C, _C, _C, _C, /* 98 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* A0 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* A8 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* B0 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* B8 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* C0 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* C8 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* D0 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* D8 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* E0 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* E8 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P, /* F0 */
|
|
|
|
_P, _P, _P, _P, _P, _P, _P, _P /* F8 */
|
2008-12-15 03:21:36 +00:00
|
|
|
};
|
|
|
|
|
2008-12-15 05:12:12 +00:00
|
|
|
const int *_switch_ctype_ = _switch_C_ctype_;
|
2008-12-15 03:21:36 +00:00
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_isalnum(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & (_U | _L | _N)));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_isalpha(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & (_U | _L)));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_iscntrl(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & _C));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_isdigit(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & _N));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_isgraph(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & (_P | _U | _L | _N)));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_islower(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & _L));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_isprint(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & (_P | _U | _L | _N | _B)));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_ispunct(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & _P));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_isspace(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & _S));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_isupper(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & _U));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_isxdigit(int c)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
return (c < 0 ? 0 : c > 255 ? 0 : ((_switch_ctype_ + 1)[(unsigned char) c] & (_N | _X)));
|
2008-12-15 03:21:36 +00:00
|
|
|
}
|
2008-12-15 02:48:50 +00:00
|
|
|
|
2009-07-06 18:31:23 +00:00
|
|
|
SWITCH_DECLARE(int) switch_number_cmp(const char *exp, int val)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
char *p;
|
|
|
|
|
|
|
|
if ((p = strchr(exp, '-'))) {
|
|
|
|
int min;
|
|
|
|
int max;
|
|
|
|
|
|
|
|
min = atol(exp);
|
|
|
|
p++;
|
|
|
|
max = atol(p);
|
|
|
|
|
|
|
|
if (val >= min && val <= max) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} else if ((p = strchr(exp, ','))) {
|
|
|
|
const char *cur = exp;
|
2009-07-06 18:31:23 +00:00
|
|
|
p++;
|
2010-02-06 03:38:24 +00:00
|
|
|
while (cur) {
|
|
|
|
if (atol(cur) == val) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
cur = p;
|
|
|
|
if (p && p + 1) {
|
|
|
|
if ((p = strchr((p + 1), ','))) {
|
2009-07-06 18:31:23 +00:00
|
|
|
p++;
|
|
|
|
}
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (atol(exp) == val) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2009-07-06 18:31:23 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-06-02 01:12:33 +00:00
|
|
|
SWITCH_DECLARE(int) switch_split_user_domain(char *in, char **user, char **domain)
|
2010-06-01 23:09:54 +00:00
|
|
|
{
|
|
|
|
char *p = NULL, *h = NULL, *u = in;
|
|
|
|
|
|
|
|
if (!in) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* First isolate the host part from the user part */
|
|
|
|
if ((h = strchr(u, '@'))) {
|
|
|
|
*h++ = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clean out the user part of its protocol prefix (if any) */
|
|
|
|
if ((p = strchr(u, ':'))) {
|
|
|
|
*p++ = '\0';
|
|
|
|
u = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clean out the host part of any suffix */
|
|
|
|
if (h) {
|
|
|
|
if ((p = strchr(h, ':'))) {
|
|
|
|
*p = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((p = strchr(h, ';'))) {
|
|
|
|
*p = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((p = strchr(h, ' '))) {
|
|
|
|
*p = '\0';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (user) {
|
|
|
|
*user = u;
|
|
|
|
}
|
|
|
|
if (domain) {
|
|
|
|
*domain = h;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2009-07-06 18:31:23 +00:00
|
|
|
|
2006-11-27 22:30:48 +00:00
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
2008-02-03 22:14:57 +00:00
|
|
|
* indent-tabs-mode:t
|
2006-11-27 22:30:48 +00:00
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
2008-07-03 19:12:26 +00:00
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4:
|
2006-11-27 22:30:48 +00:00
|
|
|
*/
|