mirror of
https://github.com/weechat/weechat.git
synced 2026-06-12 14:14:48 +02:00
0315c53a9e
Add an "int themable" field on struct t_config_option. The flag is set
automatically for every CONFIG_OPTION_TYPE_COLOR option, and may be set
explicitly on any other type by suffixing the type argument with
"|themable" in the call to config_file_new_option (e.g. "string|themable"
for a string option whose value contains "${color:...}" references).
Opt in the relevant string options in core (buffer_time_format,
day_change_message_*, item_time_format, nick_color_force, prefix_*,
chat_nick_colors, eval_syntax_colors, color palette aliases) and in the
buflist, fset, irc, relay plugins.
The flag is exposed via hdata, infolist, and print_log so scripts and
/debug can read it. This is the foundation for an upcoming /theme
command that will only be allowed to modify themable options.
2030 lines
70 KiB
C
2030 lines
70 KiB
C
/*
|
|
* SPDX-FileCopyrightText: 2003-2026 Sébastien Helleu <flashcode@flashtux.org>
|
|
*
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*
|
|
* 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/* Relay configuration options (file relay.conf) */
|
|
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <limits.h>
|
|
#include <regex.h>
|
|
#include <sys/un.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <errno.h>
|
|
|
|
#include "../weechat-plugin.h"
|
|
#include "relay.h"
|
|
#include "relay-client.h"
|
|
#include "relay-config.h"
|
|
#include "relay-buffer.h"
|
|
#include "relay-network.h"
|
|
#include "relay-remote.h"
|
|
#include "relay-server.h"
|
|
#include "irc/relay-irc.h"
|
|
|
|
|
|
struct t_config_file *relay_config_file = NULL;
|
|
|
|
/* sections */
|
|
|
|
struct t_config_section *relay_config_section_look = NULL;
|
|
struct t_config_section *relay_config_section_color = NULL;
|
|
struct t_config_section *relay_config_section_network = NULL;
|
|
struct t_config_section *relay_config_section_irc = NULL;
|
|
struct t_config_section *relay_config_section_api = NULL;
|
|
struct t_config_section *relay_config_section_port = NULL;
|
|
struct t_config_section *relay_config_section_path = NULL;
|
|
struct t_config_section *relay_config_section_remote = NULL;
|
|
|
|
/* relay config, look section */
|
|
|
|
struct t_config_option *relay_config_look_auto_open_buffer = NULL;
|
|
struct t_config_option *relay_config_look_display_clients = NULL;
|
|
struct t_config_option *relay_config_look_raw_messages = NULL;
|
|
struct t_config_option *relay_config_look_raw_messages_max_length = NULL;
|
|
|
|
/* relay config, color section */
|
|
|
|
struct t_config_option *relay_config_color_client = NULL;
|
|
struct t_config_option *relay_config_color_status[RELAY_NUM_STATUS] = {
|
|
NULL, NULL, NULL, NULL, NULL,
|
|
};
|
|
struct t_config_option *relay_config_color_text = NULL;
|
|
struct t_config_option *relay_config_color_text_bg = NULL;
|
|
struct t_config_option *relay_config_color_text_selected = NULL;
|
|
|
|
/* relay config, network section */
|
|
|
|
struct t_config_option *relay_config_network_allow_empty_password = NULL;
|
|
struct t_config_option *relay_config_network_allowed_ips = NULL;
|
|
struct t_config_option *relay_config_network_auth_timeout = NULL;
|
|
struct t_config_option *relay_config_network_bind_address = NULL;
|
|
struct t_config_option *relay_config_network_clients_purge_delay = NULL;
|
|
struct t_config_option *relay_config_network_commands = NULL;
|
|
struct t_config_option *relay_config_network_compression = NULL;
|
|
struct t_config_option *relay_config_network_ipv6 = NULL;
|
|
struct t_config_option *relay_config_network_max_clients = NULL;
|
|
struct t_config_option *relay_config_network_nonce_size = NULL;
|
|
struct t_config_option *relay_config_network_password = NULL;
|
|
struct t_config_option *relay_config_network_password_hash_algo = NULL;
|
|
struct t_config_option *relay_config_network_password_hash_iterations = NULL;
|
|
struct t_config_option *relay_config_network_time_window = NULL;
|
|
struct t_config_option *relay_config_network_tls_cert_key = NULL;
|
|
struct t_config_option *relay_config_network_tls_priorities = NULL;
|
|
struct t_config_option *relay_config_network_totp_secret = NULL;
|
|
struct t_config_option *relay_config_network_totp_window = NULL;
|
|
struct t_config_option *relay_config_network_unix_socket_permissions = NULL;
|
|
struct t_config_option *relay_config_network_websocket_allowed_origins = NULL;
|
|
struct t_config_option *relay_config_network_websocket_permessage_deflate = NULL;
|
|
|
|
/* relay config, irc section */
|
|
|
|
struct t_config_option *relay_config_irc_backlog_max_minutes = NULL;
|
|
struct t_config_option *relay_config_irc_backlog_max_number = NULL;
|
|
struct t_config_option *relay_config_irc_backlog_since_last_disconnect = NULL;
|
|
struct t_config_option *relay_config_irc_backlog_since_last_message = NULL;
|
|
struct t_config_option *relay_config_irc_backlog_tags = NULL;
|
|
struct t_config_option *relay_config_irc_backlog_time_format = NULL;
|
|
|
|
/* relay config, api section */
|
|
|
|
struct t_config_option *relay_config_api_remote_autoreconnect_delay_growing = NULL;
|
|
struct t_config_option *relay_config_api_remote_autoreconnect_delay_max = NULL;
|
|
struct t_config_option *relay_config_api_remote_get_lines = NULL;
|
|
struct t_config_option *relay_config_api_remote_input_cmd_local = NULL;
|
|
struct t_config_option *relay_config_api_remote_input_cmd_remote = NULL;
|
|
|
|
/* other */
|
|
|
|
int relay_config_auto_open_buffer[RELAY_NUM_PROTOCOLS];
|
|
int relay_config_display_clients[RELAY_NUM_PROTOCOLS];
|
|
regex_t *relay_config_regex_allowed_ips = NULL;
|
|
regex_t *relay_config_regex_websocket_allowed_origins = NULL;
|
|
struct t_hashtable *relay_config_hashtable_irc_backlog_tags = NULL;
|
|
char **relay_config_network_password_hash_algo_list = NULL;
|
|
|
|
|
|
/*
|
|
* Callback for changes on option "relay.look.auto_open_buffer".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_auto_open_buffer_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
const char *auto_open_buffer;
|
|
const char *value_on = "irc,weechat", *value_off = "";
|
|
char **items;
|
|
int i, num_items, protocol;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
auto_open_buffer = weechat_config_string (relay_config_look_auto_open_buffer);
|
|
|
|
/* old option was a boolean, use these values for compatibility */
|
|
if (strcmp (auto_open_buffer, "on") == 0)
|
|
auto_open_buffer = value_on;
|
|
else if (strcmp (auto_open_buffer, "off") == 0)
|
|
auto_open_buffer = value_off;
|
|
|
|
for (i = 0; i < RELAY_NUM_PROTOCOLS; i++)
|
|
{
|
|
relay_config_auto_open_buffer[i] = 0;
|
|
}
|
|
|
|
if (auto_open_buffer[0])
|
|
{
|
|
items = weechat_string_split (auto_open_buffer, ",", NULL, 0, 0,
|
|
&num_items);
|
|
if (items)
|
|
{
|
|
for (i = 0; i < num_items; i++)
|
|
{
|
|
protocol = relay_protocol_search (items[i]);
|
|
if (protocol >= 0)
|
|
relay_config_auto_open_buffer[protocol] = 1;
|
|
}
|
|
weechat_string_free_split (items);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on option "relay.look.display_clients".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_display_clients_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
const char *display_clients;
|
|
char **items;
|
|
int i, num_items, protocol;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
display_clients = weechat_config_string (relay_config_look_display_clients);
|
|
|
|
for (i = 0; i < RELAY_NUM_PROTOCOLS; i++)
|
|
{
|
|
relay_config_display_clients[i] = 0;
|
|
}
|
|
|
|
if (display_clients[0])
|
|
{
|
|
items = weechat_string_split (display_clients, ",", NULL, 0, 0,
|
|
&num_items);
|
|
if (items)
|
|
{
|
|
for (i = 0; i < num_items; i++)
|
|
{
|
|
protocol = relay_protocol_search (items[i]);
|
|
if (protocol >= 0)
|
|
relay_config_display_clients[protocol] = 1;
|
|
}
|
|
weechat_string_free_split (items);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on options that require a refresh of relay list.
|
|
*/
|
|
|
|
void
|
|
relay_config_refresh_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
if (relay_buffer)
|
|
relay_buffer_refresh (NULL);
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on option "relay.network.allowed_ips".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_network_allowed_ips (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
const char *allowed_ips;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
if (relay_config_regex_allowed_ips)
|
|
{
|
|
regfree (relay_config_regex_allowed_ips);
|
|
free (relay_config_regex_allowed_ips);
|
|
relay_config_regex_allowed_ips = NULL;
|
|
}
|
|
|
|
allowed_ips = weechat_config_string (relay_config_network_allowed_ips);
|
|
if (allowed_ips && allowed_ips[0])
|
|
{
|
|
relay_config_regex_allowed_ips = malloc (sizeof (*relay_config_regex_allowed_ips));
|
|
if (relay_config_regex_allowed_ips)
|
|
{
|
|
if (weechat_string_regcomp (relay_config_regex_allowed_ips,
|
|
allowed_ips,
|
|
REG_EXTENDED | REG_ICASE) != 0)
|
|
{
|
|
free (relay_config_regex_allowed_ips);
|
|
relay_config_regex_allowed_ips = NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on option "relay.network.password_hash_algo".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_network_password_hash_algo (const void *pointer,
|
|
void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
if (relay_config_network_password_hash_algo_list)
|
|
{
|
|
weechat_string_free_split (relay_config_network_password_hash_algo_list);
|
|
relay_config_network_password_hash_algo_list = NULL;
|
|
}
|
|
|
|
relay_config_network_password_hash_algo_list = weechat_string_split (
|
|
weechat_config_string (relay_config_network_password_hash_algo),
|
|
",",
|
|
NULL,
|
|
WEECHAT_STRING_SPLIT_STRIP_LEFT
|
|
| WEECHAT_STRING_SPLIT_STRIP_RIGHT
|
|
| WEECHAT_STRING_SPLIT_COLLAPSE_SEPS,
|
|
0,
|
|
NULL);
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on option "relay.network.bind_address".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_network_bind_address_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
struct t_relay_server *ptr_server;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
for (ptr_server = relay_servers; ptr_server;
|
|
ptr_server = ptr_server->next_server)
|
|
{
|
|
relay_server_close_socket (ptr_server);
|
|
relay_server_create_socket (ptr_server);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on option "relay.network.ipv6".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_network_ipv6_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
struct t_relay_server *ptr_server;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
for (ptr_server = relay_servers; ptr_server;
|
|
ptr_server = ptr_server->next_server)
|
|
{
|
|
relay_server_get_protocol_args (ptr_server->protocol_string,
|
|
&ptr_server->ipv4, &ptr_server->ipv6,
|
|
NULL, &ptr_server->unix_socket, NULL, NULL);
|
|
relay_server_close_socket (ptr_server);
|
|
relay_server_create_socket (ptr_server);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on option "relay.network.tls_cert_key".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_network_tls_cert_key (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
if (relay_network_init_ok)
|
|
relay_network_set_tls_cert_key (1);
|
|
}
|
|
|
|
/*
|
|
* Check if option "relay.network.totp_secret" is valid.
|
|
*
|
|
* Return:
|
|
* 1: value is valid
|
|
* 0: value is not valid
|
|
*/
|
|
|
|
int
|
|
relay_config_check_network_totp_secret (const void *pointer, void *data,
|
|
struct t_config_option *option,
|
|
const char *value)
|
|
{
|
|
char *totp_secret, *secret;
|
|
int rc, length;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
totp_secret = NULL;
|
|
secret = NULL;
|
|
|
|
totp_secret = weechat_string_eval_expression (value, NULL, NULL, NULL);
|
|
if (totp_secret && totp_secret[0])
|
|
{
|
|
secret = malloc (strlen (totp_secret) + 1);
|
|
if (!secret)
|
|
goto error;
|
|
length = weechat_string_base_decode ("32", totp_secret, secret);
|
|
if (length < 0)
|
|
goto error;
|
|
}
|
|
|
|
rc = 1;
|
|
goto end;
|
|
|
|
error:
|
|
rc = 0;
|
|
weechat_printf (NULL,
|
|
_("%s%s: invalid value for option "
|
|
"\"relay.network.totp_secret\"; it must be a valid "
|
|
"string encoded in base32 "
|
|
"(only letters and digits from 2 to 7)"),
|
|
weechat_prefix ("error"), RELAY_PLUGIN_NAME);
|
|
|
|
end:
|
|
free (totp_secret);
|
|
free (secret);
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* Check if option "relay.network.tls_priorities" is valid.
|
|
*
|
|
* Return:
|
|
* 1: value is valid
|
|
* 0: value is not valid
|
|
*/
|
|
|
|
int
|
|
relay_config_check_network_tls_priorities (const void *pointer, void *data,
|
|
struct t_config_option *option,
|
|
const char *value)
|
|
{
|
|
gnutls_priority_t priority_cache;
|
|
const char *pos_error;
|
|
int rc;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
pos_error = value;
|
|
|
|
if (value && value[0])
|
|
{
|
|
rc = gnutls_priority_init (&priority_cache, value, &pos_error);
|
|
if (rc == GNUTLS_E_SUCCESS)
|
|
{
|
|
gnutls_priority_deinit (priority_cache);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
weechat_printf (NULL,
|
|
_("%s%s: invalid priorities string, error "
|
|
"at this position in string: \"%s\""),
|
|
weechat_prefix ("error"), RELAY_PLUGIN_NAME,
|
|
(pos_error) ? pos_error : value);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on option "relay.network.tls_priorities".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_network_tls_priorities (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
if (relay_network_init_ok && relay_gnutls_priority_cache)
|
|
{
|
|
gnutls_priority_deinit (*relay_gnutls_priority_cache);
|
|
relay_network_set_priority ();
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Check if option "relay.network.unix_socket_permissions" is valid.
|
|
*
|
|
* Return:
|
|
* 1: value is valid
|
|
* 0: value is not valid
|
|
*/
|
|
|
|
int
|
|
relay_config_check_network_unix_socket_permissions (const void *pointer, void *data,
|
|
struct t_config_option *option,
|
|
const char *value)
|
|
{
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
return value && (strlen (value) == 3) && weechat_util_parse_long (value, 8, NULL);
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on option "relay.network.websocket_allowed_origins".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_network_websocket_allowed_origins (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
const char *allowed_origins;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
if (relay_config_regex_websocket_allowed_origins)
|
|
{
|
|
regfree (relay_config_regex_websocket_allowed_origins);
|
|
free (relay_config_regex_websocket_allowed_origins);
|
|
relay_config_regex_websocket_allowed_origins = NULL;
|
|
}
|
|
|
|
allowed_origins = weechat_config_string (relay_config_network_websocket_allowed_origins);
|
|
if (allowed_origins && allowed_origins[0])
|
|
{
|
|
relay_config_regex_websocket_allowed_origins = malloc (sizeof (*relay_config_regex_websocket_allowed_origins));
|
|
if (relay_config_regex_websocket_allowed_origins)
|
|
{
|
|
if (weechat_string_regcomp (relay_config_regex_websocket_allowed_origins,
|
|
allowed_origins,
|
|
REG_EXTENDED | REG_ICASE) != 0)
|
|
{
|
|
free (relay_config_regex_websocket_allowed_origins);
|
|
relay_config_regex_websocket_allowed_origins = NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Check if IRC backlog tags are valid.
|
|
*
|
|
* Return:
|
|
* 1: IRC backlog tags are valid
|
|
* 0: IRC backlog tags are not valid
|
|
*/
|
|
|
|
int
|
|
relay_config_check_irc_backlog_tags (const void *pointer, void *data,
|
|
struct t_config_option *option,
|
|
const char *value)
|
|
{
|
|
char **tags;
|
|
int num_tags, i, rc;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
rc = 1;
|
|
|
|
/* "*" means all tags */
|
|
if (strcmp (value, "*") == 0)
|
|
return rc;
|
|
|
|
/* split tags and check them */
|
|
tags = weechat_string_split (value, ",", NULL,
|
|
WEECHAT_STRING_SPLIT_STRIP_LEFT
|
|
| WEECHAT_STRING_SPLIT_STRIP_RIGHT
|
|
| WEECHAT_STRING_SPLIT_COLLAPSE_SEPS,
|
|
0, &num_tags);
|
|
if (tags)
|
|
{
|
|
for (i = 0; i < num_tags; i++)
|
|
{
|
|
if (relay_irc_search_backlog_commands_tags (tags[i]) < 0)
|
|
{
|
|
rc = 0;
|
|
break;
|
|
}
|
|
}
|
|
weechat_string_free_split (tags);
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on option "relay.irc.backlog_tags".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_irc_backlog_tags (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
char **items;
|
|
int num_items, i;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
if (!relay_config_hashtable_irc_backlog_tags)
|
|
{
|
|
relay_config_hashtable_irc_backlog_tags = weechat_hashtable_new (
|
|
32,
|
|
WEECHAT_HASHTABLE_STRING,
|
|
WEECHAT_HASHTABLE_STRING,
|
|
NULL, NULL);
|
|
}
|
|
else
|
|
weechat_hashtable_remove_all (relay_config_hashtable_irc_backlog_tags);
|
|
|
|
items = weechat_string_split (
|
|
weechat_config_string (relay_config_irc_backlog_tags),
|
|
",",
|
|
NULL,
|
|
WEECHAT_STRING_SPLIT_STRIP_LEFT
|
|
| WEECHAT_STRING_SPLIT_STRIP_RIGHT
|
|
| WEECHAT_STRING_SPLIT_COLLAPSE_SEPS,
|
|
0,
|
|
&num_items);
|
|
if (items)
|
|
{
|
|
for (i = 0; i < num_items; i++)
|
|
{
|
|
weechat_hashtable_set (relay_config_hashtable_irc_backlog_tags,
|
|
items[i],
|
|
NULL);
|
|
}
|
|
weechat_string_free_split (items);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Check if a port is valid.
|
|
*
|
|
* Return:
|
|
* 1: port is valid
|
|
* 0: port is not valid
|
|
*/
|
|
|
|
int
|
|
relay_config_check_port_cb (const void *pointer, void *data,
|
|
struct t_config_option *option,
|
|
const char *value)
|
|
{
|
|
char *error;
|
|
long port;
|
|
struct t_relay_server *ptr_server;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
error = NULL;
|
|
port = strtol (value, &error, 10);
|
|
ptr_server = relay_server_search_port ((int)port);
|
|
if (ptr_server)
|
|
{
|
|
weechat_printf (NULL, _("%s%s: error: port \"%d\" is already used"),
|
|
weechat_prefix ("error"),
|
|
RELAY_PLUGIN_NAME, (int)port);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Check if a UNIX path is too long or empty.
|
|
*
|
|
* Return:
|
|
* 1: path is valid
|
|
* 0: path is empty or too long
|
|
*/
|
|
|
|
int
|
|
relay_config_check_path_length (const char *path)
|
|
{
|
|
struct sockaddr_un addr;
|
|
size_t length, max_length;
|
|
|
|
length = strlen (path);
|
|
if (length == 0)
|
|
{
|
|
weechat_printf (NULL, _("%s%s: error: path is empty"),
|
|
weechat_prefix ("error"), RELAY_PLUGIN_NAME);
|
|
return 0;
|
|
}
|
|
|
|
max_length = sizeof (addr.sun_path);
|
|
if (length + 1 > max_length)
|
|
{
|
|
weechat_printf (NULL,
|
|
_("%s%s: error: path \"%s\" too long (length: %d; max: %d)"),
|
|
weechat_prefix ("error"), RELAY_PLUGIN_NAME, path,
|
|
length, max_length);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Check if a UNIX path is available: it is available if not existing, or
|
|
* if a file of type socket already exists.
|
|
*
|
|
* Return:
|
|
* 0: path is available
|
|
* -1: path already exists and is not a socket
|
|
* -2: invalid path
|
|
*/
|
|
|
|
int
|
|
relay_config_check_path_available (const char *path)
|
|
{
|
|
struct stat buf;
|
|
int rc;
|
|
|
|
rc = stat (path, &buf);
|
|
|
|
/* OK if an existing file is a socket */
|
|
if ((rc == 0) && S_ISSOCK(buf.st_mode))
|
|
return 0;
|
|
|
|
/* error if an existing file is NOT a socket */
|
|
if (rc == 0)
|
|
return -1;
|
|
|
|
/* OK if the file does not exist */
|
|
if (errno == ENOENT)
|
|
return 0;
|
|
|
|
/* on any other error, the path it considered as not available */
|
|
return -2;
|
|
}
|
|
|
|
/*
|
|
* Check if a path is valid.
|
|
*
|
|
* Return:
|
|
* 1: path is valid
|
|
* 0: path is not valid
|
|
*/
|
|
|
|
int
|
|
relay_config_check_path_cb (const void *pointer, void *data,
|
|
struct t_config_option *option,
|
|
const char *value)
|
|
{
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
if (!relay_config_check_path_length (value))
|
|
return 0;
|
|
|
|
if (relay_server_search_path (value))
|
|
{
|
|
weechat_printf (NULL, _("%s%s: error: path \"%s\" is already used"),
|
|
weechat_prefix ("error"), RELAY_PLUGIN_NAME, value);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on options in section "path".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_path_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
struct t_relay_server *ptr_server;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
|
|
ptr_server = relay_server_search (weechat_config_option_get_pointer (option, "name"));
|
|
if (ptr_server)
|
|
{
|
|
relay_server_update_path (ptr_server,
|
|
(const char *)weechat_config_option_get_pointer (option, "value"));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback called when an option is deleted in section "path".
|
|
*/
|
|
|
|
void
|
|
relay_config_delete_path_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
struct t_relay_server *ptr_server;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
|
|
ptr_server = relay_server_search (weechat_config_option_get_pointer (option, "name"));
|
|
relay_server_free (ptr_server);
|
|
}
|
|
|
|
/*
|
|
* Callback for changes on options in section "port".
|
|
*/
|
|
|
|
void
|
|
relay_config_change_port_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
struct t_relay_server *ptr_server;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
|
|
ptr_server = relay_server_search (weechat_config_option_get_pointer (option, "name"));
|
|
if (ptr_server)
|
|
{
|
|
relay_server_update_port (ptr_server,
|
|
*((int *)weechat_config_option_get_pointer (option, "value")));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback called when an option is deleted in section "port".
|
|
*/
|
|
|
|
void
|
|
relay_config_delete_port_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
struct t_relay_server *ptr_server;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
|
|
ptr_server = relay_server_search (weechat_config_option_get_pointer (option, "name"));
|
|
relay_server_free (ptr_server);
|
|
}
|
|
|
|
/*
|
|
* Callback called when an option is created in section "port" or "path".
|
|
*/
|
|
|
|
int
|
|
relay_config_create_option_port_path (const void *pointer, void *data,
|
|
struct t_config_file *config_file,
|
|
struct t_config_section *section,
|
|
const char *option_name,
|
|
const char *value)
|
|
{
|
|
int rc, protocol_number, ipv4, ipv6, tls, unix_socket;
|
|
char *error, *protocol, *protocol_args;
|
|
long port;
|
|
struct t_relay_server *ptr_server;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
|
|
rc = WEECHAT_CONFIG_OPTION_SET_OK_SAME_VALUE;
|
|
|
|
protocol_number = -1;
|
|
port = -1;
|
|
|
|
relay_server_get_protocol_args (option_name, &ipv4, &ipv6, &tls,
|
|
&unix_socket, &protocol, &protocol_args);
|
|
|
|
if (rc != WEECHAT_CONFIG_OPTION_SET_ERROR)
|
|
{
|
|
if (protocol)
|
|
protocol_number = relay_protocol_search (protocol);
|
|
|
|
if (protocol_number < 0)
|
|
{
|
|
weechat_printf (NULL, _("%s%s: error: unknown protocol \"%s\""),
|
|
weechat_prefix ("error"),
|
|
RELAY_PLUGIN_NAME, protocol);
|
|
rc = WEECHAT_CONFIG_OPTION_SET_ERROR;
|
|
}
|
|
|
|
if ((protocol_number == RELAY_PROTOCOL_WEECHAT) && protocol_args)
|
|
{
|
|
weechat_printf (NULL, _("%s%s: error: name is not allowed for "
|
|
"protocol \"%s\""),
|
|
weechat_prefix ("error"),
|
|
RELAY_PLUGIN_NAME, protocol);
|
|
rc = WEECHAT_CONFIG_OPTION_SET_ERROR;
|
|
}
|
|
}
|
|
|
|
if (rc != WEECHAT_CONFIG_OPTION_SET_ERROR)
|
|
{
|
|
if (weechat_config_search_option (config_file, section, option_name))
|
|
{
|
|
weechat_printf (NULL, _("%s%s: error: relay for \"%s\" already exists"),
|
|
weechat_prefix ("error"),
|
|
RELAY_PLUGIN_NAME, option_name);
|
|
rc = WEECHAT_CONFIG_OPTION_SET_ERROR;
|
|
}
|
|
}
|
|
|
|
if (rc != WEECHAT_CONFIG_OPTION_SET_ERROR)
|
|
{
|
|
if (unix_socket)
|
|
{
|
|
ptr_server = relay_server_search_path (value);
|
|
}
|
|
else
|
|
{
|
|
error = NULL;
|
|
port = strtol (value, &error, 10);
|
|
ptr_server = relay_server_search_port ((int)port);
|
|
}
|
|
if (ptr_server)
|
|
{
|
|
if (unix_socket)
|
|
{
|
|
weechat_printf (NULL,
|
|
_("%s%s: error: path \"%s\" is already used"),
|
|
weechat_prefix ("error"), RELAY_PLUGIN_NAME,
|
|
value);
|
|
}
|
|
else
|
|
{
|
|
weechat_printf (NULL,
|
|
_("%s%s: error: port \"%d\" is already used"),
|
|
weechat_prefix ("error"), RELAY_PLUGIN_NAME,
|
|
(int)port);
|
|
}
|
|
rc = WEECHAT_CONFIG_OPTION_SET_ERROR;
|
|
}
|
|
}
|
|
|
|
if (rc != WEECHAT_CONFIG_OPTION_SET_ERROR)
|
|
{
|
|
if (relay_server_new (option_name, protocol_number, protocol_args,
|
|
port, value, ipv4, ipv6, tls, unix_socket))
|
|
{
|
|
/* create configuration option */
|
|
if (unix_socket)
|
|
{
|
|
weechat_config_new_option (
|
|
config_file, section,
|
|
option_name, "string",
|
|
_("path to a socket file "
|
|
"(path is evaluated, see function string_eval_path_home "
|
|
"in plugin API reference)"),
|
|
NULL, 0, 0, "", value, 0,
|
|
&relay_config_check_path_cb, NULL, NULL,
|
|
&relay_config_change_path_cb, NULL, NULL,
|
|
&relay_config_delete_path_cb, NULL, NULL);
|
|
}
|
|
else
|
|
{
|
|
weechat_config_new_option (
|
|
config_file, section,
|
|
option_name, "integer",
|
|
_("port for relay"),
|
|
NULL, 0, 65535, "", value, 0,
|
|
&relay_config_check_port_cb, NULL, NULL,
|
|
&relay_config_change_port_cb, NULL, NULL,
|
|
&relay_config_delete_port_cb, NULL, NULL);
|
|
}
|
|
rc = WEECHAT_CONFIG_OPTION_SET_OK_SAME_VALUE;
|
|
}
|
|
else
|
|
rc = WEECHAT_CONFIG_OPTION_SET_ERROR;
|
|
}
|
|
|
|
free (protocol);
|
|
free (protocol_args);
|
|
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* Get remote pointer with name of option.
|
|
*/
|
|
|
|
struct t_relay_remote *
|
|
relay_config_get_remote_from_option_name (const char *name)
|
|
{
|
|
struct t_relay_remote *ptr_remote;
|
|
char *pos_option, *remote_name;
|
|
|
|
ptr_remote = NULL;
|
|
|
|
if (name)
|
|
{
|
|
pos_option = strrchr (name, '.');
|
|
if (pos_option)
|
|
{
|
|
remote_name = weechat_strndup (name, pos_option - name);
|
|
if (remote_name)
|
|
{
|
|
ptr_remote = relay_remote_search (remote_name);
|
|
free (remote_name);
|
|
}
|
|
}
|
|
}
|
|
|
|
return ptr_remote;
|
|
}
|
|
|
|
/*
|
|
* Callback called to check a server option when it is modified.
|
|
*/
|
|
|
|
int
|
|
relay_config_remote_url_check_value_cb (const void *pointer, void *data,
|
|
struct t_config_option *option,
|
|
const char *value)
|
|
{
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) option;
|
|
|
|
return relay_remote_url_valid (value);
|
|
}
|
|
|
|
/*
|
|
* Callback called when a remote URL option is modified.
|
|
*/
|
|
|
|
void
|
|
relay_config_remote_url_change_cb (const void *pointer, void *data,
|
|
struct t_config_option *option)
|
|
{
|
|
struct t_relay_remote *ptr_remote;
|
|
char *name;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
|
|
name = weechat_config_option_get_pointer (option, "name");
|
|
ptr_remote = relay_config_get_remote_from_option_name (name);
|
|
if (ptr_remote)
|
|
relay_remote_set_url (ptr_remote, weechat_config_string (option));
|
|
}
|
|
|
|
/*
|
|
* Create an option for a remote.
|
|
*
|
|
* Return pointer to new option, NULL if error.
|
|
*/
|
|
|
|
struct t_config_option *
|
|
relay_config_create_remote_option (const char *remote_name, int index_option,
|
|
const char *value)
|
|
{
|
|
struct t_config_option *ptr_option;
|
|
char *option_name;
|
|
|
|
ptr_option = NULL;
|
|
|
|
if (weechat_asprintf (&option_name,
|
|
"%s.%s",
|
|
remote_name,
|
|
relay_remote_option_string[index_option]) < 0)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
switch (index_option)
|
|
{
|
|
case RELAY_REMOTE_OPTION_URL:
|
|
ptr_option = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_remote,
|
|
option_name, "string",
|
|
N_("remote relay URL with optional port (default is 9000), "
|
|
"examples: https://example.com:9000 or http://example.com:9000 "
|
|
"(plain-text connection, not recommended)"),
|
|
NULL, 0, 0, value, NULL, 0,
|
|
&relay_config_remote_url_check_value_cb, NULL, NULL,
|
|
&relay_config_remote_url_change_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
break;
|
|
case RELAY_REMOTE_OPTION_AUTOCONNECT:
|
|
ptr_option = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_remote,
|
|
option_name, "boolean",
|
|
N_("automatically connect to the remote relay"),
|
|
NULL, 0, 0, value, NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
break;
|
|
case RELAY_REMOTE_OPTION_AUTORECONNECT_DELAY:
|
|
ptr_option = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_remote,
|
|
option_name, "integer",
|
|
N_("automatically reconnect to the remote relay after this delay, "
|
|
"in seconds (0 = disable automatic reconnection)"),
|
|
NULL, 0, 65535, value, NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
break;
|
|
case RELAY_REMOTE_OPTION_PROXY:
|
|
ptr_option = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_remote,
|
|
option_name, "string",
|
|
N_("name of proxy used for this remote relay (optional, proxy "
|
|
"must be defined with command /proxy)"),
|
|
NULL, 0, 0, value, NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
break;
|
|
case RELAY_REMOTE_OPTION_TLS_VERIFY:
|
|
ptr_option = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_remote,
|
|
option_name, "boolean",
|
|
N_("check that the TLS connection is fully trusted"),
|
|
NULL, 0, 0, value, NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
break;
|
|
case RELAY_REMOTE_OPTION_PASSWORD:
|
|
ptr_option = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_remote,
|
|
option_name, "string",
|
|
N_("password for remote relay "
|
|
"(note: content is evaluated, see /help eval)"),
|
|
NULL, 0, 0, value, NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
break;
|
|
case RELAY_REMOTE_OPTION_TOTP_SECRET:
|
|
ptr_option = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_remote,
|
|
option_name, "string",
|
|
N_("TOTP secret, encoded in base32 "
|
|
"(note: content is evaluated, see /help eval)"),
|
|
NULL, 0, 0, value, NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
break;
|
|
case RELAY_REMOTE_NUM_OPTIONS:
|
|
break;
|
|
}
|
|
|
|
free (option_name);
|
|
|
|
return ptr_option;
|
|
}
|
|
|
|
/*
|
|
* Create option for a temporary remote (when reading configuration file).
|
|
*/
|
|
|
|
void
|
|
relay_config_create_option_temp (struct t_relay_remote *temp_remote,
|
|
int index_option, const char *value)
|
|
{
|
|
struct t_config_option *new_option;
|
|
|
|
new_option = relay_config_create_remote_option (temp_remote->name,
|
|
index_option, value);
|
|
if (new_option
|
|
&& (index_option >= 0) && (index_option < RELAY_REMOTE_NUM_OPTIONS))
|
|
{
|
|
temp_remote->options[index_option] = new_option;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Use temporary remotes (created by reading configuration file).
|
|
*/
|
|
|
|
void
|
|
relay_config_use_temp_remotes (void)
|
|
{
|
|
struct t_relay_remote *ptr_temp_remote, *next_temp_remote;
|
|
int i, num_options_ok;
|
|
|
|
for (ptr_temp_remote = relay_remotes_temp; ptr_temp_remote;
|
|
ptr_temp_remote = ptr_temp_remote->next_remote)
|
|
{
|
|
num_options_ok = 0;
|
|
for (i = 0; i < RELAY_REMOTE_NUM_OPTIONS; i++)
|
|
{
|
|
if (!ptr_temp_remote->options[i])
|
|
{
|
|
ptr_temp_remote->options[i] =
|
|
relay_config_create_remote_option (
|
|
ptr_temp_remote->name,
|
|
i,
|
|
relay_remote_option_default[i]);
|
|
}
|
|
if (ptr_temp_remote->options[i])
|
|
num_options_ok++;
|
|
}
|
|
|
|
if (num_options_ok == RELAY_REMOTE_NUM_OPTIONS)
|
|
{
|
|
relay_remote_new_with_options (ptr_temp_remote->name,
|
|
ptr_temp_remote->options);
|
|
}
|
|
else
|
|
{
|
|
for (i = 0; i < RELAY_REMOTE_NUM_OPTIONS; i++)
|
|
{
|
|
if (ptr_temp_remote->options[i])
|
|
{
|
|
weechat_config_option_free (ptr_temp_remote->options[i]);
|
|
ptr_temp_remote->options[i] = NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* free all temporary remotes */
|
|
while (relay_remotes_temp)
|
|
{
|
|
next_temp_remote = relay_remotes_temp->next_remote;
|
|
|
|
free (relay_remotes_temp->name);
|
|
free (relay_remotes_temp);
|
|
|
|
relay_remotes_temp = next_temp_remote;
|
|
}
|
|
last_relay_remote_temp = NULL;
|
|
}
|
|
|
|
/*
|
|
* Read a remote option in relay configuration file.
|
|
*/
|
|
|
|
int
|
|
relay_config_remote_read_cb (const void *pointer, void *data,
|
|
struct t_config_file *config_file,
|
|
struct t_config_section *section,
|
|
const char *option_name, const char *value)
|
|
{
|
|
char *pos_option, *remote_name;
|
|
struct t_relay_remote *ptr_temp_remote;
|
|
int index_option;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) config_file;
|
|
(void) section;
|
|
|
|
if (!option_name)
|
|
return WEECHAT_CONFIG_OPTION_SET_OK_SAME_VALUE;
|
|
|
|
pos_option = strchr (option_name, '.');
|
|
if (!pos_option)
|
|
return WEECHAT_CONFIG_OPTION_SET_OK_SAME_VALUE;
|
|
|
|
remote_name = weechat_strndup (option_name, pos_option - option_name);
|
|
if (!remote_name)
|
|
return WEECHAT_CONFIG_OPTION_SET_OK_SAME_VALUE;
|
|
|
|
pos_option++;
|
|
|
|
/* search temporary remote */
|
|
for (ptr_temp_remote = relay_remotes_temp; ptr_temp_remote;
|
|
ptr_temp_remote = ptr_temp_remote->next_remote)
|
|
{
|
|
if (strcmp (ptr_temp_remote->name, remote_name) == 0)
|
|
break;
|
|
}
|
|
if (!ptr_temp_remote)
|
|
{
|
|
/* create new temporary remote */
|
|
ptr_temp_remote = relay_remote_alloc (remote_name);
|
|
if (ptr_temp_remote)
|
|
relay_remote_add (ptr_temp_remote, &relay_remotes_temp, &last_relay_remote_temp);
|
|
}
|
|
|
|
if (ptr_temp_remote)
|
|
{
|
|
index_option = relay_remote_search_option (pos_option);
|
|
if (index_option >= 0)
|
|
{
|
|
relay_config_create_option_temp (ptr_temp_remote, index_option,
|
|
value);
|
|
}
|
|
else
|
|
{
|
|
weechat_printf (NULL,
|
|
_("%sWarning: unknown option for section \"%s\": "
|
|
"%s (value: \"%s\")"),
|
|
weechat_prefix ("error"),
|
|
RELAY_CONFIG_SECTION_REMOTE,
|
|
option_name, value);
|
|
}
|
|
}
|
|
|
|
free (remote_name);
|
|
|
|
return WEECHAT_CONFIG_OPTION_SET_OK_SAME_VALUE;
|
|
}
|
|
|
|
/*
|
|
* Reload relay configuration file.
|
|
*/
|
|
|
|
int
|
|
relay_config_reload (const void *pointer, void *data,
|
|
struct t_config_file *config_file)
|
|
{
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
|
|
weechat_config_section_free_options (relay_config_section_port);
|
|
relay_server_free_all ();
|
|
|
|
return weechat_config_reload (config_file);
|
|
}
|
|
|
|
/*
|
|
* Update options in configuration file while reading the file.
|
|
*/
|
|
|
|
struct t_hashtable *
|
|
relay_config_update_cb (const void *pointer, void *data,
|
|
struct t_config_file *config_file,
|
|
int version_read,
|
|
struct t_hashtable *data_read)
|
|
{
|
|
const char *ptr_section, *ptr_option;
|
|
char *new_option, *pos;
|
|
int changes;
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
(void) config_file;
|
|
|
|
/* nothing to do if the config file is already up-to-date */
|
|
if (version_read >= RELAY_CONFIG_VERSION)
|
|
return NULL;
|
|
|
|
changes = 0;
|
|
|
|
if (version_read < 2)
|
|
{
|
|
/*
|
|
* changes in v2 (WeeChat 4.0.0):
|
|
* - options "ssl*" renamed to "tls*"
|
|
* - protocol "ssl" renamed to "tls" in port/path sections
|
|
*/
|
|
ptr_section = weechat_hashtable_get (data_read, "section");
|
|
ptr_option = weechat_hashtable_get (data_read, "option");
|
|
if (ptr_section
|
|
&& ptr_option
|
|
&& (strcmp (ptr_section, "network") == 0))
|
|
{
|
|
if (strncmp (ptr_option, "ssl", 3) == 0)
|
|
{
|
|
new_option = strdup (ptr_option);
|
|
if (new_option)
|
|
{
|
|
memcpy (new_option, "tls", 3);
|
|
weechat_printf (
|
|
NULL,
|
|
_("Relay option renamed: \"relay.network.%s\" => "
|
|
"\"relay.network.%s\""),
|
|
ptr_option, new_option);
|
|
weechat_hashtable_set (data_read, "option", new_option);
|
|
changes++;
|
|
free (new_option);
|
|
}
|
|
}
|
|
}
|
|
else if (ptr_section
|
|
&& ptr_option
|
|
&& ((strcmp (ptr_section, "port") == 0)
|
|
|| (strcmp (ptr_section, "path") == 0)))
|
|
{
|
|
new_option = strdup (ptr_option);
|
|
if (new_option)
|
|
{
|
|
pos = new_option;
|
|
while (1)
|
|
{
|
|
if (strncmp (pos, "ipv4.", 5) == 0)
|
|
{
|
|
pos += 5;
|
|
}
|
|
else if (strncmp (pos, "ipv6.", 5) == 0)
|
|
{
|
|
pos += 5;
|
|
}
|
|
else if (strncmp (pos, "ssl.", 4) == 0)
|
|
{
|
|
memcpy (pos, "tls", 3);
|
|
pos += 4;
|
|
}
|
|
else if (strncmp (pos, "unix.", 5) == 0)
|
|
{
|
|
pos += 5;
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
if (strcmp (ptr_option, new_option) != 0)
|
|
{
|
|
weechat_printf (
|
|
NULL,
|
|
_("Relay option renamed: "
|
|
"\"relay.%s.%s\" => \"relay.%s.%s\""),
|
|
ptr_section, ptr_option,
|
|
ptr_section, new_option);
|
|
weechat_hashtable_set (data_read, "option", new_option);
|
|
changes++;
|
|
}
|
|
free (new_option);
|
|
}
|
|
}
|
|
}
|
|
|
|
return (changes) ? data_read : NULL;
|
|
}
|
|
|
|
/*
|
|
* Initialize relay configuration file.
|
|
*
|
|
* Return:
|
|
* 1: OK
|
|
* 0: error
|
|
*/
|
|
|
|
int
|
|
relay_config_init (void)
|
|
{
|
|
relay_config_file = weechat_config_new (RELAY_CONFIG_PRIO_NAME,
|
|
&relay_config_reload, NULL, NULL);
|
|
if (!relay_config_file)
|
|
return 0;
|
|
|
|
if (!weechat_config_set_version (relay_config_file, RELAY_CONFIG_VERSION,
|
|
&relay_config_update_cb, NULL, NULL))
|
|
{
|
|
weechat_config_free (relay_config_file);
|
|
relay_config_file = NULL;
|
|
return 0;
|
|
}
|
|
|
|
/* section look */
|
|
relay_config_section_look = weechat_config_new_section (
|
|
relay_config_file, "look",
|
|
0, 0,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
if (relay_config_section_look)
|
|
{
|
|
relay_config_look_auto_open_buffer = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_look,
|
|
"auto_open_buffer", "string",
|
|
N_("auto open relay buffer when a new client is connecting "
|
|
"using one of these protocols (comma-separated list); "
|
|
"allowed protocols: \"api\", \"irc\", \"weechat\""),
|
|
NULL, 0, 0, "irc,weechat", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_change_auto_open_buffer_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_look_display_clients = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_look,
|
|
"display_clients", "string",
|
|
N_("display messages when clients connect/disconnect from relay "
|
|
"using one of these protocols (comma-separated list); "
|
|
"allowed protocols: \"api\", \"irc\", \"weechat\""),
|
|
NULL, 0, 0, "irc,weechat", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_change_display_clients_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_look_raw_messages = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_look,
|
|
"raw_messages", "integer",
|
|
N_("number of raw messages to save in memory when raw data buffer "
|
|
"is closed (messages will be displayed when opening raw data "
|
|
"buffer)"),
|
|
NULL, 0, 65535, "256", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_look_raw_messages_max_length = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_look,
|
|
"raw_messages_max_length", "integer",
|
|
N_("max number of chars to display in raw messages (very long "
|
|
"messages can cause slowness); 0 = display whole messages"),
|
|
NULL, 0, INT_MAX, "4096", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
}
|
|
|
|
/* section color */
|
|
relay_config_section_color = weechat_config_new_section (
|
|
relay_config_file, "color",
|
|
0, 0,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
if (relay_config_section_color)
|
|
{
|
|
relay_config_color_client = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_color,
|
|
"client", "color",
|
|
N_("text color for client description"),
|
|
NULL, 0, 0, "cyan", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_color_status[RELAY_STATUS_CONNECTED] = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_color,
|
|
"status_active", "color",
|
|
N_("text color for \"connected\" status"),
|
|
NULL, 0, 0, "green", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_refresh_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_color_status[RELAY_STATUS_AUTH_FAILED] = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_color,
|
|
"status_auth_failed", "color",
|
|
N_("text color for \"authentication failed\" status"),
|
|
NULL, 0, 0, "lightmagenta", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_refresh_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_color_status[RELAY_STATUS_CONNECTING] = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_color,
|
|
"status_connecting", "color",
|
|
N_("text color for \"connecting\" status"),
|
|
NULL, 0, 0, "white", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_refresh_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_color_status[RELAY_STATUS_DISCONNECTED] = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_color,
|
|
"status_disconnected", "color",
|
|
N_("text color for \"disconnected\" status"),
|
|
NULL, 0, 0, "lightred", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_refresh_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_color_status[RELAY_STATUS_AUTHENTICATING] = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_color,
|
|
"status_authenticating", "color",
|
|
N_("text color for \"authenticating\" status"),
|
|
NULL, 0, 0, "yellow", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_refresh_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_color_text = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_color,
|
|
"text", "color",
|
|
N_("text color in relay buffer"),
|
|
NULL, 0, 0, "default", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_refresh_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_color_text_bg = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_color,
|
|
"text_bg", "color",
|
|
N_("background color in relay buffer"),
|
|
NULL, 0, 0, "default", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_refresh_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_color_text_selected = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_color,
|
|
"text_selected", "color",
|
|
N_("text color of selected line in relay buffer"),
|
|
NULL, 0, 0, "white", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_refresh_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
}
|
|
|
|
/* section network */
|
|
relay_config_section_network = weechat_config_new_section (
|
|
relay_config_file, "network",
|
|
0, 0,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
if (relay_config_section_network)
|
|
{
|
|
relay_config_network_allow_empty_password = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"allow_empty_password", "boolean",
|
|
N_("allow empty password in relay (it should be enabled only for "
|
|
"tests or local network)"),
|
|
NULL, 0, 0, "off", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_allowed_ips = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"allowed_ips", "string",
|
|
N_("POSIX extended regular expression with IPs allowed to use relay "
|
|
"(case-insensitive, use \"(?-i)\" at beginning to make it "
|
|
"case-sensitive), example: "
|
|
"\"^(123\\.45\\.67\\.89|192\\.160\\..*)$\""),
|
|
NULL, 0, 0, "", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_change_network_allowed_ips, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_auth_timeout = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"auth_timeout", "integer",
|
|
N_("timeout (in seconds) for client authentication: connection is "
|
|
"closed if the client is still not authenticated after this "
|
|
"delay and the client status is set to \"authentication "
|
|
"failed\" (0 = wait forever)"),
|
|
NULL, 0, INT_MAX, "60", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_bind_address = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"bind_address", "string",
|
|
N_("address for bind (if empty, connection is possible on all "
|
|
"interfaces, use \"127.0.0.1\" to allow connections from "
|
|
"local machine only with IPv4 and \"::ffff:127.0.0.1\" with IPv6)"),
|
|
NULL, 0, 0, "", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_change_network_bind_address_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_clients_purge_delay = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"clients_purge_delay", "integer",
|
|
N_("delay for purging disconnected clients (in minutes, 0 = purge "
|
|
"clients immediately, -1 = never purge)"),
|
|
NULL, -1, 60 * 24 * 30, "0", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_commands = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"commands", "string",
|
|
N_("comma-separated list of commands allowed/denied when input "
|
|
"data (text or command) is received from a client (\"api\" and "
|
|
"\"weechat\" protocols); "
|
|
"\"*\" means any command, a name beginning with \"!\" is "
|
|
"a negative value to prevent a command from being executed, "
|
|
"wildcard \"*\" is allowed in names; this option should be set "
|
|
"if the relay client is not safe (someone could use it to run "
|
|
"commands); for example \"*,!exec,!quit\" allows any command "
|
|
"except /exec and /quit"),
|
|
NULL, 0, 0, "*,!quit", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_compression = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"compression", "integer",
|
|
N_("compression of messages sent to clients with \"api\" and "
|
|
"\"weechat\" protocols: 0 = disable compression, "
|
|
"1 = low compression / fast ... 100 = best compression / slow; "
|
|
"the value is a percentage converted to 1-9 for zlib and 1-19 "
|
|
"for zstd; the default value is recommended, it offers a good "
|
|
"compromise between compression and speed"),
|
|
NULL, 0, 100, "20", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_ipv6 = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"ipv6", "boolean",
|
|
N_("listen on IPv6 socket by default (in addition to IPv4 which is "
|
|
"default); protocols IPv4 and IPv6 can be forced (individually "
|
|
"or together) in the protocol name (see /help relay)"),
|
|
NULL, 0, 0, "on", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_change_network_ipv6_cb, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_max_clients = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"max_clients", "integer",
|
|
N_("maximum number of clients connecting to a port (0 = no limit)"),
|
|
NULL, 0, INT_MAX, "5", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_nonce_size = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"nonce_size", "integer",
|
|
N_("size of nonce (in bytes), generated when a client connects; "
|
|
"the client must use this nonce, concatenated to the client nonce "
|
|
"and the password when hashing the password in the \"init\" "
|
|
"command of the weechat protocol"),
|
|
NULL, 8, 128, "16", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_password = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"password", "string",
|
|
N_("password required by clients to access this relay (empty value "
|
|
"means no password required, see option "
|
|
"relay.network.allow_empty_password) (note: content is evaluated, "
|
|
"see /help eval)"),
|
|
NULL, 0, 0, "", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_password_hash_algo = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"password_hash_algo", "string",
|
|
N_("comma separated list of hash algorithms used for password "
|
|
"authentication in \"api\" and \"weechat\" protocols, among these "
|
|
"values: \"plain\" (password in plain text, not hashed), \"sha256\", "
|
|
"\"sha512\", \"pbkdf2+sha256\", \"pbkdf2+sha512\"), \"*\" means "
|
|
"all algorithms, a name beginning with \"!\" is a negative "
|
|
"value to prevent an algorithm from being used, wildcard \"*\" "
|
|
"is allowed in names (examples: \"*\", \"pbkdf2*\", "
|
|
"\"*,!plain\")"),
|
|
NULL, 0, 0, "*", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_change_network_password_hash_algo, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_password_hash_iterations = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"password_hash_iterations", "integer",
|
|
N_("number of iterations asked to the client in \"api\" and \"weechat\" "
|
|
"protocols when a hashed password with algorithm PBKDF2 is used for "
|
|
"authentication; more iterations is better in term of security "
|
|
"but is slower to compute; this number should not be too high "
|
|
"if your CPU is slow"),
|
|
NULL, 1, 1000000, "100000", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_time_window = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"time_window", "integer",
|
|
N_("number of seconds to allow before and after the current time "
|
|
"for the hash of time + password in \"api\" protocol"),
|
|
NULL, 0, 256, "5", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_tls_cert_key = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"tls_cert_key", "string",
|
|
N_("file with TLS certificate and private key (for serving clients "
|
|
"with TLS) "
|
|
"(path is evaluated, see function string_eval_path_home in "
|
|
"plugin API reference)"),
|
|
NULL, 0, 0, "${weechat_config_dir}/tls/relay.pem", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_change_network_tls_cert_key, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_tls_priorities = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"tls_priorities", "string",
|
|
N_("string with priorities for gnutls (for syntax, see "
|
|
"documentation of function gnutls_priority_init in gnutls "
|
|
"manual, common strings are: \"PERFORMANCE\", \"NORMAL\", "
|
|
"\"SECURE128\", \"SECURE256\", \"EXPORT\", \"NONE\")"),
|
|
NULL, 0, 0, "NORMAL", NULL, 0,
|
|
&relay_config_check_network_tls_priorities, NULL, NULL,
|
|
&relay_config_change_network_tls_priorities, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_totp_secret = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"totp_secret", "string",
|
|
N_("secret for the generation of the Time-based One-Time Password "
|
|
"(TOTP), encoded in base32 (only letters and digits from 2 to 7); "
|
|
"it is used as second factor in \"api\" and \"weechat\" protocols, "
|
|
"in addition to the password, which must not be empty "
|
|
"(empty value means no TOTP is required) "
|
|
"(note: content is evaluated, see /help eval)"),
|
|
NULL, 0, 0, "", NULL, 0,
|
|
&relay_config_check_network_totp_secret, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_totp_window = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"totp_window", "integer",
|
|
N_("number of Time-based One-Time Passwords to accept before and "
|
|
"after the current one: "
|
|
"0 = accept only the current password, "
|
|
"1 = accept one password before, the current, and one after, "
|
|
"2 = accept two passwords before, the current, and two after, "
|
|
"...; a high number reduces the security level "
|
|
"(0 or 1 are recommended values)"),
|
|
NULL, 0, 256, "0", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_network_unix_socket_permissions = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"unix_socket_permissions", "string",
|
|
N_("permissions for the Unix socket, as octal value (see man chmod); "
|
|
"it must be a number with 3 digits, each between 0 and 7"),
|
|
NULL, 0, 0, "700", NULL, 0,
|
|
&relay_config_check_network_unix_socket_permissions, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_websocket_allowed_origins = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"websocket_allowed_origins", "string",
|
|
N_("POSIX extended regular expression with origins allowed in "
|
|
"websockets (case-insensitive, use \"(?-i)\" at beginning to "
|
|
"make it case-sensitive), example: "
|
|
"\"^https?://(www\\.)?example\\.(com|org)\""),
|
|
NULL, 0, 0, "", NULL, 0,
|
|
NULL, NULL, NULL,
|
|
&relay_config_change_network_websocket_allowed_origins, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_network_websocket_permessage_deflate = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_network,
|
|
"websocket_permessage_deflate", "boolean",
|
|
N_("enable websocket extension \"permessage-deflate\" to compress "
|
|
"websocket frames (\"api\" protocol only); "
|
|
"if disabled, WeeChat (as server) will not enable "
|
|
"permessage-deflate even if the client supports it, and when "
|
|
"connecting to a remote WeeChat (api relay only), "
|
|
"permessage-deflate support is not advertised by WeeChat; "
|
|
"it is recommended to keep this option enabled, and you should "
|
|
"disable it only if you have troubles with this extension, "
|
|
"either with WeeChat or the client"),
|
|
NULL, 0, 100, "on", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
}
|
|
|
|
/* section irc */
|
|
relay_config_section_irc = weechat_config_new_section (
|
|
relay_config_file, "irc",
|
|
0, 0,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
if (relay_config_section_irc)
|
|
{
|
|
relay_config_irc_backlog_max_minutes = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_irc,
|
|
"backlog_max_minutes", "integer",
|
|
N_("maximum number of minutes in backlog per IRC channel "
|
|
"(0 = unlimited, examples: 1440 = one day, 10080 = one week, "
|
|
"43200 = one month, 525600 = one year)"),
|
|
NULL, 0, INT_MAX, "0", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_irc_backlog_max_number = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_irc,
|
|
"backlog_max_number", "integer",
|
|
N_("maximum number of lines in backlog per IRC channel "
|
|
"(0 = unlimited)"),
|
|
NULL, 0, INT_MAX, "1024", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_irc_backlog_since_last_disconnect = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_irc,
|
|
"backlog_since_last_disconnect", "boolean",
|
|
N_("display backlog starting from last client disconnect"),
|
|
NULL, 0, 0, "on", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_irc_backlog_since_last_message = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_irc,
|
|
"backlog_since_last_message", "boolean",
|
|
N_("display backlog starting from your last message"),
|
|
NULL, 0, 0, "off", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_irc_backlog_tags = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_irc,
|
|
"backlog_tags", "string",
|
|
N_("comma-separated list of messages tags which are displayed in "
|
|
"backlog per IRC channel (supported tags: \"irc_join\", "
|
|
"\"irc_part\", \"irc_quit\", \"irc_nick\", \"irc_privmsg\"), "
|
|
"\"*\" = all supported tags"),
|
|
NULL, 0, 0, "irc_privmsg", NULL, 0,
|
|
&relay_config_check_irc_backlog_tags, NULL, NULL,
|
|
&relay_config_change_irc_backlog_tags, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
relay_config_irc_backlog_time_format = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_irc,
|
|
"backlog_time_format", "string",
|
|
N_("format for time in backlog messages (see man strftime for "
|
|
"format) (not used if server capability \"server-time\" was "
|
|
"enabled by client, because time is sent as irc tag); empty "
|
|
"string = disable time in backlog messages"),
|
|
NULL, 0, 0, "[%H:%M] ", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
}
|
|
|
|
/* section api */
|
|
relay_config_section_api = weechat_config_new_section (
|
|
relay_config_file, "api",
|
|
0, 0,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
if (relay_config_section_api)
|
|
{
|
|
relay_config_api_remote_autoreconnect_delay_growing = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_api,
|
|
"remote_autoreconnect_delay_growing", "integer",
|
|
N_("growing factor for autoreconnect delay to remote relay (1 = always "
|
|
"same delay, 2 = delay*2 for each retry, etc.)"),
|
|
NULL, 1, 100, "2", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_api_remote_autoreconnect_delay_max = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_api,
|
|
"remote_autoreconnect_delay_max", "integer",
|
|
N_("maximum autoreconnect delay to remote relay (in seconds, 0 = no "
|
|
"maximum)"),
|
|
NULL, 0, 3600 * 24 * 7, "600", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_api_remote_get_lines = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_api,
|
|
"remote_get_lines", "integer",
|
|
N_("number of lines to retrieve on each buffer when connecting "
|
|
"to a remote relay"),
|
|
NULL, 0, INT_MAX, "1000", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_api_remote_input_cmd_local = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_api,
|
|
"remote_input_cmd_local", "string|themable",
|
|
N_("text displayed after user input when the command would be "
|
|
"executed locally (NOT sent to the remote WeeChat) "
|
|
"(note: content is evaluated, see /help eval)"),
|
|
NULL, 0, 0, " ${color:green}<local cmd>", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
relay_config_api_remote_input_cmd_remote = weechat_config_new_option (
|
|
relay_config_file, relay_config_section_api,
|
|
"remote_input_cmd_remote", "string|themable",
|
|
N_("text displayed after user input when the command would be "
|
|
"executed on the remote WeeChat (NOT executed locally) "
|
|
"(note: content is evaluated, see /help eval)"),
|
|
NULL, 0, 0, " ${color:red}<remote cmd>", NULL, 0,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
}
|
|
|
|
/* section port */
|
|
relay_config_section_port = weechat_config_new_section (
|
|
relay_config_file, "port",
|
|
1, 1,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
&relay_config_create_option_port_path, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
|
|
/* section path */
|
|
relay_config_section_path = weechat_config_new_section (
|
|
relay_config_file, "path",
|
|
1, 1,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
&relay_config_create_option_port_path, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
|
|
/* remote */
|
|
relay_config_section_remote = weechat_config_new_section (
|
|
relay_config_file,
|
|
RELAY_CONFIG_SECTION_REMOTE,
|
|
0, 0,
|
|
&relay_config_remote_read_cb, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL,
|
|
NULL, NULL, NULL);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Read relay configuration file.
|
|
*/
|
|
|
|
int
|
|
relay_config_read (void)
|
|
{
|
|
int rc;
|
|
|
|
rc = weechat_config_read (relay_config_file);
|
|
if (rc == WEECHAT_CONFIG_READ_OK)
|
|
{
|
|
relay_config_change_auto_open_buffer_cb (NULL, NULL, NULL);
|
|
relay_config_change_display_clients_cb (NULL, NULL, NULL);
|
|
relay_config_change_network_allowed_ips (NULL, NULL, NULL);
|
|
relay_config_change_network_password_hash_algo (NULL, NULL, NULL);
|
|
relay_config_change_irc_backlog_tags (NULL, NULL, NULL);
|
|
relay_config_use_temp_remotes ();
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* Write relay configuration file.
|
|
*/
|
|
|
|
int
|
|
relay_config_write (void)
|
|
{
|
|
return weechat_config_write (relay_config_file);
|
|
}
|
|
|
|
/*
|
|
* Free relay configuration.
|
|
*/
|
|
|
|
void
|
|
relay_config_free (void)
|
|
{
|
|
weechat_config_free (relay_config_file);
|
|
relay_config_file = NULL;
|
|
|
|
if (relay_config_regex_allowed_ips)
|
|
{
|
|
regfree (relay_config_regex_allowed_ips);
|
|
free (relay_config_regex_allowed_ips);
|
|
relay_config_regex_allowed_ips = NULL;
|
|
}
|
|
|
|
if (relay_config_regex_websocket_allowed_origins)
|
|
{
|
|
regfree (relay_config_regex_websocket_allowed_origins);
|
|
free (relay_config_regex_websocket_allowed_origins);
|
|
relay_config_regex_websocket_allowed_origins = NULL;
|
|
}
|
|
|
|
if (relay_config_hashtable_irc_backlog_tags)
|
|
{
|
|
weechat_hashtable_free (relay_config_hashtable_irc_backlog_tags);
|
|
relay_config_hashtable_irc_backlog_tags = NULL;
|
|
}
|
|
|
|
if (relay_config_network_password_hash_algo_list)
|
|
{
|
|
weechat_string_free_split (relay_config_network_password_hash_algo_list);
|
|
relay_config_network_password_hash_algo_list = NULL;
|
|
}
|
|
}
|