/* * Copyright (c) 2018 MariaDB Corporation Ab * * Use of this software is governed by the Business Source License included * in the LICENSE.TXT file and at www.mariadb.com/bsl11. * * Change Date: 2025-10-29 * * On the date above, in accordance with the Business Source License, use * of this software will be governed by version 2 or later of the General * Public License. */ #pragma once #include #include #include #include #include #include "filter.hh" namespace maxscale { class Monitor; } /** * @file service.h - MaxScale internal service functions */ struct LastUserLoad { time_t last = 0; // The last time the users were loaded bool warned = false; // Has a warning been logged }; // The internal service representation class Service : public SERVICE { public: using FilterList = std::vector; using RateLimits = std::vector; Service(const std::string& name, const std::string& router, MXS_CONFIG_PARAMETER* params); ~Service(); /** * Check if name matches a basic service parameter * * Basic parameters are common to all services. These include, for example, the * `user` and `password` parameters. * * @return True if the parameter is a basic service parameter */ bool is_basic_parameter(const std::string& name); /** * Update a basic service parameter * * Update a parameter that is common to all services. * * @param name Name of the parameter to update * @param value The new value of the parameter */ void update_basic_parameter(const std::string& name, const std::string& value); /** * Set the list of filters for this service * * @param filters Filters to set * * @return True if filters were all found and were valid */ bool set_filters(const std::vector& filters); /** * Get the list of filters this service uses * * @note This can lock the service if this is the first time this worker * accesses the filter list * * @return A list of filters or an empty list of no filters are in use */ const FilterList& get_filters() const; /** * Reload users for all listeners * * @return True if loading of users was successful */ bool refresh_users(); /** * Unconditionally refreshes the users for all listeners of the service */ void load_users(); /** * Dump service configuration into a file * * @param filename File where the configuration should be written * * @return True on success */ bool dump_config(const char* filename) const; // TODO: Make JSON output internal (could iterate over get_filters() but that takes the service lock) json_t* json_relationships(const char* host) const; // TODO: Make these private mutable std::mutex lock; // TODO: Make this private. mxs::Monitor* m_monitor {nullptr}; /**< A possibly associated monitor */ bool uses_cluster() const { return m_monitor != nullptr; } private: FilterList m_filters; /**< Ordered list of filters */ std::string m_user; /**< Username */ std::string m_password; /**< Password */ std::string m_weightby; /**< Weighting parameter name */ std::string m_version_string; /**< Version string sent to clients */ RateLimits m_rate_limits; /**< The refresh rate limits for users of each thread */ uint64_t m_wkey; /**< Key for worker local data */ // Get the worker local filter list FilterList* get_local_filters() const; // Update the local filter list on the current worker void update_local_filters(); // Load the users bool do_load_users(); // Callback for updating the local filter list static void update_filters_cb(void* data) { Service* service = static_cast(data); service->update_local_filters(); } }; /** * Service life cycle management * * These functions should only be called by the MaxScale core. */ /** * @brief Allocate a new service * * @param name The service name * @param router The router module this service uses * @param params Service parameters * * @return The newly created service or NULL if an error occurred */ Service* service_alloc(const char* name, const char* router, MXS_CONFIG_PARAMETER* params); /** * Free a service * * @note Must not be called if the service has any active client connections or * active listeners * * @param service Service to free */ void service_free(Service* service); /** * Mark a service for destruction * * Once the service reference count drops down to zero, the service is destroyed. * * @param service Service to destroy */ void service_destroy(Service* service); /** * Check whether a service can be destroyed * * @param service Service to check * * @return True if service can be destroyed */ bool service_can_be_destroyed(Service* service); /** * @brief Shut all services down * * Turns on the shutdown flag in each service. This should be done as * part of the MaxScale shutdown. */ void service_shutdown(void); /** * @brief Destroy all service router and filter instances * * Calls the @c destroyInstance entry point of each service' router and * filters. This should be done after all worker threads have exited. */ void service_destroy_instances(void); /** * @brief Launch all services * * Initialize and start all services. This should only be called once by the * main initialization code. * * @return False if a fatal error occurred */ bool service_launch_all(void); /** * @brief Remove a listener from use * * @note This does not free the memory * * @param service Service that owns the listener * @param char Name of the listener to remove * * @return True if listener was found and removed */ bool service_remove_listener(Service* service, const char* target); void serviceRemoveBackend(Service* service, const SERVER* server); /** * @brief Serialize a service to a file * * This converts @c service into an INI format file. * * NOTE: This does not persist the complete service configuration and requires * that an existing service configuration is in the main configuration file. * * @param service Service to serialize * @return False if the serialization of the service fails, true if it was successful */ bool service_serialize(const Service* service); /** * Internal utility functions */ bool service_all_services_have_listeners(void); bool service_isvalid(Service* service); /** * Check if a service uses @c servers * @param server Server that is queried * @return True if server is used by at least one service */ bool service_server_in_use(const SERVER* server); /** * Check if filter is used by any service * * @param filter Filter to inspect * * @return True if at least one service uses the filter */ bool service_filter_in_use(const SFilterDef& filter); /** Update the server weights used by services */ void service_update_weights(); /** * @brief Add parameters to a service * * A copy of @c param is added to @c service. * * @param service Service where the parameters are added * @param param Parameters to add */ void service_add_parameters(Service* service, const MXS_CONFIG_PARAMETER* param); /** * @brief Add parameters to a service * * A copy of @c param is added to @c service. * * @param service Service where the parameters are added * @param key Parameter name * @param value Parameter value */ void service_add_parameter(Service* service, const char* key, const char* value); /** * @brief Remove service parameter * * @param service Service to modify * @param key Parameter to remove */ void service_remove_parameter(Service* service, const char* key); /** * @brief Replace service parameter * * @param service Service to modify * @param key Parameter name * @param value Parameter value */ void service_replace_parameter(Service* service, const char* key, const char* value); // Internal search function Service* service_internal_find(const char* name); /** * @brief Check if a service uses a server * @param service Service to check * @param server Server being used * @return True if service uses the server */ bool serviceHasBackend(Service* service, SERVER* server); /** * @brief Find listener with specified properties. * * @param service Service to check * @param socket Listener socket path * @param address Listener address * @param port Listener port number * * @note Either socket should be NULL and port non-zero or socket * non-NULL and port zero. * * @return True if service has the listener */ SListener service_find_listener(Service* service, const std::string& socket, const std::string& address, unsigned short port); /** * @brief Check if a MaxScale service listens on a port * * @param port The port to check * @return True if a MaxScale service uses the port */ bool service_port_is_used(int port); /** * @brief Check if a MaxScale service listens on a Unix domain socket * * @param path The socket path to check * @return True if a MaxScale service uses the socket */ bool service_socket_is_used(const std::string& socket_path); /** * @brief Check if the service has a listener with a matching name * * @param service Service to check * @param name Name to compare to * * @return True if the service has a listener with a matching name */ bool service_has_named_listener(Service* service, const char* name); /** * See if a monitor is used by any service * * @param monitor Monitor to look for * * @return The first service that uses the monitor or nullptr if no service uses it */ Service* service_uses_monitor(mxs::Monitor* monitor); // Required by MaxAdmin int service_enable_root(Service* service, int action); /** * @brief Convert a service to JSON * * @param service Service to convert * @param host Hostname of this server * * @return JSON representation of the service */ json_t* service_to_json(const Service* service, const char* host); /** * @brief Convert all services to JSON * * @param host Hostname of this server * * @return A JSON array with all services */ json_t* service_list_to_json(const char* host); /** * @brief Convert service listeners to JSON * * @param service Service whose listeners are converted * @param host Hostname of this server * * @return Array of JSON format listeners */ json_t* service_listener_list_to_json(const Service* service, const char* host); /** * @brief Convert service listener to JSON * * @param service Service whose listener is converted * @param name The name of the listener * @param host Hostname of this server * * @return JSON format listener */ json_t* service_listener_to_json(const Service* service, const char* name, const char* host); /** * @brief Get links to services that relate to a server * * @param server Server to inspect * @param host Hostname of this server * * @return Array of service links or NULL if no relations exist */ json_t* service_relations_to_server(const SERVER* server, const char* host); /** * @brief Get links to services that relate to a filter * * @param filter Filter to inspect * @param host Hostname of this server * * @return Array of service links */ json_t* service_relations_to_filter(const SFilterDef& filter, const char* host); /** * @brief Add server to all services associated with a monitor * * @param monitor A monitor. * @param server A server. */ void service_add_server(mxs::Monitor* pMonitor, SERVER* pServer); /** * @brief Remove server from all services associated with a monitor * * @param monitor A monitor. * @param server A server. */ void service_remove_server(mxs::Monitor* pMonitor, SERVER* pServer); std::unique_ptr serviceGetList(void); std::unique_ptr serviceGetListenerList(void);