mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-25 08:58:14 +02:00 
			
		
		
		
	 aed5928207
			
		
	
	
	aed5928207
	
	
	
		
			
			into a separate transaction_participant structure handlerton inherits it, so handlerton itself doesn't change. but entities that only need to participate in a transaction, like binlog or online alter log, use a transaction_participant and no longer need to pretend to be a full-blown but invisible storage engine which doesn't support create table.
		
			
				
	
	
		
			484 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			484 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #ifndef HA_FEDERATEDX_INCLUDED
 | |
| #define HA_FEDERATEDX_INCLUDED
 | |
| /*
 | |
| Copyright (c) 2008, Patrick Galbraith
 | |
| All rights reserved.
 | |
| 
 | |
| Redistribution and use in source and binary forms, with or without
 | |
| modification, are permitted provided that the following conditions are
 | |
| met:
 | |
| 
 | |
|     * Redistributions of source code must retain the above copyright
 | |
| notice, this list of conditions and the following disclaimer.
 | |
| 
 | |
|     * Redistributions in binary form must reproduce the above
 | |
| copyright notice, this list of conditions and the following disclaimer
 | |
| in the documentation and/or other materials provided with the
 | |
| distribution.
 | |
| 
 | |
|     * Neither the name of Patrick Galbraith nor the names of its
 | |
| contributors may be used to endorse or promote products derived from
 | |
| this software without specific prior written permission.
 | |
| 
 | |
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 | |
| "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 | |
| LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 | |
| A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 | |
| OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 | |
| SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 | |
| LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 | |
| DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 | |
| THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 | |
| (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 | |
| OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | |
| */
 | |
| 
 | |
| //#include <mysql.h>
 | |
| #include <my_global.h>
 | |
| #include <thr_lock.h>
 | |
| #include "handler.h"
 | |
| 
 | |
| class federatedx_io;
 | |
| 
 | |
| /*
 | |
|   FEDERATEDX_SERVER will eventually be a structure that will be shared among
 | |
|   all FEDERATEDX_SHARE instances so that the federated server can minimise
 | |
|   the number of open connections. This will eventually lead to the support
 | |
|   of reliable XA federated tables.
 | |
| */
 | |
| typedef struct st_fedrated_server {
 | |
|   MEM_ROOT mem_root;
 | |
|   uint use_count, io_count;
 | |
| 
 | |
|   uchar *key;
 | |
|   uint key_length;
 | |
| 
 | |
|   const char *scheme;
 | |
|   const char *hostname;
 | |
|   const char *username;
 | |
|   const char *password;
 | |
|   const char *database;
 | |
|   const char *socket;
 | |
|   ushort port;
 | |
| 
 | |
|   const char *csname;
 | |
| 
 | |
|   mysql_mutex_t mutex;
 | |
|   federatedx_io *idle_list;
 | |
| } FEDERATEDX_SERVER;
 | |
| 
 | |
| /*
 | |
|   Please read ha_exmple.cc before reading this file.
 | |
|   Please keep in mind that the federatedx storage engine implements all methods
 | |
|   that are required to be implemented. handler.h has a full list of methods
 | |
|   that you can implement.
 | |
| */
 | |
| 
 | |
| /*
 | |
|   handler::print_error has a case statement for error numbers.
 | |
|   This value is (10000) is far out of range and will envoke the
 | |
|   default: case.
 | |
|   (Current error range is 120-159 from include/my_base.h)
 | |
| */
 | |
| #define HA_FEDERATEDX_ERROR_WITH_REMOTE_SYSTEM 10000
 | |
| 
 | |
| #define FEDERATEDX_QUERY_BUFFER_SIZE STRING_BUFFER_USUAL_SIZE * 5
 | |
| #define FEDERATEDX_RECORDS_IN_RANGE 2
 | |
| #define FEDERATEDX_MAX_KEY_LENGTH 3500 // Same as innodb
 | |
| 
 | |
| /*
 | |
|   FEDERATEDX_SHARE is a structure that will be shared amoung all open handlers
 | |
|   The example implements the minimum of what you will probably need.
 | |
| */
 | |
| typedef struct st_federatedx_share {
 | |
|   MEM_ROOT mem_root;
 | |
| 
 | |
|   bool parsed;
 | |
|   /* this key is unique db/tablename */
 | |
|   const char *share_key;
 | |
|   /*
 | |
|     the primary select query to be used in rnd_init
 | |
|   */
 | |
|   LEX_CSTRING select_query;
 | |
|   /*
 | |
|     remote host info, parse_url supplies
 | |
|   */
 | |
|   char *server_name;
 | |
|   char *connection_string;
 | |
|   char *scheme;
 | |
|   char *hostname;
 | |
|   char *username;
 | |
|   char *password;
 | |
|   char *database;
 | |
|   char *table_name;
 | |
|   char *table;
 | |
|   char *socket;
 | |
|   char *sport;
 | |
|   int share_key_length;
 | |
|   ushort port;
 | |
| 
 | |
|   size_t table_name_length, server_name_length, connect_string_length;
 | |
|   uint use_count;
 | |
|   THR_LOCK lock;
 | |
|   FEDERATEDX_SERVER *s;
 | |
| } FEDERATEDX_SHARE;
 | |
| 
 | |
| 
 | |
| typedef struct st_federatedx_result FEDERATEDX_IO_RESULT;
 | |
| typedef struct st_federatedx_row FEDERATEDX_IO_ROW;
 | |
| typedef struct st_federatedx_rows FEDERATEDX_IO_ROWS;
 | |
| typedef ptrdiff_t FEDERATEDX_IO_OFFSET;
 | |
| 
 | |
| class federatedx_io
 | |
| {
 | |
|   friend class federatedx_txn;
 | |
|   FEDERATEDX_SERVER * const server;
 | |
|   federatedx_io **owner_ptr;
 | |
|   federatedx_io *txn_next;
 | |
|   federatedx_io *idle_next;
 | |
|   bool active;  /* currently participating in a transaction */
 | |
|   bool busy;    /* in use by a ha_federated instance */
 | |
|   bool readonly;/* indicates that no updates have occurred */
 | |
| 
 | |
| protected:
 | |
|   void set_active(bool new_active)
 | |
|   { active= new_active; }
 | |
| public:
 | |
|   federatedx_io(FEDERATEDX_SERVER *);
 | |
|   virtual ~federatedx_io();
 | |
| 
 | |
|   bool is_readonly() const { return readonly; }
 | |
|   bool is_active() const { return active; }
 | |
| 
 | |
|   const char * get_charsetname() const
 | |
|   { return server->csname ? server->csname : "latin1"; }
 | |
| 
 | |
|   const char * get_hostname() const { return server->hostname; }
 | |
|   const char * get_username() const { return server->username; }
 | |
|   const char * get_password() const { return server->password; }
 | |
|   const char * get_database() const { return server->database; }
 | |
|   ushort       get_port() const     { return server->port; }
 | |
|   const char * get_socket() const   { return server->socket; }
 | |
| 
 | |
|   static bool handles_scheme(const char *scheme);
 | |
|   static federatedx_io *construct(MEM_ROOT *server_root,
 | |
|                                   FEDERATEDX_SERVER *server);
 | |
| 
 | |
|   static void *operator new(size_t size, MEM_ROOT *mem_root) throw ()
 | |
|   { return alloc_root(mem_root, size); }
 | |
|   static void operator delete(void *ptr, size_t size)
 | |
|   { TRASH_FREE(ptr, size); }
 | |
|   static void operator delete(void *, MEM_ROOT *)
 | |
|   { }
 | |
| 
 | |
|   virtual int query(const char *buffer, size_t length)=0;
 | |
|   virtual FEDERATEDX_IO_RESULT *store_result()=0;
 | |
| 
 | |
|   virtual size_t max_query_size() const=0;
 | |
| 
 | |
|   virtual my_ulonglong affected_rows() const=0;
 | |
|   virtual my_ulonglong last_insert_id() const=0;
 | |
| 
 | |
|   virtual int error_code()=0;
 | |
|   virtual const char *error_str()=0;
 | |
| 
 | |
|   virtual void reset()=0;
 | |
|   virtual int commit()=0;
 | |
|   virtual int rollback()=0;
 | |
| 
 | |
|   virtual int savepoint_set(ulong sp)=0;
 | |
|   virtual ulong savepoint_release(ulong sp)=0;
 | |
|   virtual ulong savepoint_rollback(ulong sp)=0;
 | |
|   virtual void savepoint_restrict(ulong sp)=0;
 | |
| 
 | |
|   virtual ulong last_savepoint() const=0;
 | |
|   virtual ulong actual_savepoint() const=0;
 | |
|   virtual bool is_autocommit() const=0;
 | |
| 
 | |
|   virtual bool table_metadata(ha_statistics *stats, const char *table_name,
 | |
|                               uint table_name_length, uint flag) = 0;
 | |
| 
 | |
|   /* resultset operations */
 | |
| 
 | |
|   virtual void free_result(FEDERATEDX_IO_RESULT *io_result)=0;
 | |
|   virtual unsigned int get_num_fields(FEDERATEDX_IO_RESULT *io_result)=0;
 | |
|   virtual my_ulonglong get_num_rows(FEDERATEDX_IO_RESULT *io_result)=0;
 | |
|   virtual FEDERATEDX_IO_ROW *fetch_row(FEDERATEDX_IO_RESULT *io_result,
 | |
|                                        FEDERATEDX_IO_ROWS **current= NULL)=0;
 | |
|   virtual ulong *fetch_lengths(FEDERATEDX_IO_RESULT *io_result)=0;
 | |
|   virtual const char *get_column_data(FEDERATEDX_IO_ROW *row,
 | |
|                                       unsigned int column)=0;
 | |
|   virtual bool is_column_null(const FEDERATEDX_IO_ROW *row,
 | |
|                               unsigned int column) const=0;
 | |
| 
 | |
|   virtual size_t get_ref_length() const=0;
 | |
|   virtual void mark_position(FEDERATEDX_IO_RESULT *io_result,
 | |
|                              void *ref, FEDERATEDX_IO_ROWS *current)=0;
 | |
|   virtual int seek_position(FEDERATEDX_IO_RESULT **io_result,
 | |
|                             const void *ref)=0;
 | |
|   virtual void set_thd(void *thd) { }
 | |
| };
 | |
| 
 | |
| 
 | |
| class federatedx_txn
 | |
| {
 | |
|   federatedx_io *txn_list;
 | |
|   ulong savepoint_level;
 | |
|   ulong savepoint_stmt;
 | |
|   ulong savepoint_next;
 | |
| 
 | |
|   void release_scan();
 | |
| public:
 | |
|   federatedx_txn();
 | |
|   ~federatedx_txn();
 | |
| 
 | |
|   bool has_connections() const { return txn_list != NULL; }
 | |
|   bool in_transaction() const { return savepoint_next != 0; }
 | |
|   int acquire(FEDERATEDX_SHARE *share, void *thd, bool readonly, federatedx_io **io);
 | |
|   void release(federatedx_io **io);
 | |
|   void close(FEDERATEDX_SERVER *);
 | |
| 
 | |
|   bool txn_begin();
 | |
|   int txn_commit();
 | |
|   int txn_rollback();
 | |
| 
 | |
|   bool sp_acquire(ulong *save);
 | |
|   int sp_rollback(ulong *save);
 | |
|   int sp_release(ulong *save);
 | |
| 
 | |
|   bool stmt_begin();
 | |
|   int stmt_commit();
 | |
|   int stmt_rollback();
 | |
|   void stmt_autocommit();
 | |
| };
 | |
| 
 | |
| /*
 | |
|   Class definition for the storage engine
 | |
| */
 | |
| class ha_federatedx final : public handler
 | |
| {
 | |
|   friend int federatedx_db_init(void *p);
 | |
| 
 | |
|   THR_LOCK_DATA lock;      /* MySQL lock */
 | |
|   FEDERATEDX_SHARE *share;    /* Shared lock info */
 | |
|   federatedx_txn *txn;
 | |
|   federatedx_io *io;
 | |
|   FEDERATEDX_IO_RESULT *stored_result;
 | |
|   FEDERATEDX_IO_ROWS *current;
 | |
|   /**
 | |
|       Array of all stored results we get during a query execution.
 | |
|   */
 | |
|   DYNAMIC_ARRAY results;
 | |
|   bool position_called;
 | |
|   int remote_error_number;
 | |
|   char remote_error_buf[FEDERATEDX_QUERY_BUFFER_SIZE];
 | |
|   bool ignore_duplicates, replace_duplicates;
 | |
|   bool insert_dup_update, table_will_be_deleted;
 | |
|   DYNAMIC_STRING bulk_insert;
 | |
| 
 | |
| private:
 | |
|   /*
 | |
|       return 0 on success
 | |
|       return errorcode otherwise
 | |
|   */
 | |
|   uint convert_row_to_internal_format(uchar *buf, FEDERATEDX_IO_ROW *row,
 | |
|                                       FEDERATEDX_IO_RESULT *result);
 | |
|   bool create_where_from_key(String *to, KEY *key_info,
 | |
|                              const key_range *start_key,
 | |
|                              const key_range *end_key, bool eq_range);
 | |
|   int stash_remote_error();
 | |
| 
 | |
|   static federatedx_txn *get_txn(THD *thd, bool no_create= FALSE);
 | |
|   static int disconnect(MYSQL_THD thd);
 | |
|   static int savepoint_set(MYSQL_THD thd, void *sv);
 | |
|   static int savepoint_rollback(MYSQL_THD thd, void *sv);
 | |
|   static int savepoint_release(MYSQL_THD thd, void *sv);
 | |
|   static int commit(MYSQL_THD thd, bool all);
 | |
|   static int rollback(MYSQL_THD thd, bool all);
 | |
|   static int discover_assisted(handlerton *, THD*, TABLE_SHARE *,
 | |
|                                HA_CREATE_INFO *);
 | |
| 
 | |
|   bool append_stmt_insert(String *query);
 | |
| 
 | |
|   int read_next(uchar *buf, FEDERATEDX_IO_RESULT *result);
 | |
|   int index_read_idx_with_result_set(uchar *buf, uint index,
 | |
|                                      const uchar *key,
 | |
|                                      uint key_len,
 | |
|                                      ha_rkey_function find_flag,
 | |
|                                      FEDERATEDX_IO_RESULT **result);
 | |
|   int real_query(const char *query, uint length);
 | |
|   int real_connect(FEDERATEDX_SHARE *my_share, uint create_flag);
 | |
| public:
 | |
|   ha_federatedx(handlerton *hton, TABLE_SHARE *table_arg);
 | |
|   ~ha_federatedx() = default;
 | |
|   /*
 | |
|     The name of the index type that will be used for display
 | |
|     don't implement this method unless you really have indexes
 | |
|    */
 | |
|   // perhaps get index type
 | |
|   const char *index_type(uint inx) override { return "REMOTE"; }
 | |
|   /*
 | |
|     This is a list of flags that says what the storage engine
 | |
|     implements. The current table flags are documented in
 | |
|     handler.h
 | |
|   */
 | |
|   ulonglong table_flags() const override
 | |
|   {
 | |
|     /* fix server to be able to get remote server table flags */
 | |
|     return (HA_PRIMARY_KEY_IN_READ_INDEX | HA_FILE_BASED
 | |
|             | HA_REC_NOT_IN_SEQ | HA_AUTO_PART_KEY | HA_CAN_INDEX_BLOBS |
 | |
|             HA_BINLOG_ROW_CAPABLE | HA_BINLOG_STMT_CAPABLE | HA_CAN_REPAIR |
 | |
|             HA_PRIMARY_KEY_REQUIRED_FOR_DELETE | HA_CAN_ONLINE_BACKUPS |
 | |
|             HA_PARTIAL_COLUMN_READ | HA_NULL_IN_KEY | HA_NON_COMPARABLE_ROWID);
 | |
|   }
 | |
|   /*
 | |
|     This is a bitmap of flags that says how the storage engine
 | |
|     implements indexes. The current index flags are documented in
 | |
|     handler.h. If you do not implement indexes, just return zero
 | |
|     here.
 | |
| 
 | |
|     part is the key part to check. First key part is 0
 | |
|     If all_parts it's set, MySQL want to know the flags for the combined
 | |
|     index up to and including 'part'.
 | |
|   */
 | |
|     /* fix server to be able to get remote server index flags */
 | |
|   ulong index_flags(uint inx, uint part, bool all_parts) const override
 | |
|   {
 | |
|     return (HA_READ_NEXT | HA_READ_RANGE);
 | |
|   }
 | |
|   uint max_supported_record_length() const override { return HA_MAX_REC_LENGTH; }
 | |
|   uint max_supported_keys() const       override { return MAX_KEY; }
 | |
|   uint max_supported_key_parts() const  override { return MAX_REF_PARTS; }
 | |
|   uint max_supported_key_length() const override { return FEDERATEDX_MAX_KEY_LENGTH; }
 | |
|   uint max_supported_key_part_length() const override { return FEDERATEDX_MAX_KEY_LENGTH; }
 | |
|   /*
 | |
|     Called in test_quick_select to determine if indexes should be used.
 | |
|     Normally, we need to know number of blocks . For federatedx we need to
 | |
|     know number of blocks on remote side, and number of packets and blocks
 | |
|     on the network side (?)
 | |
|     Talk to Kostja about this - how to get the
 | |
|     number of rows * ...
 | |
|     disk scan time on other side (block size, size of the row) + network time ...
 | |
|     The reason for "1000" is that such a large number forces this to use indexes "
 | |
|   */
 | |
|   IO_AND_CPU_COST scan_time() override
 | |
|   {
 | |
|     DBUG_PRINT("info", ("records %lu", (ulong) stats.records));
 | |
|     return
 | |
|     {
 | |
|       0,
 | |
|         (double) (stats.mean_rec_length * stats.records)/8192 * DISK_READ_COST+
 | |
|         1000,
 | |
|     };
 | |
|   }
 | |
|   IO_AND_CPU_COST keyread_time(uint index, ulong ranges, ha_rows rows,
 | |
|                                ulonglong blocks) override
 | |
|   {
 | |
|     return {0, (double) (ranges + rows) * DISK_READ_COST };
 | |
|   }
 | |
|   IO_AND_CPU_COST rnd_pos_time(ha_rows rows) override
 | |
|   {
 | |
|     return {0, (double) rows * DISK_READ_COST };
 | |
|   }
 | |
| 
 | |
|   const key_map *keys_to_use_for_scanning() override { return &key_map_full; }
 | |
|   /*
 | |
|     Everything below are methods that we implement in ha_federatedx.cc.
 | |
| 
 | |
|     Most of these methods are not obligatory, skip them and
 | |
|     MySQL will treat them as not implemented
 | |
|   */
 | |
|   int open(const char *name, int mode, uint test_if_locked) override;    // required
 | |
|   int close(void) override;                                              // required
 | |
| 
 | |
|   void start_bulk_insert(ha_rows rows, uint flags) override;
 | |
|   int end_bulk_insert() override;
 | |
|   int write_row(const uchar *buf) override;
 | |
|   int update_row(const uchar *old_data, const uchar *new_data) override;
 | |
|   int delete_row(const uchar *buf) override;
 | |
|   int index_init(uint keynr, bool sorted) override;
 | |
|   ha_rows estimate_rows_upper_bound() override;
 | |
|   int index_read(uchar *buf, const uchar *key,
 | |
|                  uint key_len, enum ha_rkey_function find_flag) override;
 | |
|   int index_read_idx(uchar *buf, uint idx, const uchar *key,
 | |
|                      uint key_len, enum ha_rkey_function find_flag);
 | |
|   int index_next(uchar *buf) override;
 | |
|   int index_end() override;
 | |
|   int read_range_first(const key_range *start_key,
 | |
|                                const key_range *end_key,
 | |
|                                bool eq_range, bool sorted) override;
 | |
|   int read_range_next() override;
 | |
|   /*
 | |
|     unlike index_init(), rnd_init() can be called two times
 | |
|     without rnd_end() in between (it only makes sense if scan=1).
 | |
|     then the second call should prepare for the new table scan
 | |
|     (e.g if rnd_init allocates the cursor, second call should
 | |
|     position it to the start of the table, no need to deallocate
 | |
|     and allocate it again
 | |
|   */
 | |
|   int rnd_init(bool scan) override;                                      //required
 | |
|   int rnd_end() override;
 | |
|   int rnd_next(uchar *buf) override;                                      //required
 | |
|   int rnd_pos(uchar *buf, uchar *pos) override;                            //required
 | |
|   void position(const uchar *record) override;                            //required
 | |
|   /*
 | |
|     A ref is a pointer inside a local buffer. It is not comparable to
 | |
|     other ref's. This is never called as HA_NON_COMPARABLE_ROWID is set.
 | |
|   */
 | |
|   int cmp_ref(const uchar *ref1, const uchar *ref2) override
 | |
|   {
 | |
| #ifdef NOT_YET
 | |
|     DBUG_ASSERT(0);
 | |
|     return 0;
 | |
| #else
 | |
|     return handler::cmp_ref(ref1,ref2);         /* Works if table scan is used */
 | |
| #endif
 | |
|   }
 | |
|   int info(uint) override;                                              //required
 | |
|   int extra(ha_extra_function operation) override;
 | |
| 
 | |
|   void update_auto_increment(void);
 | |
|   int repair(THD* thd, HA_CHECK_OPT* check_opt) override;
 | |
|   int optimize(THD* thd, HA_CHECK_OPT* check_opt) override;
 | |
|   int delete_table(const char *name) override
 | |
|   {
 | |
|     return 0;
 | |
|   }
 | |
|   int delete_all_rows(void) override;
 | |
|   int create(const char *name, TABLE *form,
 | |
|              HA_CREATE_INFO *create_info) override;                      //required
 | |
|   ha_rows records_in_range(uint inx, const key_range *start_key,
 | |
|                            const key_range *end_key, page_range *pages) override;
 | |
|   uint8 table_cache_type() override { return HA_CACHE_TBL_NOCACHE; }
 | |
| 
 | |
|   THR_LOCK_DATA **store_lock(THD *thd, THR_LOCK_DATA **to,
 | |
|                              enum thr_lock_type lock_type) override;     //required
 | |
|   bool get_error_message(int error, String *buf) override;
 | |
|   int start_stmt(THD *thd, thr_lock_type lock_type) override;
 | |
|   int external_lock(THD *thd, int lock_type) override;
 | |
|   int reset(void) override;
 | |
|   int free_result(void);
 | |
| 
 | |
|   const FEDERATEDX_SHARE *get_federatedx_share() const { return share; }
 | |
|   friend class ha_federatedx_derived_handler;
 | |
|   friend class ha_federatedx_select_handler;
 | |
|   friend class federatedx_handler_base;
 | |
| };
 | |
| 
 | |
| extern const char ident_quote_char;              // Character for quoting
 | |
|                                                  // identifiers
 | |
| extern const char value_quote_char;              // Character for quoting
 | |
|                                                  // literals
 | |
| 
 | |
| extern bool append_ident(String *string, const char *name, size_t length,
 | |
|                          const char quote_char);
 | |
| 
 | |
| 
 | |
| extern federatedx_io *instantiate_io_mysql(MEM_ROOT *server_root,
 | |
|                                            FEDERATEDX_SERVER *server);
 | |
| extern federatedx_io *instantiate_io_null(MEM_ROOT *server_root,
 | |
|                                           FEDERATEDX_SERVER *server);
 | |
| 
 | |
| #include "federatedx_pushdown.h"
 | |
| 
 | |
| #endif /* HA_FEDERATEDX_INCLUDED */
 |