/* * relay-client.c - client functions for relay plugin * * Copyright (C) 2003-2024 Sébastien Helleu * * This file is part of WeeChat, the extensible chat client. * * WeeChat 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 3 of the License, or * (at your option) any later version. * * WeeChat 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 WeeChat. If not, see . */ #include #include #include #include #include #include #include #include #include #include #include #include "../weechat-plugin.h" #include "relay.h" #include "relay-client.h" #include "relay-auth.h" #include "relay-config.h" #include "relay-buffer.h" #include "relay-http.h" #include "relay-network.h" #include "relay-raw.h" #include "relay-server.h" #include "relay-websocket.h" #ifdef HAVE_CJSON #include "api/relay-api.h" #endif #include "irc/relay-irc.h" #include "weechat/relay-weechat.h" char *relay_client_data_type_string[] = /* strings for data types */ { "text", "binary", "http" }; struct t_relay_client *relay_clients = NULL; struct t_relay_client *last_relay_client = NULL; int relay_client_count = 0; /* number of clients */ /* * Checks if a client pointer is valid. * * Returns: * 1: client exists * 0: client does not exist */ int relay_client_valid (struct t_relay_client *client) { struct t_relay_client *ptr_client; if (!client) return 0; for (ptr_client = relay_clients; ptr_client; ptr_client = ptr_client->next_client) { if (ptr_client == client) return 1; } /* client not found */ return 0; } /* * Searches for a client by number (first client is 0). * * Returns pointer to client found, NULL if not found. */ struct t_relay_client * relay_client_search_by_number (int number) { struct t_relay_client *ptr_client; int i; i = 0; for (ptr_client = relay_clients; ptr_client; ptr_client = ptr_client->next_client) { if (i == number) return ptr_client; i++; } /* client not found */ return NULL; } /* * Searches for a client by id. * * Returns pointer to client found, NULL if not found. */ struct t_relay_client * relay_client_search_by_id (int id) { struct t_relay_client *ptr_client; for (ptr_client = relay_clients; ptr_client; ptr_client = ptr_client->next_client) { if (ptr_client->id == id) return ptr_client; } /* client not found */ return NULL; } /* * Returns the number of active clients (connecting or connected) on a given * server port. */ int relay_client_count_active_by_port (int server_port) { struct t_relay_client *ptr_client; int count; count = 0; for (ptr_client = relay_clients; ptr_client; ptr_client = ptr_client->next_client) { if ((ptr_client->server_port == server_port) && !RELAY_STATUS_HAS_ENDED(ptr_client->status)) { count++; } } return count; } /* * Sends a signal with the status of client ("relay_client_xxx"). */ void relay_client_send_signal (struct t_relay_client *client) { char signal[128]; snprintf (signal, sizeof (signal), "relay_client_%s", relay_status_name[client->status]); weechat_hook_signal_send (signal, WEECHAT_HOOK_SIGNAL_POINTER, client); } /* * Sets description for a client. */ void relay_client_set_desc (struct t_relay_client *client) { char desc[512]; free (client->desc); snprintf (desc, sizeof (desc), "%d/%s%s%s%s/%s%s%s%s", client->id, (client->tls) ? "tls." : "", relay_protocol_string[client->protocol], (client->protocol_args) ? "." : "", (client->protocol_args) ? client->protocol_args : "", client->address, (client->real_ip) ? "(" : "", (client->real_ip) ? client->real_ip : "", (client->real_ip) ? ")" : ""); client->desc = strdup (desc); } /* * Timer callback for handshake with client (for TLS connection only). */ int relay_client_handshake_timer_cb (const void *pointer, void *data, int remaining_calls) { struct t_relay_client *client; int rc; /* make C compiler happy */ (void) data; client = (struct t_relay_client *)pointer; rc = gnutls_handshake (client->gnutls_sess); if (rc == GNUTLS_E_SUCCESS) { /* handshake OK, set status to "connected" */ weechat_unhook (client->hook_timer_handshake); client->hook_timer_handshake = NULL; client->gnutls_handshake_ok = 1; switch (client->protocol) { case RELAY_PROTOCOL_WEECHAT: relay_client_set_status ( client, relay_weechat_get_initial_status (client)); break; case RELAY_PROTOCOL_IRC: relay_client_set_status ( client, relay_irc_get_initial_status (client)); break; case RELAY_PROTOCOL_API: #ifdef HAVE_CJSON relay_client_set_status ( client, relay_api_get_initial_status (client)); #endif /* HAVE_CJSON */ break; case RELAY_NUM_PROTOCOLS: break; } return WEECHAT_RC_OK; } if (gnutls_error_is_fatal (rc)) { /* handshake error, disconnect client */ weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: TLS handshake failed for client %s%s%s: error %d %s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT, rc, gnutls_strerror (rc)); weechat_unhook (client->hook_timer_handshake); client->hook_timer_handshake = NULL; relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); return WEECHAT_RC_OK; } if (remaining_calls == 0) { /* handshake timeout, disconnect client */ weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: TLS handshake timeout for client %s%s%s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT); weechat_unhook (client->hook_timer_handshake); client->hook_timer_handshake = NULL; relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); return WEECHAT_RC_OK; } /* handshake in progress, we will try again on next call to timer */ return WEECHAT_RC_OK; } /* * Reads text data from a client (messages on a single line): splits data on * '\n' and keeps a partial message if data does not end with '\n'. */ void relay_client_recv_text_single_line (struct t_relay_client *client) { char *pos, *raw_msg, **lines, *tmp; int i, num_lines, length; if (!client->partial_message) return; pos = strrchr (client->partial_message, '\n'); if (!pos) return; /* print message in raw buffer */ raw_msg = weechat_strndup (client->partial_message, pos - client->partial_message + 1); if (raw_msg) { relay_raw_print_client (client, RELAY_MSG_STANDARD, RELAY_RAW_FLAG_RECV, raw_msg, strlen (raw_msg) + 1); free (raw_msg); } pos[0] = '\0'; lines = weechat_string_split (client->partial_message, "\n", NULL, WEECHAT_STRING_SPLIT_STRIP_LEFT | WEECHAT_STRING_SPLIT_STRIP_RIGHT | WEECHAT_STRING_SPLIT_COLLAPSE_SEPS, 0, &num_lines); if (lines) { for (i = 0; i < num_lines; i++) { /* remove final '\r' */ length = strlen (lines[i]); if ((length > 0) && (lines[i][length - 1] == '\r')) lines[i][length - 1] = '\0'; /* * interpret text from client, according to the relay * protocol used */ switch (client->protocol) { case RELAY_PROTOCOL_WEECHAT: relay_weechat_recv (client, lines[i]); break; case RELAY_PROTOCOL_IRC: relay_irc_recv (client, lines[i]); break; case RELAY_PROTOCOL_API: /* api is multi-line only (JSON) */ break; case RELAY_NUM_PROTOCOLS: break; } } weechat_string_free_split (lines); } if (pos[1]) { tmp = strdup (pos + 1); free (client->partial_message); client->partial_message = tmp; } else { free (client->partial_message); client->partial_message = NULL; } } /* * Reads text data from a client (multi-line messages). */ void relay_client_recv_text_multi_line (struct t_relay_client *client) { if (!client->partial_message) return; /* print message in raw buffer */ relay_raw_print_client (client, RELAY_MSG_STANDARD, RELAY_RAW_FLAG_RECV, client->partial_message, strlen (client->partial_message) + 1); /* * interpret text from client, according to the relay * protocol used */ switch (client->protocol) { case RELAY_PROTOCOL_WEECHAT: /* weechat is single line only */ break; case RELAY_PROTOCOL_IRC: /* irc is single line only */ break; case RELAY_PROTOCOL_API: #ifdef HAVE_CJSON relay_api_recv_json (client, client->partial_message); #endif /* HAVE_CJSON */ break; case RELAY_NUM_PROTOCOLS: break; } free (client->partial_message); client->partial_message = NULL; } /* * Reads text data from a client: splits data on '\n' and keeps a partial * message if data does not end with '\n'. */ void relay_client_recv_text (struct t_relay_client *client, const char *data) { char *new_partial; if (client->partial_message) { new_partial = realloc (client->partial_message, strlen (client->partial_message) + strlen (data) + 1); if (!new_partial) return; client->partial_message = new_partial; strcat (client->partial_message, data); } else client->partial_message = strdup (data); if (client->partial_message) { if (client->recv_data_type == RELAY_CLIENT_DATA_TEXT_LINE) relay_client_recv_text_single_line (client); else if (client->recv_data_type == RELAY_CLIENT_DATA_TEXT_MULTILINE) relay_client_recv_text_multi_line (client); } } /* * Reads text buffer from a client. */ void relay_client_recv_text_buffer (struct t_relay_client *client, const char *buffer, unsigned long long length_buffer) { unsigned long long index; unsigned char msg_type; index = 0; while (index < length_buffer) { msg_type = RELAY_MSG_STANDARD; /* * in case of websocket, we can receive PING from client: * trace this PING in raw buffer and answer with a PONG */ if (client->websocket == RELAY_CLIENT_WEBSOCKET_READY) { msg_type = (unsigned char)buffer[index]; if (msg_type == RELAY_MSG_PING) { /* print message in raw buffer */ relay_raw_print_client (client, RELAY_MSG_PING, RELAY_RAW_FLAG_RECV | RELAY_RAW_FLAG_BINARY, buffer + index + 1, strlen (buffer + index + 1)); /* answer with a PONG */ relay_client_send (client, RELAY_MSG_PONG, buffer + index + 1, strlen (buffer + index + 1), NULL); } else if (msg_type == RELAY_MSG_CLOSE) { /* print message in raw buffer */ relay_raw_print_client (client, RELAY_MSG_CLOSE, RELAY_RAW_FLAG_RECV | RELAY_RAW_FLAG_BINARY, buffer + index + 1, strlen (buffer + index + 1)); /* answer with a CLOSE */ relay_client_send (client, RELAY_MSG_CLOSE, buffer + index + 1, strlen (buffer + index + 1), NULL); /* close the connection */ relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); /* ignore any other message after the close */ return; } index++; } if (msg_type == RELAY_MSG_STANDARD) { if ((client->websocket == RELAY_CLIENT_WEBSOCKET_INITIALIZING) || (client->recv_data_type == RELAY_CLIENT_DATA_HTTP)) { relay_http_recv (client, buffer + index); } else { relay_client_recv_text (client, buffer + index); } } index += strlen (buffer + index) + 1; } } /* * Reads websocket frames. */ void relay_client_read_websocket_frames (struct t_relay_client *client, struct t_relay_websocket_frame *frames, int num_frames) { int i; if (!frames || (num_frames <= 0)) return; for (i = 0; i < num_frames; i++) { if (frames[i].payload_size == 0) { /* * When decoded length is 0, assume client sent a PONG frame. * * RFC 6455 Section 5.5.3: * * "A Pong frame MAY be sent unsolicited. This serves as a * unidirectional heartbeat. A response to an unsolicited * Pong frame is not expected." */ continue; } switch (frames[i].opcode) { case RELAY_MSG_PING: /* print message in raw buffer */ relay_raw_print_client (client, RELAY_MSG_PING, RELAY_RAW_FLAG_RECV | RELAY_RAW_FLAG_BINARY, frames[i].payload, frames[i].payload_size); /* answer with a PONG */ relay_client_send (client, RELAY_MSG_PONG, frames[i].payload, frames[i].payload_size, NULL); break; case RELAY_MSG_CLOSE: /* print message in raw buffer */ relay_raw_print_client (client, RELAY_MSG_CLOSE, RELAY_RAW_FLAG_RECV | RELAY_RAW_FLAG_BINARY, frames[i].payload, frames[i].payload_size); /* answer with a CLOSE */ relay_client_send (client, RELAY_MSG_CLOSE, frames[i].payload, frames[i].payload_size, NULL); /* close the connection */ relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); /* ignore any other message after the close */ return; default: if (frames[i].payload) { if ((client->websocket == RELAY_CLIENT_WEBSOCKET_INITIALIZING) || (client->recv_data_type == RELAY_CLIENT_DATA_HTTP)) { relay_http_recv (client, frames[i].payload); } else if ((client->recv_data_type == RELAY_CLIENT_DATA_TEXT_LINE) || (client->recv_data_type == RELAY_CLIENT_DATA_TEXT_MULTILINE)) { relay_client_recv_text (client, frames[i].payload); } } break; } } } /* * Reads a buffer of bytes from a client. */ void relay_client_recv_buffer (struct t_relay_client *client, const char *buffer, int buffer_size) { struct t_relay_websocket_frame *frames; char *buffer2; int rc, i, buffer2_size, num_frames; /* * if we are receiving the first message from client, check if it looks * like a websocket */ if (client->bytes_recv == 0) { if (relay_websocket_is_valid_http_get (client->protocol, buffer)) { /* * web socket is just initializing for now, it's not accepted * (we will check later with "http_headers" if web socket is * valid or not) */ client->websocket = RELAY_CLIENT_WEBSOCKET_INITIALIZING; } } client->bytes_recv += buffer_size; if (client->websocket == RELAY_CLIENT_WEBSOCKET_READY) { /* websocket used, decode message */ buffer2 = NULL; buffer2_size = 0; if (client->partial_ws_frame) { buffer2_size = buffer_size + client->partial_ws_frame_size; buffer2 = malloc (buffer2_size); if (!buffer2) { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: not enough memory for received message"), weechat_prefix ("error"), RELAY_PLUGIN_NAME); return; } memcpy (buffer2, client->partial_ws_frame, client->partial_ws_frame_size); memcpy (buffer2 + client->partial_ws_frame_size, buffer, buffer_size); } frames = NULL; num_frames = 0; rc = relay_websocket_decode_frame ( (buffer2) ? (unsigned char *)buffer2 : (unsigned char *)buffer, (buffer2) ? (unsigned long long)buffer2_size : (unsigned long long)buffer_size, 1, /* expect_masked_frame */ client->ws_deflate, &frames, &num_frames, &client->partial_ws_frame, &client->partial_ws_frame_size); free (buffer2); if (!rc) { /* fatal error when decoding frame: close connection */ if (frames) { for (i = 0; i < num_frames; i++) { free (frames[i].payload); } free (frames); } weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: error decoding websocket frame for client " "%s%s%s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT); relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); return; } relay_client_read_websocket_frames (client, frames, num_frames); for (i = 0; i < num_frames; i++) { free (frames[i].payload); } free (frames); } else { if ((client->websocket == RELAY_CLIENT_WEBSOCKET_INITIALIZING) || (client->recv_data_type == RELAY_CLIENT_DATA_HTTP)) { relay_http_recv (client, buffer); } else if ((client->recv_data_type == RELAY_CLIENT_DATA_TEXT_LINE) || (client->recv_data_type == RELAY_CLIENT_DATA_TEXT_MULTILINE)) { relay_client_recv_text (client, buffer); } } relay_buffer_refresh (NULL); } /* * Reads data from a client. */ int relay_client_recv_cb (const void *pointer, void *data, int fd) { struct t_relay_client *client; static char buffer[4096]; int num_read; /* make C compiler happy */ (void) data; (void) fd; client = (struct t_relay_client *)pointer; if (client->sock < 0) return WEECHAT_RC_OK; /* * data can be received only during authentication * or if connected (authentication was OK) */ if ((client->status != RELAY_STATUS_AUTHENTICATING) && (client->status != RELAY_STATUS_CONNECTED)) { return WEECHAT_RC_OK; } if (client->tls) num_read = gnutls_record_recv (client->gnutls_sess, buffer, sizeof (buffer) - 1); else num_read = recv (client->sock, buffer, sizeof (buffer) - 1, 0); if (num_read > 0) { buffer[num_read] = '\0'; relay_client_recv_buffer (client, buffer, num_read); } else { if (client->tls) { if ((num_read == 0) || ((num_read != GNUTLS_E_AGAIN) && (num_read != GNUTLS_E_INTERRUPTED))) { if (relay_config_display_clients[client->protocol]) { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: reading data on socket for client %s%s%s: " "error %d %s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT, num_read, (num_read == 0) ? _("(connection closed by peer)") : gnutls_strerror (num_read)); } relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); } } else { if ((num_read == 0) || ((errno != EAGAIN) && (errno != EWOULDBLOCK))) { if (relay_config_display_clients[client->protocol]) { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: reading data on socket for client %s%s%s: " "error %d %s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT, errno, (num_read == 0) ? _("(connection closed by peer)") : strerror (errno)); } relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); } } } return WEECHAT_RC_OK; } /* * Frees a message in out queue. */ void relay_client_outqueue_free (struct t_relay_client *client, struct t_relay_client_outqueue *outqueue) { struct t_relay_client_outqueue *new_outqueue; if (!client || !outqueue) return; /* remove outqueue message */ if (client->last_outqueue == outqueue) client->last_outqueue = outqueue->prev_outqueue; if (outqueue->prev_outqueue) { (outqueue->prev_outqueue)->next_outqueue = outqueue->next_outqueue; new_outqueue = client->outqueue; } else new_outqueue = outqueue->next_outqueue; if (outqueue->next_outqueue) (outqueue->next_outqueue)->prev_outqueue = outqueue->prev_outqueue; /* free data */ free (outqueue->data); free (outqueue->raw_message[0]); free (outqueue->raw_message[1]); free (outqueue); /* set new head */ client->outqueue = new_outqueue; } /* * Frees all messages in out queue. */ void relay_client_outqueue_free_all (struct t_relay_client *client) { while (client->outqueue) { relay_client_outqueue_free (client, client->outqueue); } } /* * Sends data to a client. * * Returns the number of bytes sent to the client. */ int relay_client_send_data (struct t_relay_client *client, const char *data, int data_size) { if (client->tls) { return (client->sock >= 0) ? gnutls_record_send (client->gnutls_sess, data, data_size) : data_size; } else { return (client->sock >= 0) ? send (client->sock, data, data_size, 0) : data_size; } } /* * Sends messages in outqueue for a client. */ void relay_client_send_outqueue (struct t_relay_client *client) { int i, num_sent; char *buf; while (client->outqueue) { num_sent = relay_client_send_data (client, client->outqueue->data, client->outqueue->data_size); if (num_sent >= 0) { for (i = 0; i < 2; i++) { if (client->outqueue->raw_message[i]) { /* * print raw message and remove it from outqueue * (so that it is displayed only one time, even if * message is sent in many chunks) */ relay_raw_print_client ( client, client->outqueue->raw_msg_type[i], client->outqueue->raw_flags[i], client->outqueue->raw_message[i], client->outqueue->raw_size[i]); client->outqueue->raw_flags[i] = 0; free (client->outqueue->raw_message[i]); client->outqueue->raw_message[i] = NULL; client->outqueue->raw_size[i] = 0; } } if (num_sent > 0) { client->bytes_sent += num_sent; relay_buffer_refresh (NULL); } if (num_sent == client->outqueue->data_size) { /* whole data sent, remove outqueue */ relay_client_outqueue_free (client, client->outqueue); } else { /* * some data was not sent, update outqueue and stop * sending data from outqueue */ if (num_sent > 0) { buf = malloc (client->outqueue->data_size - num_sent); if (buf) { memcpy (buf, client->outqueue->data + num_sent, client->outqueue->data_size - num_sent); free (client->outqueue->data); client->outqueue->data = buf; client->outqueue->data_size = client->outqueue->data_size - num_sent; } } break; } } else { if (client->tls) { if ((num_sent == GNUTLS_E_AGAIN) || (num_sent == GNUTLS_E_INTERRUPTED)) { /* we will retry later this client's queue */ break; } else { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: sending data to client %s%s%s: " "error %d %s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT, num_sent, gnutls_strerror (num_sent)); relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); } } else { if ((errno == EAGAIN) || (errno == EWOULDBLOCK)) { /* we will retry later this client's queue */ break; } else { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: sending data to client %s%s%s: " "error %d %s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT, errno, strerror (errno)); relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); } } } } if (!client->outqueue && client->hook_timer_send) { weechat_unhook (client->hook_timer_send); client->hook_timer_send = NULL; } } /* * Timer called for a client when outqueue is not empty. * * This timer is automatically removed when the outqueue becomes empty * (done in the function relay_client_send_outqueue). */ int relay_client_timer_send_cb (const void *pointer, void *data, int remaining_calls) { struct t_relay_client *ptr_client; /* make C compiler happy */ (void) data; (void) remaining_calls; ptr_client = (struct t_relay_client *)pointer; relay_client_send_outqueue (ptr_client); return WEECHAT_RC_OK; } /* * Adds a message in out queue. */ void relay_client_outqueue_add (struct t_relay_client *client, const char *data, int data_size, enum t_relay_msg_type raw_msg_type[2], int raw_flags[2], const char *raw_message[2], int raw_size[2]) { struct t_relay_client_outqueue *new_outqueue; int i; if (!client || !data || (data_size <= 0)) return; new_outqueue = malloc (sizeof (*new_outqueue)); if (!new_outqueue) return; new_outqueue->data = malloc (data_size); if (!new_outqueue->data) { free (new_outqueue); return; } memcpy (new_outqueue->data, data, data_size); new_outqueue->data_size = data_size; for (i = 0; i < 2; i++) { new_outqueue->raw_msg_type[i] = RELAY_MSG_STANDARD; new_outqueue->raw_flags[i] = 0; new_outqueue->raw_message[i] = NULL; new_outqueue->raw_size[i] = 0; if (raw_message && raw_message[i] && (raw_size[i] > 0)) { new_outqueue->raw_message[i] = malloc (raw_size[i]); if (new_outqueue->raw_message[i]) { new_outqueue->raw_msg_type[i] = raw_msg_type[i]; new_outqueue->raw_flags[i] = raw_flags[i]; memcpy (new_outqueue->raw_message[i], raw_message[i], raw_size[i]); new_outqueue->raw_size[i] = raw_size[i]; } } } new_outqueue->prev_outqueue = client->last_outqueue; new_outqueue->next_outqueue = NULL; if (client->last_outqueue) client->last_outqueue->next_outqueue = new_outqueue; else client->outqueue = new_outqueue; client->last_outqueue = new_outqueue; if (!client->hook_timer_send) { client->hook_timer_send = weechat_hook_timer ( 1, 0, 0, &relay_client_timer_send_cb, client, NULL); } } /* * Sends data to client (adds in out queue if it's impossible to send now). * * If "message_raw_buffer" is not NULL, it is used for display in raw buffer * and replaces display of data, which is default. * * Returns number of bytes sent to client, -1 if error. */ int relay_client_send (struct t_relay_client *client, enum t_relay_msg_type msg_type, const char *data, int data_size, const char *message_raw_buffer) { int num_sent, raw_size[2], raw_flags[2], opcode, i; enum t_relay_msg_type raw_msg_type[2]; char *websocket_frame; unsigned long long length_frame; const char *ptr_data, *raw_msg[2]; ptr_data = data; websocket_frame = NULL; if (client->fake_send_func) (void) (*client->fake_send_func) (client, ptr_data, data_size); /* set raw messages */ for (i = 0; i < 2; i++) { raw_msg_type[i] = msg_type; raw_flags[i] = RELAY_RAW_FLAG_SEND; raw_msg[i] = NULL; raw_size[i] = 0; } if (message_raw_buffer) { if (weechat_relay_plugin->debug >= 2) { raw_msg[0] = message_raw_buffer; raw_size[0] = strlen (message_raw_buffer) + 1; raw_msg[1] = data; raw_size[1] = data_size; raw_flags[1] |= RELAY_RAW_FLAG_BINARY; if ((client->websocket == RELAY_CLIENT_WEBSOCKET_INITIALIZING) || (client->send_data_type == RELAY_CLIENT_DATA_TEXT_LINE) || (client->send_data_type == RELAY_CLIENT_DATA_TEXT_MULTILINE)) { raw_size[1]--; } } else { raw_msg[0] = message_raw_buffer; raw_size[0] = strlen (message_raw_buffer) + 1; } } else { raw_msg[0] = data; raw_size[0] = data_size; if ((msg_type == RELAY_MSG_PING) || (msg_type == RELAY_MSG_PONG) || (msg_type == RELAY_MSG_CLOSE) || ((client->websocket != RELAY_CLIENT_WEBSOCKET_INITIALIZING) && (client->send_data_type == RELAY_CLIENT_DATA_BINARY))) { /* * set binary flag if we send binary to client * (except if websocket is initializing and then we are sending * HTTP data, as text) */ raw_flags[0] |= RELAY_RAW_FLAG_BINARY; } else { /* count the final '\0' in size */ raw_size[0]++; } } /* if websocket is initialized, encode data in a websocket frame */ if (client->websocket == RELAY_CLIENT_WEBSOCKET_READY) { switch (msg_type) { case RELAY_MSG_PING: opcode = WEBSOCKET_FRAME_OPCODE_PING; break; case RELAY_MSG_PONG: opcode = WEBSOCKET_FRAME_OPCODE_PONG; break; case RELAY_MSG_CLOSE: opcode = WEBSOCKET_FRAME_OPCODE_CLOSE; break; default: opcode = ((client->send_data_type == RELAY_CLIENT_DATA_TEXT_LINE) || (client->send_data_type == RELAY_CLIENT_DATA_TEXT_MULTILINE)) ? WEBSOCKET_FRAME_OPCODE_TEXT : WEBSOCKET_FRAME_OPCODE_BINARY; break; } websocket_frame = relay_websocket_encode_frame ( client->ws_deflate, opcode, 0, data, data_size, &length_frame); if (websocket_frame) { ptr_data = websocket_frame; data_size = length_frame; } } num_sent = -1; /* * if outqueue is not empty, add to outqueue * (because message must be sent *after* messages already in outqueue) */ if (client->outqueue) { relay_client_outqueue_add (client, ptr_data, data_size, raw_msg_type, raw_flags, raw_msg, raw_size); } else { num_sent = relay_client_send_data (client, ptr_data, data_size); if (num_sent >= 0) { for (i = 0; i < 2; i++) { if (raw_msg[i]) { relay_raw_print_client (client, raw_msg_type[i], raw_flags[i], raw_msg[i], raw_size[i]); } } if (num_sent > 0) { client->bytes_sent += num_sent; relay_buffer_refresh (NULL); } if (num_sent < data_size) { /* some data was not sent, add it to outqueue */ relay_client_outqueue_add (client, ptr_data + num_sent, data_size - num_sent, NULL, NULL, NULL, NULL); } } else { if (client->tls) { if ((num_sent == GNUTLS_E_AGAIN) || (num_sent == GNUTLS_E_INTERRUPTED)) { /* add message to queue (will be sent later) */ relay_client_outqueue_add (client, ptr_data, data_size, raw_msg_type, raw_flags, raw_msg, raw_size); } else { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: sending data to client %s%s%s: error %d %s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT, num_sent, gnutls_strerror (num_sent)); relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); } } else { if ((errno == EAGAIN) || (errno == EWOULDBLOCK)) { /* add message to queue (will be sent later) */ relay_client_outqueue_add (client, ptr_data, data_size, raw_msg_type, raw_flags, raw_msg, raw_size); } else { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: sending data to client %s%s%s: error %d %s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT, errno, strerror (errno)); relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); } } } } free (websocket_frame); return num_sent; } /* * Timer callback, called each second. */ int relay_client_timer_cb (const void *pointer, void *data, int remaining_calls) { struct t_relay_client *ptr_client, *ptr_next_client; int purge_delay, auth_timeout; time_t current_time; /* make C compiler happy */ (void) pointer; (void) data; (void) remaining_calls; purge_delay = weechat_config_integer (relay_config_network_clients_purge_delay); auth_timeout = weechat_config_integer (relay_config_network_auth_timeout); current_time = time (NULL); ptr_client = relay_clients; while (ptr_client) { ptr_next_client = ptr_client->next_client; if (RELAY_STATUS_HAS_ENDED(ptr_client->status)) { if ((purge_delay >= 0) && (current_time >= ptr_client->end_time + (purge_delay * 60))) { relay_client_free (ptr_client); relay_buffer_refresh (NULL); } } else { /* send messages in outqueue */ relay_client_send_outqueue (ptr_client); /* disconnect clients not authenticated */ if ((auth_timeout > 0) && (ptr_client->status == RELAY_STATUS_AUTHENTICATING)) { if (current_time - ptr_client->start_time > auth_timeout) { relay_client_set_status (ptr_client, RELAY_STATUS_AUTH_FAILED); } } } ptr_client = ptr_next_client; } return WEECHAT_RC_OK; } /* * Creates a new client. * * Returns pointer to new client, NULL if error. */ struct t_relay_client * relay_client_new (int sock, const char *address, struct t_relay_server *server) { struct t_relay_client *new_client; int plain_text_password; int bits; struct t_config_option *ptr_option; new_client = malloc (sizeof (*new_client)); if (new_client) { new_client->id = (relay_clients) ? relay_clients->id + 1 : 1; new_client->desc = NULL; new_client->sock = sock; new_client->server_port = server->port; new_client->tls = server->tls; new_client->gnutls_sess = NULL; new_client->fake_send_func = NULL; new_client->hook_timer_handshake = NULL; new_client->gnutls_handshake_ok = 0; new_client->websocket = RELAY_CLIENT_WEBSOCKET_NOT_USED; new_client->ws_deflate = relay_websocket_deflate_alloc (); new_client->http_req = relay_http_request_alloc (); new_client->address = strdup ((address && address[0]) ? address : "local"); new_client->real_ip = NULL; new_client->status = RELAY_STATUS_CONNECTING; new_client->protocol = server->protocol; new_client->protocol_string = (server->protocol_string) ? strdup (server->protocol_string) : NULL; new_client->protocol_args = (server->protocol_args) ? strdup (server->protocol_args) : NULL; new_client->nonce = relay_auth_generate_nonce ( weechat_config_integer (relay_config_network_nonce_size)); plain_text_password = weechat_string_match_list ( relay_auth_password_hash_algo_name[0], (const char **)relay_config_network_password_hash_algo_list, 1); new_client->password_hash_algo = (plain_text_password) ? 0 : -1; new_client->listen_start_time = server->start_time; new_client->start_time = time (NULL); new_client->end_time = 0; new_client->hook_fd = NULL; new_client->hook_timer_send = NULL; new_client->last_activity = new_client->start_time; new_client->bytes_recv = 0; new_client->bytes_sent = 0; switch (new_client->protocol) { case RELAY_PROTOCOL_WEECHAT: new_client->recv_data_type = RELAY_CLIENT_DATA_TEXT_LINE; new_client->send_data_type = RELAY_CLIENT_DATA_BINARY; break; case RELAY_PROTOCOL_IRC: new_client->recv_data_type = RELAY_CLIENT_DATA_TEXT_LINE; new_client->send_data_type = RELAY_CLIENT_DATA_TEXT_LINE; break; case RELAY_PROTOCOL_API: new_client->recv_data_type = RELAY_CLIENT_DATA_HTTP; new_client->send_data_type = RELAY_CLIENT_DATA_HTTP; break; default: new_client->recv_data_type = RELAY_CLIENT_DATA_TEXT_MULTILINE; new_client->send_data_type = RELAY_CLIENT_DATA_TEXT_MULTILINE; break; } new_client->partial_ws_frame = NULL; new_client->partial_ws_frame_size = 0; new_client->partial_message = NULL; relay_client_set_desc (new_client); if (new_client->tls) { if (!relay_network_init_tls_cert_key_ok) { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s%s: warning: no TLS certificate/key found (option " "relay.network.tls_cert_key)"), weechat_prefix ("error"), RELAY_PLUGIN_NAME); } new_client->status = RELAY_STATUS_CONNECTING; /* * set Diffie-Hellman parameters on first TLS connection from a * client (done only one time) */ if (!relay_gnutls_dh_params) { relay_gnutls_dh_params = malloc (sizeof (*relay_gnutls_dh_params)); if (relay_gnutls_dh_params) { gnutls_dh_params_init (relay_gnutls_dh_params); #if LIBGNUTLS_VERSION_NUMBER >= 0x020c00 /* 2.12.0 */ bits = gnutls_sec_param_to_pk_bits (GNUTLS_PK_DH, GNUTLS_SEC_PARAM_LOW); #else /* default for old gnutls */ bits = 1024; #endif /* LIBGNUTLS_VERSION_NUMBER >= 0x020c00 */ gnutls_dh_params_generate2 (*relay_gnutls_dh_params, bits); gnutls_certificate_set_dh_params (relay_gnutls_x509_cred, *relay_gnutls_dh_params); } } gnutls_init (&(new_client->gnutls_sess), GNUTLS_SERVER); if (relay_gnutls_priority_cache) gnutls_priority_set (new_client->gnutls_sess, *relay_gnutls_priority_cache); gnutls_credentials_set (new_client->gnutls_sess, GNUTLS_CRD_CERTIFICATE, relay_gnutls_x509_cred); gnutls_certificate_server_set_request (new_client->gnutls_sess, GNUTLS_CERT_IGNORE); gnutls_transport_set_ptr (new_client->gnutls_sess, (gnutls_transport_ptr_t) ((ptrdiff_t) new_client->sock)); ptr_option = weechat_config_get ("weechat.network.gnutls_handshake_timeout"); new_client->hook_timer_handshake = weechat_hook_timer (1000 / 10, 0, (ptr_option) ? weechat_config_integer (ptr_option) * 10 : 30 * 10, &relay_client_handshake_timer_cb, new_client, NULL); } new_client->protocol_data = NULL; switch (new_client->protocol) { case RELAY_PROTOCOL_WEECHAT: relay_weechat_alloc (new_client); if (!new_client->tls) { new_client->status = relay_weechat_get_initial_status (new_client); } break; case RELAY_PROTOCOL_IRC: relay_irc_alloc (new_client); if (!new_client->tls) { new_client->status = relay_irc_get_initial_status (new_client); } break; case RELAY_PROTOCOL_API: #ifdef HAVE_CJSON relay_api_alloc (new_client); if (!new_client->tls) { new_client->status = relay_api_get_initial_status (new_client); } #endif /* HAVE_CJSON */ break; case RELAY_NUM_PROTOCOLS: break; } new_client->outqueue = NULL; new_client->last_outqueue = NULL; new_client->prev_client = NULL; new_client->next_client = relay_clients; if (relay_clients) relay_clients->prev_client = new_client; else last_relay_client = new_client; relay_clients = new_client; if (relay_config_display_clients[new_client->protocol]) { if (server->unix_socket) { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s: new client on path %s: %s%s%s (%s)"), RELAY_PLUGIN_NAME, server->path, RELAY_COLOR_CHAT_CLIENT, new_client->desc, RELAY_COLOR_CHAT, _(relay_status_string[new_client->status])); } else { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s: new client on port %s: %s%s%s (%s)"), RELAY_PLUGIN_NAME, server->path, RELAY_COLOR_CHAT_CLIENT, new_client->desc, RELAY_COLOR_CHAT, _(relay_status_string[new_client->status])); } } if (new_client->sock >= 0) { new_client->hook_fd = weechat_hook_fd (new_client->sock, 1, 0, 0, &relay_client_recv_cb, new_client, NULL); } relay_client_count++; if (!relay_buffer && relay_config_auto_open_buffer[new_client->protocol]) relay_buffer_open (); relay_client_send_signal (new_client); relay_buffer_refresh (WEECHAT_HOTLIST_PRIVATE); } else { weechat_printf_date_tags (NULL, 0, "relay_client", _("%s%s: not enough memory for new client"), weechat_prefix ("error"), RELAY_PLUGIN_NAME); } return new_client; } /* * Creates a new client using an infolist. * * This is called to restore clients after /upgrade. */ struct t_relay_client * relay_client_new_with_infolist (struct t_infolist *infolist) { struct t_relay_client *new_client; const char *str; Bytef *ptr_dict; int dict_size, ws_frame_size; void *ptr_ws_frame; new_client = malloc (sizeof (*new_client)); if (new_client) { new_client->id = weechat_infolist_integer (infolist, "id"); new_client->desc = NULL; new_client->sock = weechat_infolist_integer (infolist, "sock"); new_client->server_port = weechat_infolist_integer (infolist, "server_port"); /* "tls" replaces "ssl" in WeeChat 4.0.0 */ if (weechat_infolist_search_var (infolist, "tls")) new_client->tls = weechat_infolist_integer (infolist, "tls"); else new_client->tls = weechat_infolist_integer (infolist, "ssl"); new_client->gnutls_sess = NULL; new_client->fake_send_func = NULL; new_client->hook_timer_handshake = NULL; new_client->gnutls_handshake_ok = 0; new_client->websocket = weechat_infolist_integer (infolist, "websocket"); new_client->ws_deflate = relay_websocket_deflate_alloc (); new_client->ws_deflate->enabled = weechat_infolist_integer (infolist, "ws_deflate_enabled"); new_client->ws_deflate->server_context_takeover = weechat_infolist_integer (infolist, "ws_deflate_server_context_takeover"); new_client->ws_deflate->client_context_takeover = weechat_infolist_integer (infolist, "ws_deflate_client_context_takeover"); new_client->ws_deflate->window_bits_deflate = weechat_infolist_integer (infolist, "ws_deflate_window_bits_deflate"); new_client->ws_deflate->window_bits_inflate = weechat_infolist_integer (infolist, "ws_deflate_window_bits_inflate"); new_client->ws_deflate->strm_deflate = NULL; new_client->ws_deflate->strm_inflate = NULL; if (weechat_infolist_search_var (infolist, "ws_deflate_strm_deflate_dict")) { ptr_dict = weechat_infolist_buffer (infolist, "ws_deflate_strm_deflate_dict", &dict_size); if (ptr_dict) { new_client->ws_deflate->strm_deflate = calloc (1, sizeof (*new_client->ws_deflate->strm_deflate)); if (new_client->ws_deflate->strm_deflate) { if (relay_websocket_deflate_init_stream_deflate (new_client->ws_deflate)) { deflateSetDictionary (new_client->ws_deflate->strm_deflate, ptr_dict, dict_size); } } } } if (weechat_infolist_search_var (infolist, "ws_deflate_strm_inflate_dict")) { ptr_dict = weechat_infolist_buffer (infolist, "ws_deflate_strm_inflate_dict", &dict_size); if (ptr_dict) { new_client->ws_deflate->strm_inflate = calloc (1, sizeof (*new_client->ws_deflate->strm_inflate)); if (new_client->ws_deflate->strm_inflate) { if (relay_websocket_deflate_init_stream_inflate (new_client->ws_deflate)) { inflateSetDictionary (new_client->ws_deflate->strm_inflate, ptr_dict, dict_size); } } } } new_client->http_req = relay_http_request_alloc (); new_client->address = strdup (weechat_infolist_string (infolist, "address")); str = weechat_infolist_string (infolist, "real_ip"); new_client->real_ip = (str) ? strdup (str) : NULL; new_client->status = weechat_infolist_integer (infolist, "status"); new_client->protocol = weechat_infolist_integer (infolist, "protocol"); str = weechat_infolist_string (infolist, "protocol_string"); new_client->protocol_string = (str) ? strdup (str) : NULL; str = weechat_infolist_string (infolist, "protocol_args"); new_client->protocol_args = (str) ? strdup (str) : NULL; /* "nonce" is new in WeeChat 2.9 */ if (weechat_infolist_search_var (infolist, "nonce")) new_client->nonce = strdup (weechat_infolist_string (infolist, "nonce")); else new_client->nonce = relay_auth_generate_nonce ( weechat_config_integer (relay_config_network_nonce_size)); /* "password_hash_algo" is new in WeeChat 2.9 */ if (weechat_infolist_search_var (infolist, "password_hash_algo")) new_client->password_hash_algo = weechat_infolist_integer (infolist, "password_hash_algo"); else new_client->password_hash_algo = RELAY_AUTH_PASSWORD_HASH_PLAIN; new_client->listen_start_time = weechat_infolist_time (infolist, "listen_start_time"); new_client->start_time = weechat_infolist_time (infolist, "start_time"); new_client->end_time = weechat_infolist_time (infolist, "end_time"); if (new_client->sock >= 0) { new_client->hook_fd = weechat_hook_fd (new_client->sock, 1, 0, 0, &relay_client_recv_cb, new_client, NULL); } else new_client->hook_fd = NULL; new_client->hook_timer_send = NULL; new_client->last_activity = weechat_infolist_time (infolist, "last_activity"); sscanf (weechat_infolist_string (infolist, "bytes_recv"), "%llu", &(new_client->bytes_recv)); sscanf (weechat_infolist_string (infolist, "bytes_sent"), "%llu", &(new_client->bytes_sent)); new_client->recv_data_type = weechat_infolist_integer (infolist, "recv_data_type"); new_client->send_data_type = weechat_infolist_integer (infolist, "send_data_type"); ptr_ws_frame = weechat_infolist_buffer (infolist, "partial_ws_frame", &ws_frame_size); if (ptr_ws_frame && (ws_frame_size > 0)) { new_client->partial_ws_frame = malloc (ws_frame_size); if (new_client->partial_ws_frame) { memcpy (new_client->partial_ws_frame, ptr_ws_frame, ws_frame_size); new_client->partial_ws_frame_size = ws_frame_size; } } str = weechat_infolist_string (infolist, "partial_message"); new_client->partial_message = (str) ? strdup (str) : NULL; str = weechat_infolist_string (infolist, "desc"); if (str) new_client->desc = strdup (str); else relay_client_set_desc (new_client); switch (new_client->protocol) { case RELAY_PROTOCOL_WEECHAT: relay_weechat_alloc_with_infolist (new_client, infolist); break; case RELAY_PROTOCOL_IRC: relay_irc_alloc_with_infolist (new_client, infolist); break; case RELAY_PROTOCOL_API: #ifdef HAVE_CJSON relay_api_alloc_with_infolist (new_client, infolist); #endif /* HAVE_CJSON */ break; case RELAY_NUM_PROTOCOLS: break; } new_client->outqueue = NULL; new_client->last_outqueue = NULL; new_client->prev_client = NULL; new_client->next_client = relay_clients; if (relay_clients) relay_clients->prev_client = new_client; else last_relay_client = new_client; relay_clients = new_client; relay_client_count++; } return new_client; } /* * Sets status for a client. */ void relay_client_set_status (struct t_relay_client *client, enum t_relay_status status) { struct t_relay_server *ptr_server; int old_status; /* * IMPORTANT: if changes are made in this function or sub-functions called, * please also update the function relay_client_add_to_infolist: * when the flag force_disconnected_state is set to 1 we simulate * a disconnected state for client in infolist (used on /upgrade -save) */ old_status = client->status; client->status = status; if ((client->status == RELAY_STATUS_CONNECTED) && relay_config_display_clients[client->protocol]) { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s: client %s%s%s connected/authenticated"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT); } else if (RELAY_STATUS_HAS_ENDED(client->status)) { client->end_time = time (NULL); if (old_status == RELAY_STATUS_CONNECTED) { /* * set the last client disconnect time * (only if the client was connected) */ ptr_server = relay_server_search (client->protocol_string); if (ptr_server) ptr_server->last_client_disconnect = client->end_time; } relay_client_outqueue_free_all (client); if (client->hook_timer_handshake) { weechat_unhook (client->hook_timer_handshake); client->hook_timer_handshake = NULL; } client->gnutls_handshake_ok = 0; if (client->hook_fd) { weechat_unhook (client->hook_fd); client->hook_fd = NULL; } if (client->hook_timer_send) { weechat_unhook (client->hook_timer_send); client->hook_timer_send = NULL; } switch (client->protocol) { case RELAY_PROTOCOL_WEECHAT: relay_weechat_close_connection (client); break; case RELAY_PROTOCOL_IRC: relay_irc_close_connection (client); break; case RELAY_PROTOCOL_API: #ifdef HAVE_CJSON relay_api_close_connection (client); #endif /* HAVE_CJSON */ break; case RELAY_NUM_PROTOCOLS: break; } switch (client->status) { case RELAY_STATUS_AUTH_FAILED: weechat_printf_date_tags (NULL, 0, "relay_client", _("%s%s: authentication failed with " "client %s%s%s"), weechat_prefix ("error"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT); break; case RELAY_STATUS_DISCONNECTED: if (relay_config_display_clients[client->protocol]) { weechat_printf_date_tags ( NULL, 0, "relay_client", _("%s: disconnected from client %s%s%s"), RELAY_PLUGIN_NAME, RELAY_COLOR_CHAT_CLIENT, client->desc, RELAY_COLOR_CHAT); } break; default: break; } if (client->sock >= 0) { if (client->tls && client->gnutls_handshake_ok) gnutls_bye (client->gnutls_sess, GNUTLS_SHUT_WR); close (client->sock); client->sock = -1; if (client->tls) gnutls_deinit (client->gnutls_sess); } } relay_client_send_signal (client); relay_buffer_refresh (WEECHAT_HOTLIST_MESSAGE); } /* * Removes a client. */ void relay_client_free (struct t_relay_client *client) { struct t_relay_client *new_relay_clients; if (!client) return; /* remove client from list */ if (last_relay_client == client) last_relay_client = client->prev_client; if (client->prev_client) { (client->prev_client)->next_client = client->next_client; new_relay_clients = relay_clients; } else new_relay_clients = client->next_client; if (client->next_client) (client->next_client)->prev_client = client->prev_client; /* free data */ free (client->desc); free (client->address); free (client->real_ip); free (client->protocol_string); free (client->protocol_args); free (client->nonce); weechat_unhook (client->hook_timer_handshake); relay_websocket_deflate_free (client->ws_deflate); relay_http_request_free (client->http_req); weechat_unhook (client->hook_fd); weechat_unhook (client->hook_timer_send); free (client->partial_ws_frame); free (client->partial_message); if (client->protocol_data) { switch (client->protocol) { case RELAY_PROTOCOL_WEECHAT: relay_weechat_free (client); break; case RELAY_PROTOCOL_IRC: relay_irc_free (client); break; case RELAY_PROTOCOL_API: #ifdef HAVE_CJSON relay_api_free (client); #endif /* HAVE_CJSON */ break; case RELAY_NUM_PROTOCOLS: break; } } relay_client_outqueue_free_all (client); free (client); relay_clients = new_relay_clients; relay_client_count--; if (relay_buffer_selected_line >= relay_client_count) { relay_buffer_selected_line = (relay_client_count == 0) ? 0 : relay_client_count - 1; } } /* * Removes all clients. */ void relay_client_free_all () { while (relay_clients) { relay_client_free (relay_clients); } } /* * Disconnects one client. */ void relay_client_disconnect (struct t_relay_client *client) { if (client->sock >= 0) { relay_client_set_status (client, RELAY_STATUS_DISCONNECTED); } } /* * Disconnects all clients. */ void relay_client_disconnect_all () { struct t_relay_client *ptr_client; for (ptr_client = relay_clients; ptr_client; ptr_client = ptr_client->next_client) { relay_client_disconnect (ptr_client); } } /* * Adds a client in an infolist. * * If force_disconnected_state == 1, the infolist contains the client * in a disconnected state (but the client is unchanged, still connected if it * was). * * Returns: * 1: OK * 0: error */ int relay_client_add_to_infolist (struct t_infolist *infolist, struct t_relay_client *client, int force_disconnected_state) { struct t_infolist_item *ptr_item; char value[128]; Bytef *dict; uInt dict_size; if (!infolist || !client) return 0; ptr_item = weechat_infolist_new_item (infolist); if (!ptr_item) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "id", client->id)) return 0; if (!weechat_infolist_new_var_string (ptr_item, "desc", client->desc)) return 0; if (!RELAY_STATUS_HAS_ENDED(client->status) && force_disconnected_state) { if (!weechat_infolist_new_var_integer (ptr_item, "sock", -1)) return 0; if (!weechat_infolist_new_var_pointer (ptr_item, "hook_timer_handshake", NULL)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "gnutls_handshake_ok", 0)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "status", RELAY_STATUS_DISCONNECTED)) return 0; if (!weechat_infolist_new_var_time (ptr_item, "end_time", time (NULL))) return 0; if (!weechat_infolist_new_var_buffer (ptr_item, "partial_ws_frame", NULL, 0)) return 0; if (!weechat_infolist_new_var_string (ptr_item, "partial_message", NULL)) return 0; } else { if (!weechat_infolist_new_var_integer (ptr_item, "sock", client->sock)) return 0; if (!weechat_infolist_new_var_pointer (ptr_item, "hook_timer_handshake", client->hook_timer_handshake)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "gnutls_handshake_ok", client->gnutls_handshake_ok)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "status", client->status)) return 0; if (!weechat_infolist_new_var_time (ptr_item, "end_time", client->end_time)) return 0; if (!weechat_infolist_new_var_buffer (ptr_item, "partial_ws_frame", client->partial_ws_frame, client->partial_ws_frame_size)) return 0; if (!weechat_infolist_new_var_string (ptr_item, "partial_message", client->partial_message)) return 0; } if (!weechat_infolist_new_var_integer (ptr_item, "server_port", client->server_port)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "tls", client->tls)) return 0; if (!weechat_infolist_new_var_pointer (ptr_item, "fake_send_func", client->fake_send_func)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "websocket", client->websocket)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "ws_deflate_enabled", client->ws_deflate->enabled)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "ws_deflate_server_context_takeover", client->ws_deflate->server_context_takeover)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "ws_deflate_client_context_takeover", client->ws_deflate->client_context_takeover)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "ws_deflate_window_bits_deflate", client->ws_deflate->window_bits_deflate)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "ws_deflate_window_bits_inflate", client->ws_deflate->window_bits_inflate)) return 0; if (!weechat_infolist_new_var_pointer (ptr_item, "ws_deflate_strm_deflate", client->ws_deflate->strm_deflate)) return 0; if (!weechat_infolist_new_var_pointer (ptr_item, "ws_deflate_strm_inflate", client->ws_deflate->strm_inflate)) return 0; if (client->ws_deflate->strm_deflate || client->ws_deflate->strm_inflate) { /* save the deflate/inflate dictionary, as it's required after /upgrade */ dict = malloc (32768); if (dict) { if (client->ws_deflate->strm_deflate) { if (deflateGetDictionary (client->ws_deflate->strm_deflate, dict, &dict_size) == Z_OK) { weechat_infolist_new_var_buffer (ptr_item, "ws_deflate_strm_deflate_dict", dict, dict_size); } } if (client->ws_deflate->strm_inflate) { if (inflateGetDictionary (client->ws_deflate->strm_inflate, dict, &dict_size) == Z_OK) { weechat_infolist_new_var_buffer (ptr_item, "ws_deflate_strm_inflate_dict", dict, dict_size); } } free (dict); } } if (!weechat_infolist_new_var_string (ptr_item, "address", client->address)) return 0; if (!weechat_infolist_new_var_string (ptr_item, "real_ip", client->real_ip)) return 0; if (!weechat_infolist_new_var_string (ptr_item, "status_string", relay_status_string[client->status])) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "protocol", client->protocol)) return 0; if (!weechat_infolist_new_var_string (ptr_item, "protocol_string", relay_protocol_string[client->protocol])) return 0; if (!weechat_infolist_new_var_string (ptr_item, "protocol_string", client->protocol_string)) return 0; if (!weechat_infolist_new_var_string (ptr_item, "protocol_args", client->protocol_args)) return 0; if (!weechat_infolist_new_var_string (ptr_item, "nonce", client->nonce)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "password_hash_algo", client->password_hash_algo)) return 0; if (!weechat_infolist_new_var_time (ptr_item, "listen_start_time", client->listen_start_time)) return 0; if (!weechat_infolist_new_var_time (ptr_item, "start_time", client->start_time)) return 0; if (!weechat_infolist_new_var_pointer (ptr_item, "hook_fd", client->hook_fd)) return 0; if (!weechat_infolist_new_var_pointer (ptr_item, "hook_timer_send", client->hook_timer_send)) return 0; if (!weechat_infolist_new_var_time (ptr_item, "last_activity", client->last_activity)) return 0; snprintf (value, sizeof (value), "%llu", client->bytes_recv); if (!weechat_infolist_new_var_string (ptr_item, "bytes_recv", value)) return 0; snprintf (value, sizeof (value), "%llu", client->bytes_sent); if (!weechat_infolist_new_var_string (ptr_item, "bytes_sent", value)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "recv_data_type", client->recv_data_type)) return 0; if (!weechat_infolist_new_var_integer (ptr_item, "send_data_type", client->send_data_type)) return 0; switch (client->protocol) { case RELAY_PROTOCOL_WEECHAT: relay_weechat_add_to_infolist (ptr_item, client, force_disconnected_state); break; case RELAY_PROTOCOL_IRC: relay_irc_add_to_infolist (ptr_item, client, force_disconnected_state); break; case RELAY_PROTOCOL_API: #ifdef HAVE_CJSON relay_api_add_to_infolist (ptr_item, client, force_disconnected_state); #endif /* HAVE_CJSON */ break; case RELAY_NUM_PROTOCOLS: break; } return 1; } /* * Prints clients in WeeChat log file (usually for crash dump). */ void relay_client_print_log () { struct t_relay_client *ptr_client; for (ptr_client = relay_clients; ptr_client; ptr_client = ptr_client->next_client) { weechat_log_printf (""); weechat_log_printf ("[relay client (addr:%p)]", ptr_client); weechat_log_printf (" id. . . . . . . . . . . . : %d", ptr_client->id); weechat_log_printf (" desc. . . . . . . . . . . : '%s'", ptr_client->desc); weechat_log_printf (" sock. . . . . . . . . . . : %d", ptr_client->sock); weechat_log_printf (" server_port . . . . . . . : %d", ptr_client->server_port); weechat_log_printf (" tls . . . . . . . . . . . : %d", ptr_client->tls); weechat_log_printf (" gnutls_sess . . . . . . . : %p", ptr_client->gnutls_sess); weechat_log_printf (" fake_send_func. . . . . . : %p", ptr_client->fake_send_func); weechat_log_printf (" hook_timer_handshake. . . : %p", ptr_client->hook_timer_handshake); weechat_log_printf (" gnutls_handshake_ok . . . : %p", ptr_client->gnutls_handshake_ok); weechat_log_printf (" websocket . . . . . . . . ; %d", ptr_client->websocket); relay_websocket_deflate_print_log (ptr_client->ws_deflate, ""); relay_http_print_log_request (ptr_client->http_req); weechat_log_printf (" address . . . . . . . . . : '%s'", ptr_client->address); weechat_log_printf (" real_ip . . . . . . . . . : '%s'", ptr_client->real_ip); weechat_log_printf (" status. . . . . . . . . . : %d (%s)", ptr_client->status, relay_status_string[ptr_client->status]); weechat_log_printf (" protocol. . . . . . . . . : %d (%s)", ptr_client->protocol, relay_protocol_string[ptr_client->protocol]); weechat_log_printf (" protocol_string . . . . . : '%s'", ptr_client->protocol_string); weechat_log_printf (" protocol_args . . . . . . : '%s'", ptr_client->protocol_args); weechat_log_printf (" nonce . . . . . . . . . . : '%s'", ptr_client->nonce); weechat_log_printf (" password_hash_algo. . . . : %d (%s)", ptr_client->password_hash_algo, (ptr_client->password_hash_algo >= 0) ? relay_auth_password_hash_algo_name[ptr_client->password_hash_algo] : ""); weechat_log_printf (" listen_start_time . . . . : %lld", (long long)ptr_client->listen_start_time); weechat_log_printf (" start_time. . . . . . . . : %lld", (long long)ptr_client->start_time); weechat_log_printf (" end_time. . . . . . . . . : %lld", (long long)ptr_client->end_time); weechat_log_printf (" hook_fd . . . . . . . . . : %p", ptr_client->hook_fd); weechat_log_printf (" hook_timer_send . . . . . : %p", ptr_client->hook_timer_send); weechat_log_printf (" last_activity . . . . . . : %lld", (long long)ptr_client->last_activity); weechat_log_printf (" bytes_recv. . . . . . . . : %llu", ptr_client->bytes_recv); weechat_log_printf (" bytes_sent. . . . . . . . : %llu", ptr_client->bytes_sent); weechat_log_printf (" recv_data_type. . . . . . : %d (%s)", ptr_client->recv_data_type, relay_client_data_type_string[ptr_client->recv_data_type]); weechat_log_printf (" send_data_type. . . . . . : %d (%s)", ptr_client->send_data_type, relay_client_data_type_string[ptr_client->send_data_type]); weechat_log_printf (" partial_ws_frame. . . . . : %p (%d bytes)", ptr_client->partial_ws_frame, ptr_client->partial_ws_frame_size); weechat_log_printf (" partial_message . . . . . : '%s'", ptr_client->partial_message); weechat_log_printf (" protocol_data . . . . . . : %p", ptr_client->protocol_data); switch (ptr_client->protocol) { case RELAY_PROTOCOL_WEECHAT: relay_weechat_print_log (ptr_client); break; case RELAY_PROTOCOL_IRC: relay_irc_print_log (ptr_client); break; case RELAY_PROTOCOL_API: #ifdef HAVE_CJSON relay_api_print_log (ptr_client); #endif /* HAVE_CJSON */ break; case RELAY_NUM_PROTOCOLS: break; } weechat_log_printf (" outqueue. . . . . . . . . : %p", ptr_client->outqueue); weechat_log_printf (" last_outqueue . . . . . . : %p", ptr_client->last_outqueue); weechat_log_printf (" prev_client . . . . . . . : %p", ptr_client->prev_client); weechat_log_printf (" next_client . . . . . . . : %p", ptr_client->next_client); } }