res_websocket_client: Create common utilities for websocket clients.

Since multiple Asterisk capabilities now need to create websocket clients
it makes sense to create a common set of utilities rather than making
each of those capabilities implement their own.

* A new configuration file "websocket_client.conf" is used to store common
client parameters in named configuration sections.
* APIs are provided to list and retrieve ast_websocket_client objects created
from the named configurations.
* An API is provided that accepts an ast_websocket_client object, connects
to the remote server with retries and returns an ast_websocket object. TLS is
supported as is basic authentication.
* An observer can be registered to receive notification of loaded or reloaded
client objects.
* An API is provided to compare an existing client object to one just
reloaded and return the fields that were changed. The caller can then decide
what action to take based on which fields changed.

Also as part of thie commit, several sorcery convenience macros were created
to make registering common object fields easier.

UserNote: A new module "res_websocket_client" and config file
"websocket_client.conf" have been added to support several upcoming new
capabilities that need common websocket client configuration.
This commit is contained in:
George Joseph
2025-05-02 08:52:54 -06:00
committed by github-actions[bot]
parent 3a5ffe2842
commit 5a3164c0b2
7 changed files with 944 additions and 0 deletions

View File

@@ -47,6 +47,26 @@
*
*/
/*! \brief WebSocket connection/configuration types.
*
* These may look like they overlap or are redundant, but
* they're shared by other modules like ari and chan_websocket
* and it didn't make sense to force them to define their
* own types.
*/
enum ast_websocket_type {
AST_WS_TYPE_CLIENT_PERSISTENT = (1 << 0),
AST_WS_TYPE_CLIENT_PER_CALL_CONFIG = (1 << 1),
AST_WS_TYPE_CLIENT_PER_CALL = (1 << 2),
AST_WS_TYPE_CLIENT = (1 << 3),
AST_WS_TYPE_INBOUND = (1 << 4),
AST_WS_TYPE_SERVER = (1 << 5),
AST_WS_TYPE_ANY = (0xFFFFFFFF),
};
const char *ast_websocket_type_to_str(enum ast_websocket_type type);
/*! \brief WebSocket operation codes */
enum ast_websocket_opcode {
AST_WEBSOCKET_OPCODE_TEXT = 0x1, /*!< Text frame */

View File

@@ -1623,6 +1623,135 @@ int ast_sorcery_is_object_field_registered(const struct ast_sorcery_object_type
*/
const char *ast_sorcery_get_module(const struct ast_sorcery *sorcery);
/*!
* \section AstSorceryConvenienceMacros Simple Sorcery Convenience Macros
*
* For simple scenarios, the following macros can be used to register
* common object fields. The only requirement is that your source code's
* definition of it's sorcery handle be named "sorcery".
*
* Example structure:
* \code
* struct my_sorcery_object {
* SORCERY_OBJECT(details);
* AST_DECLARE_STRING_FIELDS(
* AST_STRING_FIELD(mystring);
* );
* enum some_enum_type myenum;
* int myint;
* unsigned int myuint;
* int mybool;
* };
* \endcode
*
* Example object type registration:
* \code
* ast_sorcery_object_register(sorcery, "myobject", ...);
* \endcode
*/
/*!
* \brief Register a boolean field as type OPT_YESNO_T within an object.
* \param object The unquoted object type.
* \param structure The unquoted name of the structure that contains the field
* without the "struct" prefix.
* \param option The unquoted name of the option as it appears in the config file.
* \param field The unquoted name of the field in the structure.
* \param def_value The quoted default value of the field. Should be "yes" or "no"
*
* \code
* ast_sorcery_register_bool(myobject, my_sorcery_object, mybool, mybool, "yes");
* \endcode
*/
#define ast_sorcery_register_bool(object, structure, option, field, def_value) \
ast_sorcery_object_field_register(sorcery, #object, #option, \
def_value, OPT_YESNO_T, 1, \
FLDSET(struct structure, field))
/*!
* \internal
* \brief Stringify a value.
*
* Needed because the preprocessor doesn't evaluate macros before it stringifies them.
*/
#define _sorcery_stringify(val) #val
/*!
* \brief Register an int field as type OPT_INT_T within an object.
* \param object The unquoted object type.
* \param structure The unquoted name of the structure that contains the field
* without the "struct" prefix.
* \param option The unquoted name of the option as it appears in the config file.
* \param field The unquoted name of the field in the structure.
* \param def_value The unquoted default value of the field.
*
* \code
* ast_sorcery_register_int(myobject, my_sorcery_object, myint, myint, -32);
* \endcode
*/
#define ast_sorcery_register_int(object, structure, option, field, def_value) \
ast_sorcery_object_field_register(sorcery, #object, #option, \
_sorcery_stringify(def_value), OPT_INT_T, PARSE_IN_RANGE, \
FLDSET(struct structure, field), INT_MIN, INT_MAX)
/*!
* \brief Register an unsigned int field as type OPT_UINT_T within an object.
* \param object The unquoted object type.
* \param structure The unquoted name of the structure that contains the field
* without the "struct" prefix.
* \param option The unquoted name of the option as it appears in the config file.
* \param field The unquoted name of the field in the structure.
* \param def_value The unquoted default value of the field.
*
* \code
* ast_sorcery_register_uint(myobject, my_sorcery_object, myint, myint, 32);
* \endcode
*/
#define ast_sorcery_register_uint(object, structure, option, field, def_value) \
ast_sorcery_object_field_register(sorcery, #object, #option, \
_stringify(def_value), OPT_UINT_T, PARSE_IN_RANGE, \
FLDSET(struct structure, field), 0, UINT_MAX)
/*!
* \brief Register a stringfield field as type OPT_STRINGFIELD_T within an object.
* \param object The unquoted object type.
* \param structure The unquoted name of the structure that contains the field
* without the "struct" prefix.
* \param option The unquoted name of the option as it appears in the config file.
* \param field The unquoted name of the field in the structure.
* \param def_value The quoted default value of the field.
*
* \code
* ast_sorcery_register_sf(myobject, my_sorcery_object, mystring, mystring, "");
* \endcode
*/
#define ast_sorcery_register_sf(object, structure, option, field, def_value) \
ast_sorcery_object_field_register(sorcery, #object, #option, \
def_value, OPT_STRINGFIELD_T, 0, \
STRFLDSET(struct structure, field))
/*!
* \brief Register a custom field within an object.
* \param object The unquoted object type.
* \param structure The unquoted name of the structure that contains the field
* without the "struct" prefix.
* \param option The unquoted name of the option as it appears in the config file.
*
* \code
* ast_sorcery_register_cust(myobject, my_sorcery_object, mystring);
* \endcode
*
* \note
* You must have defined the following standard sorcery custom handler functions:
* \li myobject_mystring_from_str(const struct aco_option *opt, struct ast_variable *var, void *obj)
* \li myobject_mystring_to_str(const void *obj, const intptr_t *args, char **buf)
*/
#define ast_sorcery_register_cust(object, option, def_value) \
ast_sorcery_object_field_register_custom(sorcery, #object, #option, \
def_value, object ## _ ## option ## _from_str, \
object ## _ ## option ## _to_str, NULL, 0, 0)
#if defined(__cplusplus) || defined(c_plusplus)
}
#endif

View File

@@ -0,0 +1,146 @@
/*
* Asterisk -- An open source telephony toolkit.
*
* Copyright (C) 2025, Sangoma Technologies Corporation
*
* George Joseph <gjoseph@sangoma.com>
*
* See http://www.asterisk.org for more information about
* the Asterisk project. Please do not directly contact
* any of the maintainers of this project for assistance;
* the project provides a web site, mailing lists and IRC
* channels for your use.
*
* This program is free software, distributed under the terms of
* the GNU General Public License Version 2. See the LICENSE file
* at the top of the source tree.
*/
#ifndef _RES_WEBSOCKET_CLIENT_H
#define _RES_WEBSOCKET_CLIENT_H
#include "asterisk/http_websocket.h"
#include "asterisk/sorcery.h"
enum ast_ws_client_fields {
AST_WS_CLIENT_FIELD_NONE = 0,
AST_WS_CLIENT_FIELD_URI = (1 << 0),
AST_WS_CLIENT_FIELD_PROTOCOLS = (1 << 1),
AST_WS_CLIENT_FIELD_USERNAME = (1 << 3),
AST_WS_CLIENT_FIELD_PASSWORD = (1 << 4),
AST_WS_CLIENT_FIELD_TLS_ENABLED = (1 << 7),
AST_WS_CLIENT_FIELD_CA_LIST_FILE = (1 << 8),
AST_WS_CLIENT_FIELD_CA_LIST_PATH = (1 << 9),
AST_WS_CLIENT_FIELD_CERT_FILE = (1 << 10),
AST_WS_CLIENT_FIELD_PRIV_KEY_FILE = (1 << 11),
AST_WS_CLIENT_FIELD_CONNECTION_TYPE = (1 << 13),
AST_WS_CLIENT_FIELD_RECONNECT_INTERVAL = (1 << 14),
AST_WS_CLIENT_FIELD_RECONNECT_ATTEMPTS = (1 << 15),
AST_WS_CLIENT_FIELD_CONNECTION_TIMEOUT = (1 << 16),
AST_WS_CLIENT_FIELD_VERIFY_SERVER_CERT = (1 << 17),
AST_WS_CLIENT_FIELD_VERIFY_SERVER_HOSTNAME = (1 << 18),
AST_WS_CLIENT_NEEDS_RECONNECT = AST_WS_CLIENT_FIELD_URI | AST_WS_CLIENT_FIELD_PROTOCOLS
| AST_WS_CLIENT_FIELD_CONNECTION_TYPE
| AST_WS_CLIENT_FIELD_USERNAME | AST_WS_CLIENT_FIELD_PASSWORD
| AST_WS_CLIENT_FIELD_TLS_ENABLED | AST_WS_CLIENT_FIELD_CA_LIST_FILE
| AST_WS_CLIENT_FIELD_CA_LIST_PATH | AST_WS_CLIENT_FIELD_CERT_FILE
| AST_WS_CLIENT_FIELD_PRIV_KEY_FILE | AST_WS_CLIENT_FIELD_VERIFY_SERVER_CERT
| AST_WS_CLIENT_FIELD_VERIFY_SERVER_HOSTNAME,
};
/*
* The first 23 fields are reserved for the websocket client core.
*/
#define AST_WS_CLIENT_FIELD_USER_START 24
struct ast_websocket_client {
SORCERY_OBJECT(details);
AST_DECLARE_STRING_FIELDS(
AST_STRING_FIELD(uri); /*!< Server URI */
AST_STRING_FIELD(protocols); /*!< Websocket protocols to use with server */
AST_STRING_FIELD(username); /*!< Auth user name */
AST_STRING_FIELD(password); /*!< Auth password */
AST_STRING_FIELD(ca_list_file); /*!< CA file */
AST_STRING_FIELD(ca_list_path); /*!< CA path */
AST_STRING_FIELD(cert_file); /*!< Certificate file */
AST_STRING_FIELD(priv_key_file); /*!< Private key file */
);
int invalid; /*!< Invalid configuration */
enum ast_ws_client_fields invalid_fields; /*!< Invalid fields */
enum ast_websocket_type connection_type; /*!< Connection type */
int connect_timeout; /*!< Connection timeout (ms) */
unsigned int reconnect_attempts; /*!< How many attempts before returning an error */
unsigned int reconnect_interval; /*!< How often to attempt a reconnect (ms) */
int tls_enabled; /*!< TLS enabled */
int verify_server_cert; /*!< Verify server certificate */
int verify_server_hostname; /*!< Verify server hostname */
};
/*!
* \brief Retrieve a container of all websocket client objects.
*
* \return The container. It may be empty but must always be cleaned up by the caller.
*/
struct ao2_container *ast_websocket_client_retrieve_all(void);
/*!
* \brief Retrieve a websocket client object by ID.
*
* \param id The ID of the websocket client object.
* \return The websocket client ao2 object or NULL if not found. The reference
* must be cleaned up by the caller.
*/
struct ast_websocket_client *ast_websocket_client_retrieve_by_id(const char *id);
/*!
* \brief Detect changes between two websocket client configurations.
*
* \param old_ow The old websocket configuration.
* \param new_ow The new websocket configuration.
* \return A bitmask of changed fields.
*/
enum ast_ws_client_fields ast_websocket_client_get_field_diff(
struct ast_websocket_client *old_wc,
struct ast_websocket_client *new_wc);
/*!
* \brief Add sorcery observers for websocket client events.
*
* \param callbacks The observer callbacks to add.
* \return 0 on success, -1 on failure.
*/
int ast_websocket_client_observer_add(
const struct ast_sorcery_observer *callbacks);
/*!
* \brief Remove sorcery observers for websocket client events.
*
* \param callbacks The observer callbacks to remove.
*/
void ast_websocket_client_observer_remove(
const struct ast_sorcery_observer *callbacks);
/*!
* \brief Connect to a websocket server using the configured authentication,
* retry and TLS options.
*
* \param wc A pointer to the ast_websocket_structure
* \param lock_obj A pointer to an ao2 object to lock while the
* connection is being attempted or NULL if no locking is needed.
* \param display_name An id string to use for logging messages.
* If NULL or empty the connection's ID will be used.
* \param result A pointer to an enum ast_websocket_result to store the
* result of the connection attempt.
*
* \return A pointer to the ast_websocket structure on success, or NULL on failure.
*/
struct ast_websocket *ast_websocket_client_connect(struct ast_websocket_client *wc,
void *lock_obj, const char *display_name, enum ast_websocket_result *result);
/*!
* \brief Force res_websocket_client to reload its configuration.
* \return 0 on success, -1 on failure.
*/
int ast_websocket_client_reload(void);
#endif /* _RES_WEBSOCKET_CLIENT_H */