/****************************************************************** * $Id: soap-nudp.c,v 1.10 2007/11/03 22:40:09 m0gg Exp $ * * CSOAP Project: A SOAP client/server library in C * Copyright (C) 2006 Heiko Ronsdorf * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. * * Email: hero@persua.de ******************************************************************/ #ifdef HAVE_CONFIG_H #include #endif #ifdef HAVE_SYS_TYPES_H #include #endif #ifdef HAVE_SYS_SOCKET_H #include #endif #ifdef HAVE_STDLIB_H #include #endif #ifdef HAVE_STDIO_H #include #endif #ifdef HAVE_STRING_H #include #endif #ifdef HAVE_ERRNO_H #include #endif #ifdef HAVE_UNISTD_H #include #endif #ifdef HAVE_PTHREAD_H #include #endif #ifdef HAVE_NETDB_H #include #endif #ifdef HAVE_NETINET_IN_H #include #endif #ifdef HAVE_ARPA_INET_H #include #endif #include #include #ifdef WIN32 #include #include int inet_pton(int af, const char *src, void *dst) { /* struct sockaddr can't accomodate struct sockaddr_in6. */ union { struct sockaddr_in6 sin6; struct sockaddr_in sin; } sa; size_t srcsize; switch(af) { case AF_INET: sa.sin.sin_family = AF_INET; srcsize = sizeof (sa.sin); break; case AF_INET6: sa.sin6.sin6_family = AF_INET6; srcsize = sizeof (sa.sin6); break; default: errno = 76; return -1; } if (WSAStringToAddress((LPSTR)src, af, NULL, (struct sockaddr *) &sa, &srcsize) != 0) { errno = WSAGetLastError(); return -1; } switch(af) { case AF_INET: memcpy(dst, &sa.sin.sin_addr, sizeof(sa.sin.sin_addr)); break; case AF_INET6: memcpy(dst, &sa.sin6.sin6_addr, sizeof(sa.sin6.sin6_addr)); break; } return 1; } #endif #include #include "soap-logging.h" #include "soap-fault.h" #include "soap-env.h" #include "soap-ctx.h" #include "soap-service.h" #include "soap-router.h" #include "soap-server.h" #include "soap-transport.h" #include "soap-addressing.h" #include "soap-nudp.h" static short _soap_nudp_port = NUDP_DEFAULT_PORT; static int _soap_nudp_socket; static pthread_t _soap_nudp_thread; static pthread_attr_t _soap_nudp_attr; static volatile int _soap_nudp_running = 0; static short _soap_nudp_server_set_port(void) { struct servent *entry; if (!(entry = getservbyname("soap", "udp"))) { log_warn("getservbyname(\"soap\", \"udp\") returned NULL, please edit services database"); _soap_nudp_port = NUDP_DEFAULT_PORT; } else { _soap_nudp_port = ntohs(entry->s_port); } return _soap_nudp_port; } static void _soap_nudp_server_parse_arguments(int argc, char **argv) { int i; for (i=1; ienv))) { log_error("soap_addressing_get_to_address returned NULL"); return herror_new("soap_nudp_client_invoke", 0, "Destination address is missing"); } memset(&addr, '\0', sizeof(struct sockaddr_in)); addr.sin_family = AF_INET; if (to->port == 0) addr.sin_port = htons(NUDP_DEFAULT_PORT); else addr.sin_port = htons(to->port); if (inet_pton(AF_INET, "127.0.0.1", &addr.sin_addr) != 1) { log_error("inet_pton failed (%s)", strerror(errno)); return herror_new("soap_nudp_client_invoke", 0, "Cannot resolve destination address"); } if ((sd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) { log_error("socket failed (%s)", strerror(errno)); return herror_new("soap_nudp_client_invoke", 0, "Cannot create socket"); } _soap_nudp_send_document(sd, request->env->root->doc, (struct sockaddr *)&addr, sizeof(struct sockaddr_in)); saddr_len = sizeof(struct sockaddr); if ((status = _soap_nudp_receive_document(sd, &doc, &saddr, &saddr_len)) != H_OK) { log_error("_soap_nudp_receive_document failed (%s)", herror_message(status)); return status; } *response = soap_ctx_new(NULL); if ((status = soap_env_new_from_doc(doc, &(*response)->env)) != H_OK) { log_error("soap_env_new_from_doc failed (%s)", herror_message(status)); return status; } /** @todo: set saddr in SOAP:Header */ return H_OK; } herror_t soap_nudp_server_init_args(int argc, char **argv) { struct sockaddr_in addr; _soap_nudp_server_set_port(); _soap_nudp_server_parse_arguments(argc, argv); if ((_soap_nudp_socket = socket(AF_INET, SOCK_DGRAM, 0)) < 0) { log_error("socket failed (%s)", strerror(errno)); return herror_new("soap_nudp_server_init_args", 0, "Cannot create socket (%s)", strerror(errno)); } memset(&addr, 0, sizeof(struct sockaddr_in)); addr.sin_family = AF_INET; addr.sin_addr.s_addr = htonl(INADDR_ANY); addr.sin_port = htons(_soap_nudp_port); if (bind(_soap_nudp_socket, (struct sockaddr *)&addr, sizeof(struct sockaddr_in)) < 0) { log_error("bind failed (%s)", strerror(errno)); return herror_new("soap_nudp_server_init_args", 0, "Cannot bind socket (%s)", strerror(errno)); } return H_OK; } herror_t soap_nudp_register(const void *data) { return H_OK; } void * soap_nudp_server_run(void *unused) { xmlDocPtr doc; struct sockaddr addr; socklen_t addr_len; struct SoapCtx *req; struct SoapCtx *res; xmlURI *to; herror_t status; while (_soap_nudp_running) { /* XXX: select with timeout */ addr_len = sizeof(struct sockaddr); if (_soap_nudp_receive_document(_soap_nudp_socket, &doc, &addr, &addr_len) != H_OK) { log_error("_soap_nudp_receive_document failed (%s)", herror_message(status)); herror_release(status); continue; } /* log_error(__FUNCTION__); xmlDocFormatDump(stdout, doc, 1); */ req = soap_ctx_new(NULL); soap_env_new_from_doc(doc, &(req->env)); /* only local part is interesting */ to = soap_addressing_get_to_address(req->env); soap_addressing_set_to_address_string(req->env, to->path); // xmlFreeDoc(doc); soap_transport_process(req, &res); xmlDocFormatDump(stderr, res->env->root->doc, 1); _soap_nudp_send_document(_soap_nudp_socket, res->env->root->doc, &addr, addr_len); soap_ctx_free(res); soap_ctx_free(req); } close(_soap_nudp_socket); return NULL; } herror_t soap_nudp_server_run_threaded(void) { int err; _soap_nudp_running = 1; if ((err = pthread_create(&_soap_nudp_thread, &_soap_nudp_attr, soap_nudp_server_run, NULL)) < 0) { log_error("pthread_create failed (%s)", strerror(err)); return herror_new("soap_nudp_server_run_threaded", 0, "pthread_create failed (%s)", strerror(err)); } return H_OK; } void soap_nudp_server_destroy(void) { _soap_nudp_running = 0; /* XXX: sleep for timeout, wait for select */ return; } herror_t soap_nudp_client_init_args(int argc, char **argv) { soap_transport_add("soap.udp", NULL, _soap_nudp_client_invoke); return H_OK; } void soap_nudp_client_destroy(void) { return; }