2005-11-19 20:07:43 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
* Copyright (C) 2005/2006, Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
|
|
|
* Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* switch_utils.c -- Compatability and Helper Code
|
|
|
|
*
|
|
|
|
*/
|
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"
|
2006-02-26 00:12:17 +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)
|
|
|
|
{
|
|
|
|
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) {
|
|
|
|
while (l < 6) {
|
|
|
|
out[bytes++] = '=', l += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-10-12 17:12:31 +00:00
|
|
|
|
|
|
|
static int write_buf(int fd, char *buf)
|
|
|
|
{
|
|
|
|
|
|
|
|
int len = (int) strlen(buf);
|
|
|
|
if (fd && write(fd, buf, len) != len) {
|
|
|
|
close(fd);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_simple_email(char *to, char *from, char *headers, char *body, char *file)
|
|
|
|
{
|
|
|
|
char *bound = "XXXX_boundary_XXXX";
|
|
|
|
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];
|
|
|
|
char *path = NULL;
|
|
|
|
|
2007-11-05 22:36:59 +00:00
|
|
|
snprintf(filename, 80, "%smail.%d%04x", SWITCH_GLOBAL_dirs.temp_dir, (int)time(NULL), rand() & 0xffff);
|
2007-10-12 17:12:31 +00:00
|
|
|
|
|
|
|
if ((fd = open(filename, O_WRONLY | O_CREAT | O_TRUNC, 0644))) {
|
|
|
|
if (file) {
|
|
|
|
path = file;
|
|
|
|
if ((ifd = open(path, O_RDONLY)) < 1) {
|
|
|
|
return SWITCH_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
snprintf(buf, B64BUFFLEN, "MIME-Version: 1.0\nContent-Type: multipart/mixed; boundary=\"%s\"\n", bound);
|
|
|
|
if (!write_buf(fd, buf)) {
|
|
|
|
return SWITCH_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (headers && !write_buf(fd, headers))
|
|
|
|
return SWITCH_FALSE;
|
|
|
|
|
|
|
|
if (!write_buf(fd, "\n\n"))
|
|
|
|
return SWITCH_FALSE;
|
|
|
|
|
|
|
|
if (file) {
|
|
|
|
snprintf(buf, B64BUFFLEN, "--%s\nContent-Type: text/plain\n\n", bound);
|
|
|
|
if (!write_buf(fd, buf))
|
|
|
|
return SWITCH_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (body) {
|
|
|
|
if (!write_buf(fd, body)) {
|
|
|
|
return SWITCH_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (file) {
|
|
|
|
snprintf(buf, B64BUFFLEN, "\n\n--%s\nContent-Type: application/octet-stream\n"
|
|
|
|
"Content-Transfer-Encoding: base64\n"
|
|
|
|
"Content-Description: Sound attachment.\n" "Content-Disposition: attachment; filename=\"%s\"\n\n", bound, switch_cut_path(file));
|
|
|
|
if (!write_buf(fd, buf))
|
|
|
|
return SWITCH_FALSE;
|
|
|
|
|
|
|
|
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) {
|
|
|
|
return -1;
|
|
|
|
} 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) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (file) {
|
|
|
|
snprintf(buf, B64BUFFLEN, "\n\n--%s--\n.\n", bound);
|
|
|
|
if (!write_buf(fd, buf))
|
|
|
|
return SWITCH_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fd) {
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
if (ifd) {
|
|
|
|
close(ifd);
|
|
|
|
}
|
2007-11-05 19:13:17 +00:00
|
|
|
snprintf(buf, B64BUFFLEN, "/bin/cat %s | %s %s %s", filename, runtime.mailer_app, runtime.mailer_app_args, to);
|
2007-10-12 17:12:31 +00:00
|
|
|
if(system(buf)) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Unable to execute command: %s\n", buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
unlink(filename);
|
|
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_TRUE;
|
|
|
|
}
|
|
|
|
|
2007-11-08 23:46:26 +00:00
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_is_lan_addr(const char *ip)
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
strncmp(ip, "10.", 3) &&
|
|
|
|
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) &&
|
|
|
|
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-09 19:25:46 +00:00
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_ast2regex(char *pat, char *rbuf, size_t len)
|
|
|
|
{
|
|
|
|
char *p = pat;
|
|
|
|
memset(rbuf, 0, len);
|
2007-11-09 21:38:51 +00:00
|
|
|
|
|
|
|
*(rbuf + strlen(rbuf)) = '^';
|
2007-11-09 19:25:46 +00:00
|
|
|
|
|
|
|
while(p && *p) {
|
|
|
|
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
|
|
|
|
|
|
|
return strcmp(pat,rbuf) ? SWITCH_TRUE : SWITCH_FALSE;
|
|
|
|
}
|
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);
|
|
|
|
assert(p);
|
|
|
|
} else {
|
|
|
|
p = str;
|
|
|
|
}
|
|
|
|
|
|
|
|
for(;p && *p; p++) {
|
|
|
|
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;
|
|
|
|
|
|
|
|
while(sp && *sp && *sp == ' ') {
|
|
|
|
sp++;
|
|
|
|
}
|
|
|
|
|
|
|
|
s = strdup(sp);
|
|
|
|
|
|
|
|
p = s + (strlen(s) - 1);
|
|
|
|
|
|
|
|
while(*p == ' ') {
|
|
|
|
*p-- = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
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';
|
|
|
|
while(*e == ' ') {
|
|
|
|
*e-- = '\0';
|
|
|
|
}
|
|
|
|
e = args;
|
|
|
|
br = 1;
|
|
|
|
while(e && *e) {
|
|
|
|
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;
|
|
|
|
|
|
|
|
for (p = str; p && *p; p++) {
|
|
|
|
if (!(*p == '.' || (*p > 47 && *p < 58))) {
|
|
|
|
r = SWITCH_FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2007-10-12 17:12:31 +00:00
|
|
|
|
2007-10-16 14:24:02 +00:00
|
|
|
SWITCH_DECLARE(const char *) switch_stristr(const char *str, const char *instr)
|
|
|
|
{
|
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 */
|
|
|
|
for ( ; ((*start) && (toupper(*start) != toupper(*instr))); start++);
|
|
|
|
|
|
|
|
if (!*start)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
pptr = instr;
|
|
|
|
sptr = start;
|
|
|
|
|
|
|
|
while (toupper(*sptr) == toupper(*pptr)) {
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2007-10-29 06:16:41 +00:00
|
|
|
|
2007-01-19 19:11:44 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_find_local_ip(char *buf, int len, int family)
|
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
char *base;
|
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
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_copy_string(buf, "127.0.0.1", len);
|
2007-01-19 19:11:44 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch (family) {
|
|
|
|
case AF_INET:
|
|
|
|
base = "82.45.148.209";
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
base = "52.2d.94.d1";
|
|
|
|
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
|
|
|
|
2006-11-30 21:28:32 +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,
|
2007-03-29 22:31:56 +00:00
|
|
|
SIO_ROUTING_INTERFACE_QUERY,
|
2007-03-30 00:13:31 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
closesocket(tmp_socket);
|
|
|
|
freeaddrinfo(address_info);
|
|
|
|
|
|
|
|
if (!getnameinfo((const struct sockaddr *) &l_address, l_address_len, buf, len, NULL, 0, NI_NUMERICHOST)) {
|
|
|
|
|
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
2007-01-19 19:11:44 +00:00
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_copy_string(buf, get_addr(abuf, sizeof(abuf), &iface_out.sin_addr), len);
|
|
|
|
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;
|
|
|
|
inet_pton(AF_INET6, buf, &remote.sin6_addr);
|
|
|
|
remote.sin6_port = htons(4242);
|
|
|
|
|
|
|
|
memset(&iface_out, 0, sizeof(iface_out));
|
|
|
|
tmp_socket = socket(AF_INET6, 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.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;
|
|
|
|
}
|
|
|
|
|
2007-01-19 19:11:44 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
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-01-03 00:21:17 +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*)";
|
|
|
|
|
|
|
|
switch_time_exp_lt(&tm, switch_time_now());
|
|
|
|
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";
|
|
|
|
|
2007-02-08 20:16:08 +00:00
|
|
|
SWITCH_DECLARE(char *) get_addr(char *buf, switch_size_t len, struct in_addr *in)
|
2006-11-15 03:17:28 +00:00
|
|
|
{
|
|
|
|
uint8_t x, *i;
|
|
|
|
char *p = buf;
|
|
|
|
|
|
|
|
|
2007-01-19 19:11:44 +00:00
|
|
|
i = (uint8_t *) in;
|
2006-11-15 03:17:28 +00:00
|
|
|
|
|
|
|
memset(buf, 0, len);
|
2007-03-29 22:31:56 +00:00
|
|
|
for (x = 0; x < 4; x++) {
|
2006-11-15 03:17:28 +00:00
|
|
|
sprintf(p, "%u%s", i[x], x == 3 ? "" : ".");
|
|
|
|
p = buf + strlen(buf);
|
|
|
|
}
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
key = (char) toupper(key);
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-06 17:18:56 +00:00
|
|
|
SWITCH_DECLARE(unsigned int) switch_separate_string(char *buf, char delim, char **array, int arraylen)
|
|
|
|
{
|
|
|
|
int argc;
|
2006-08-17 00:53:09 +00:00
|
|
|
char *ptr;
|
|
|
|
int quot = 0;
|
2007-11-08 23:46:26 +00:00
|
|
|
char qc = '\'';
|
2006-10-31 21:38:06 +00:00
|
|
|
int x;
|
2005-12-06 17:18:56 +00:00
|
|
|
|
|
|
|
if (!buf || !array || !arraylen) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(array, 0, arraylen * sizeof(*array));
|
|
|
|
|
2006-08-17 00:53:09 +00:00
|
|
|
ptr = buf;
|
2005-12-06 17:18:56 +00:00
|
|
|
|
2006-08-17 00:53:09 +00:00
|
|
|
for (argc = 0; *ptr && (argc < arraylen - 1); argc++) {
|
|
|
|
array[argc] = ptr;
|
|
|
|
for (; *ptr; ptr++) {
|
|
|
|
if (*ptr == qc) {
|
|
|
|
if (quot) {
|
|
|
|
quot--;
|
|
|
|
} else {
|
|
|
|
quot++;
|
|
|
|
}
|
|
|
|
} else if ((*ptr == delim) && !quot) {
|
|
|
|
*ptr++ = '\0';
|
2005-12-06 17:18:56 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-08-17 00:53:09 +00:00
|
|
|
if (*ptr) {
|
|
|
|
array[argc++] = ptr;
|
2005-12-06 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
2007-11-08 23:46:26 +00:00
|
|
|
/* strip quotes and leading / trailing spaces */
|
2007-03-29 22:31:56 +00:00
|
|
|
for (x = 0; x < argc; x++) {
|
2007-11-08 23:46:26 +00:00
|
|
|
char *p;
|
|
|
|
|
|
|
|
while(*(array[x]) == ' ') {
|
2006-10-31 21:38:06 +00:00
|
|
|
(array[x])++;
|
2007-11-08 23:46:26 +00:00
|
|
|
}
|
|
|
|
p = array[x];
|
|
|
|
while((p = strchr(array[x], qc))) {
|
|
|
|
memmove(p, p+1, strlen(p));
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
p = array[x] + (strlen(array[x]) - 1);
|
|
|
|
while(*p == ' ') {
|
|
|
|
*p-- = '\0';
|
2006-10-31 21:38:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-12-06 17:18:56 +00:00
|
|
|
return argc;
|
|
|
|
}
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-11-19 20:07:43 +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;
|
|
|
|
char *dest;
|
2007-03-29 22:31:56 +00:00
|
|
|
|
|
|
|
dest = (char *) malloc(sizeof(char));
|
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-03-29 22:31:56 +00:00
|
|
|
dest = (char *) realloc(dest, sizeof(char) * (dest_len + 1));
|
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-03-29 22:31:56 +00:00
|
|
|
dest = (char *) realloc(dest, sizeof(char) * (dest_len + 1));
|
2006-04-14 16:45:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dest[dest_len] = 0;
|
|
|
|
return dest;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-03-30 00:13:31 +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
|
|
|
|
2006-05-10 15:47:54 +00:00
|
|
|
|
2007-11-01 11:28:26 +00:00
|
|
|
SWITCH_DECLARE(size_t) 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;
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
memset(buf, 0, len);
|
2007-02-09 22:56:42 +00:00
|
|
|
|
|
|
|
if (!url) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
for (p = url; *p; p++) {
|
|
|
|
if (*p < ' ' || *p > '~' || strchr(urlunsafe, *p)) {
|
|
|
|
if ((x + 3) > len) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
buf[x++] = '%';
|
|
|
|
buf[x++] = hex[*p >> 4];
|
|
|
|
buf[x++] = hex[*p & 0x0f];
|
|
|
|
} else {
|
|
|
|
buf[x++] = *p;
|
|
|
|
}
|
|
|
|
if (x == len) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return x;
|
2006-05-10 15:47:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(char *) switch_url_decode(char *s)
|
|
|
|
{
|
|
|
|
char *o;
|
|
|
|
unsigned int tmp;
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-27 22:30:48 +00:00
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
2007-02-09 02:36:03 +00:00
|
|
|
* indent-tabs-mode:t
|
2006-11-27 22:30:48 +00:00
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 expandtab:
|
|
|
|
*/
|