mirror of https://github.com/sipwise/kamailio.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
176 lines
3.8 KiB
176 lines
3.8 KiB
/*
|
|
* XMPP Module
|
|
* This file is part of Kamailio, a free SIP server.
|
|
*
|
|
* Copyright (C) 2006 Voice Sistem S.R.L.
|
|
*
|
|
* Kamailio is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version
|
|
*
|
|
* Kamailio is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
* Author: Andreea Spirea
|
|
*
|
|
*/
|
|
/*! \file
|
|
* \brief Kamailio XMPP module
|
|
* \ingroup xmpp
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <stdarg.h>
|
|
#include <errno.h>
|
|
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#include <netdb.h>
|
|
|
|
#include "../../sr_module.h"
|
|
|
|
int net_listen(char *server, int port)
|
|
{
|
|
int fd;
|
|
struct sockaddr_in sin;
|
|
int on = 1;
|
|
|
|
memset(&sin, 0, sizeof(struct sockaddr_in));
|
|
sin.sin_family = AF_INET;
|
|
sin.sin_port = htons(port);
|
|
|
|
if (!inet_aton(server, &sin.sin_addr)) {
|
|
struct hostent *host;
|
|
|
|
LM_DBG("resolving %s...\n", server);
|
|
|
|
if (!(host = gethostbyname(server))) {
|
|
LM_ERR("resolving %s failed (%s).\n", server,
|
|
hstrerror(h_errno));
|
|
return -1;
|
|
}
|
|
memcpy(&sin.sin_addr, host->h_addr_list[0], host->h_length);
|
|
}
|
|
|
|
if ((fd = socket(PF_INET, SOCK_STREAM, 0)) < 0) {
|
|
LM_ERR("socket() failed: %s\n", strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
LM_DBG("listening on %s:%d\n", inet_ntoa(sin.sin_addr), port);
|
|
|
|
if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < 0) {
|
|
LM_WARN("setsockopt(SO_REUSEADDR) failed: %s\n",strerror(errno));
|
|
}
|
|
|
|
if (bind(fd, (struct sockaddr *) &sin, sizeof(struct sockaddr_in)) < 0) {
|
|
LM_ERR("bind() failed: %s\n", strerror(errno));
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
|
|
if (listen(fd, 1) < 0) {
|
|
LM_ERR("listen() failed: %s\n", strerror(errno));
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
|
|
return fd;
|
|
}
|
|
|
|
int net_connect(char *server, int port)
|
|
{
|
|
int fd;
|
|
struct sockaddr_in sin;
|
|
|
|
memset(&sin, 0, sizeof(struct sockaddr_in));
|
|
sin.sin_family = AF_INET;
|
|
sin.sin_port = htons(port);
|
|
|
|
if (!inet_aton(server, &sin.sin_addr)) {
|
|
struct hostent *host;
|
|
|
|
LM_DBG("resolving %s...\n", server);
|
|
|
|
if (!(host = gethostbyname(server))) {
|
|
LM_ERR("resolving %s failed (%s).\n", server,
|
|
hstrerror(h_errno));
|
|
return -1;
|
|
}
|
|
memcpy(&sin.sin_addr, host->h_addr_list[0], host->h_length);
|
|
}
|
|
|
|
if ((fd = socket(PF_INET, SOCK_STREAM, 0)) < 0) {
|
|
LM_ERR("socket() failed: %s\n", strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
LM_DBG("connecting to %s:%d...\n", inet_ntoa(sin.sin_addr), port);
|
|
|
|
if (connect(fd, (struct sockaddr *) &sin, sizeof(struct sockaddr_in)) < 0) {
|
|
LM_ERR("connect() failed: %s\n", strerror(errno));
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
|
|
LM_DBG("connected to %s:%d...\n", inet_ntoa(sin.sin_addr), port);
|
|
return fd;
|
|
}
|
|
|
|
int net_send(int fd, const char *buf, int len)
|
|
{
|
|
const char *p = buf;
|
|
int res;
|
|
|
|
do {
|
|
res = send(fd, p, len, 0);
|
|
if (res <= 0)
|
|
return res;
|
|
len -= res;
|
|
p += res;
|
|
} while (len);
|
|
|
|
return (p - buf);
|
|
}
|
|
|
|
int net_printf(int fd, char *format, ...)
|
|
{
|
|
va_list args;
|
|
char buf[4096];
|
|
|
|
va_start(args, format);
|
|
vsnprintf(buf, sizeof(buf) - 1, format, args);
|
|
va_end(args);
|
|
|
|
LM_DBG("net_printf: [%s]\n", buf);
|
|
|
|
return net_send(fd, buf, strlen(buf));
|
|
}
|
|
|
|
char *net_read_static(int fd)
|
|
{
|
|
static char buf[4096];
|
|
int res;
|
|
|
|
res = recv(fd, buf, sizeof(buf) - 1, 0);
|
|
if (res < 0) {
|
|
LM_ERR("recv() failed: %s\n", strerror(errno));
|
|
return NULL;
|
|
}
|
|
if (!res)
|
|
return NULL;
|
|
buf[res] = 0;
|
|
return buf;
|
|
}
|