28#include <sys/socket.h>
29#include <netinet/in.h>
30#include <netinet/tcp.h>
40#include "../dpip/dpip.h"
42#define DPI_EXT (".dpi" EXEEXT)
66 if (dpi_attr->
id != NULL) {
70 if (dpi_attr->
path != NULL) {
72 dpi_attr->
path = NULL;
90 *dpi_attr_list_ptr = NULL;
120 struct sigaction act;
124 sigaddset(&block, SIGHUP);
125 sigaddset(&block, SIGINT);
126 sigaddset(&block, SIGQUIT);
127 sigaddset(&block, SIGTERM);
133 if (sigaction(SIGHUP, &act, NULL) ||
134 sigaction(SIGINT, &act, NULL) ||
135 sigaction(SIGQUIT, &act, NULL) ||
136 sigaction(SIGTERM, &act, NULL)) {
137 ERRMSG(
"est_dpi_terminator",
"sigaction", errno);
142 ERRMSG(
"est_dpi_terminator",
"atexit", 0);
143 MSG_ERR(
"Hey! atexit failed, how did that happen?\n");
148static int ends_with(
const char *str,
const char *suffix)
152 size_t lenstr = strlen(str);
153 size_t lensuffix = strlen(suffix);
154 if (lensuffix > lenstr)
156 return strncmp(str + lenstr - lensuffix, suffix, lensuffix) == 0;
169 MSG_ERR(
"get_file_type: Unknown file type for %s\n", file_name);
184 char *rcline = NULL, *value = NULL, *p;
186 if ((In = fopen(dpidrc,
"r")) == NULL) {
187 ERRMSG(
"dpi_dir",
"fopen", errno);
192 while ((rcline =
dGetline(In)) != NULL) {
193 if (strncmp(rcline,
"dpi_dir", 7) == 0)
200 ERRMSG(
"dpi_dir",
"Failed to find a dpi_dir entry in dpidrc", 0);
201 MSG_ERR(
"Put your dillo plugins path in %s\n", dpidrc);
202 MSG_ERR(
"e.g. dpi_dir=/usr/local/lib/dillo/dpi\n");
203 MSG_ERR(
"with no leading spaces.\n");
206 len = (int) strlen(rcline);
207 if (len && rcline[len - 1] ==
'\n')
210 if ((p = strchr(rcline,
'='))) {
214 ERRMSG(
"dpi_dir",
"strchr", 0);
215 MSG_ERR(
" - '=' not found in %s\n", rcline);
236 char *service_dir = NULL;
237 struct stat statinfo;
241 struct dirent *dir_entry = NULL;
244 if (stat(service_dir, &statinfo) == -1) {
245 ERRMSG(
"get_dpi_attr",
"stat", errno);
246 MSG_ERR(
"file=%s\n", service_dir);
247 }
else if ((dir_stream = opendir(service_dir)) == NULL) {
248 ERRMSG(
"get_dpi_attr",
"opendir", errno);
253 while ( (dir_entry = readdir(dir_stream)) != NULL) {
254 if (dir_entry->d_name[0] ==
'.')
261 dStrconcat(service_dir,
"/", dir_entry->d_name, NULL);
265 if (strstr(dpi_attr->
path,
".filter") != NULL)
275 closedir(dir_stream);
278 MSG_ERR(
"get_dpi_attr: No dpi plug-in in %s/%s\n",
296 char *user_dpi_dir, *dpidrc, *user_service_dir, *dir = NULL;
304 if (access(dpidrc, F_OK) == -1) {
305 if (access(DPIDRC_SYS, F_OK) == -1) {
306 ERRMSG(
"register_service",
"Error ", 0);
307 MSG_ERR(
"\n - There is no %s or %s file\n", dpidrc,
310 dFree(user_service_dir);
319 if (access(user_service_dir, F_OK) == 0) {
324 if (access(dir, F_OK) == 0) {
328 ERRMSG(
"register_service",
"get_dpi_attr failed", 0);
331 ERRMSG(
"register_service",
"dpi_dir: Error getting dpi dir.", 0);
335 dFree(user_service_dir);
350 char *user_dpidir = NULL, *sys_dpidir = NULL, *dpidrc = NULL;
351 struct dirent *user_dirent, *sys_dirent;
354 size_t dp_sz =
sizeof(
struct dp);
356 if (*attlist != NULL) {
357 ERRMSG(
"register_all",
"attlist parameter should be NULL", 0);
362 if (access(user_dpidir, F_OK) == -1) {
368 if (access(dpidrc, F_OK) == -1) {
371 if (access(dpidrc, F_OK) == -1) {
376 if (!dpidrc || (sys_dpidir =
get_dpi_dir(dpidrc)) == NULL)
380 if (!user_dpidir && !sys_dpidir) {
381 ERRMSG(
"register_all",
"Fatal error ", 0);
382 MSG_ERR(
"\n - Can't find the directory for dpis.\n");
389 DIR *user_dir_stream = opendir(user_dpidir);
392 if (user_dir_stream == NULL && errno != ENOENT) {
393 MSG_ERR(
"cannot open user dpi directory '%s': %s\n",
394 user_dpidir, strerror(errno));
396 while ((user_dirent = readdir(user_dir_stream)) != NULL) {
397 if (user_dirent->d_name[0] ==
'.')
399 *attlist = (
struct dp *)
dRealloc(*attlist, (snum + 1) * dp_sz);
400 st=
get_dpi_attr(user_dpidir, user_dirent->d_name, &(*attlist)[snum]);
404 closedir(user_dir_stream);
408 DIR *sys_dir_stream = opendir(sys_dpidir);
410 if (sys_dir_stream == NULL) {
411 MSG_ERR(
"cannot open system dpi directory '%s': %s\n",
412 sys_dpidir, strerror(errno));
415 while ((sys_dirent = readdir(sys_dir_stream)) != NULL) {
416 if (sys_dirent->d_name[0] ==
'.')
418 *attlist = (
struct dp *)
dRealloc(*attlist, (snum + 1) * dp_sz);
419 st=
get_dpi_attr(sys_dpidir, sys_dirent->d_name, &(*attlist)[snum]);
423 closedir(sys_dir_stream);
458 char *user_dpidir = NULL, *sys_dpidir = NULL, *dpidrc = NULL;
461 if (access(user_dpidir, F_OK) == -1) {
467 if (access(dpidrc, F_OK) == -1) {
470 if (access(dpidrc, F_OK) == -1) {
475 if (!dpidrc || (sys_dpidir =
get_dpi_dir(dpidrc)) == NULL)
478 if (!user_dpidir && !sys_dpidir) {
479 ERRMSG(
"fill_services_list",
"Fatal error ", 0);
480 MSG_ERR(
"\n - Can't find the directory for dpis.\n");
484 if ((dpidrc_stream = fopen(dpidrc,
"r")) == NULL) {
485 ERRMSG(
"fill_services_list",
"popen failed", errno);
493 ERRMSG(
"fill_services_list",
"services_list parameter is not NULL", 0);
494 fclose(dpidrc_stream);
500 for (;(line =
dGetline(dpidrc_stream)) != NULL;
dFree(line)) {
503 MSG_ERR(
"dpid: Syntax error in %s: service=\"%s\" path=\"%s\"\n",
506 }
else if (st != 0) {
513 if (strcmp(
service,
"dpi_dir") == 0)
524 if ((p = strstr(attlist[i].
path,
path)) && *(p - 1) ==
'/' &&
525 strlen(p) == strlen(
path))
531 fclose(dpidrc_stream);
550 if ((ret = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
551 ERRMSG(
"make_socket_fd",
"socket", errno);
554 setsockopt(ret, IPPROTO_TCP, TCP_NODELAY, &one,
sizeof(one));
572 struct sockaddr_in sin;
573 int ok = 0, last_port = base_port + 50;
579 fcntl(sock_fd, F_SETFD, FD_CLOEXEC | fcntl(sock_fd, F_GETFD));
582 memset(&sin, 0,
sizeof(sin));
583 sin.sin_family = AF_INET;
584 sin.sin_addr.s_addr = inet_addr(
"127.0.0.1");
587 for (port = base_port; port <= last_port; ++port) {
588 sin.sin_port = htons(port);
589 if ((bind(sock_fd, (
struct sockaddr *)&sin,
sizeof(sin))) == -1) {
590 if (errno == EADDRINUSE || errno == EADDRNOTAVAIL)
592 ERRMSG(
"bind_socket_fd",
"bind", errno);
593 }
else if (listen(sock_fd,
QUEUE) == -1) {
594 ERRMSG(
"bind_socket_fd",
"listen", errno);
601 if (port > last_port) {
602 MSG_ERR(
"Hey! Can't find an available port from %d to %d\n",
603 base_port, last_port);
606 return ok ? sock_fd : -1;
616 char *fname, port_str[32];
619 fd = open(fname, O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR);
624 snprintf(port_str, 16,
"%d %s\n", srs_port,
SharedKey);
640 int srs_port, ret = -1;
664 int s_fd, port, ret = -1;
668 dpi_attr->
port = port;
691 for (i = 0; i <
numdpis; i++) {
718 for (i = 0; i <
numdpis; i++) {
727 while (waitpid(-1, &status, WNOHANG) > 0)
734 struct sigaction sigact;
737 (void) sigemptyset(&set);
739 sigact.sa_mask = set;
740 sigact.sa_flags = SA_NOCLDSTOP;
741 if (sigaction(SIGCHLD, &sigact, NULL) == -1) {
742 ERRMSG(
"est_dpi_sigchld",
"sigaction", errno);
748int ckd_connect (
int sock_fd,
struct sockaddr *addr, socklen_t len)
753 ret = connect(sock_fd, addr, len);
754 }
while (ret == -1 && errno == EINTR);
756 ERRMSG(
"dpid.c",
"connect", errno);
765 char *bye_cmd, *auth_cmd;
767 struct sockaddr_in sin;
772 memset(&sin, 0,
sizeof(sin));
773 sin.sin_family = AF_INET;
774 sin.sin_addr.s_addr = inet_addr(
"127.0.0.1");
776 for (i = 0; i <
numdpis; i++) {
782 ERRMSG(
"stop_active_dpis",
"socket", errno);
787 if (
ckd_connect(sock_fd, (
struct sockaddr *)&sin,
sizeof(sin)) == -1) {
788 ERRMSG(
"stop_active_dpis",
"connect", errno);
790 }
else if (
CKD_WRITE(sock_fd, auth_cmd) == -1) {
791 ERRMSG(
"stop_active_dpis",
"write", errno);
792 }
else if (
CKD_WRITE(sock_fd, bye_cmd) == -1) {
793 ERRMSG(
"stop_active_dpis",
"write", errno);
813 for (i = 0; i <
numdpis; i++) {
853 ERRMSG(
"get_message",
"failed to parse msg", 0);
855 "DpiError",
"Failed to parse request");
868 int A_len, B_len, len;
870 A_len = strlen(A->
name);
872 len =
MAX (A_len, B_len);
874 if (A->
name[A_len - 1] ==
'*')
886 char *dpi_id, *d_cmd, port_str[16];
893 if (serv == NULL || (i = serv->
dp_index) == -1)
char * dGetline(FILE *stream)
Get a line from a FILE stream.
char * dStrconcat(const char *s1,...)
Concatenate a NULL-terminated list of strings.
char * dStrdup(const char *s)
Dlist * dList_new(int size)
Create a new empty list.
int dStrnAsciiCasecmp(const char *s1, const char *s2, size_t n)
int dList_length(Dlist *lp)
For completing the ADT.
void * dList_nth_data(Dlist *lp, int n0)
Return the nth data item, NULL when not found or 'n0' is out of range.
int dClose(int fd)
Close a FD handling EINTR.
void dList_sort(Dlist *lp, dCompareFunc func)
Sort the list using a custom function.
void dList_append(Dlist *lp, void *data)
Append a data item to the list.
void dList_free(Dlist *lp)
Free a list (not its elements)
void * dList_find_custom(Dlist *lp, const void *data, dCompareFunc func)
Search a data item using a custom function.
int dParser_parse_rc_line(char **line, char **name, char **value)
Take a dillo rc line and return 'name' and 'value' pointers to it.
void * dRealloc(void *mem, size_t size)
char * dGethomedir(void)
Return the home directory in a static string (don't free)
#define dReturn_if_fail(expr)
int(* dCompareFunc)(const void *a, const void *b)
#define dNew(type, count)
void send_sockport(int sock_fd, char *dpi_tag, struct dp *dpi_attr_list)
void free_services_list(Dlist *s_list)
static void terminator(int sig)
void free_dpi_attr(struct dp *dpi_attr)
int init_all_dpi_sockets(struct dp *dpi_attr_list)
int bind_socket_fd(int base_port, int *p_port)
volatile sig_atomic_t caught_sigchld
int register_all_cmd(void)
char * get_message(int sock_fd, char *dpi_tag)
static int ends_with(const char *str, const char *suffix)
int register_all(struct dp **attlist)
int init_dpi_socket(struct dp *dpi_attr)
int init_ids_srs_socket(void)
int ckd_connect(int sock_fd, struct sockaddr *addr, socklen_t len)
void est_dpi_sigchld(void)
void handle_sigchld(void)
int get_dpi_attr(char *dpi_dir, char *service, struct dp *dpi_attr)
int service_match(const struct service *A, const char *B)
void stop_active_dpis(struct dp *dpi_attr_list, int numdpis)
static int make_socket_fd(void)
int register_service(struct dp *dpi_attr, char *service)
void est_dpi_terminator(void)
static int services_alpha_comp(const struct service *s1, const struct service *s2)
enum file_type get_file_type(char *file_name)
int fill_services_list(struct dp *attlist, int numdpis, Dlist **services_list)
char * get_dpi_dir(char *dpidrc)
void free_plugin_list(struct dp **dpi_attr_list_ptr, int numdpis)
void dpi_sigchld(int sig)
void ignore_dpi_sockets(struct dp *dpi_attr_list, int numdpis)
int save_comm_keys(int srs_port)
struct dp * dpi_attr_list
#define CKD_WRITE(fd, msg)
#define ERRMSG(CALLER, CALLED, ERR)
#define dotDILLO_DPID_COMM_KEYS
char * a_Dpip_build_cmd(const char *format,...)
Printf like function for building dpip commands.
char * a_Dpip_get_attr(const char *tag, const char *attrname)
Task: given a tag and an attribute name, return its value.
char * a_Misc_mksecret(int nchar)
Return a new, random hexadecimal string of 'nchar' characters.