mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-26 01:18:31 +02:00 
			
		
		
		
	 608b0ee52e
			
		
	
	
	608b0ee52e
	
	
	
		
			
			Problem:
=======
Upon deleting or updating a row in a parent table (with primary key), if
the child table has virtual column and an associated key with ON UPDATE
CASCADE/ON DELETE CASCADE, it will result in slave crash.
Analysis:
========
Tables which are related through foreign key require prelocking similar to
triggers. i.e If a table has triggers/foreign keys we should add all tables
and routines used by them to the prelocking set.  This prelocking happens
during 'open_and_lock_tables' call.  Each table being opened is checked for
foreign key references. If foreign key reference exists then the child
table is opened and it is linked to the table_list. Upon any modification
to  parent table its corresponding child tables are retried from table_list
and they are updated accordingly. This prelocking work fine on master.
On slave  prelocking works for following cases.
 - Statement/mixed based replication
 - In row based replication when trigger execution is enabled through
   'slave_run_triggers_for_rbr=YES/LOGGING/ENFORCE'
Otherwise it results in an assert/crash, as the parent table will not find
the corresponding child table and it will be NULL. Dereferencing NULL
pointer leads to slave server exit.
Fix:
===
Introduce a new 'slave_fk_event_map' flag similar to 'trg_event_map'. This
flag will ensure that when foreign key is enabled in row based replication
all the parent and child tables are prelocked, so that parent is able to
locate the child table.
Note: This issue is specific to slave, hence only slave needs to be
      upgraded.
		
	
			
		
			
				
	
	
		
			425 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
			
		
		
	
	
			425 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
| # ==== Purpose ====
 | |
| #
 | |
| # Test verifies that, slave doesn't report any assert on UPDATE or DELETE of
 | |
| # row which tries to update the virtual columns with associated KEYs.
 | |
| #
 | |
| # Test scenarios are listed below.
 | |
| #   1) KEY on a virtual column with ON DELETE CASCADE
 | |
| #   2) Verify "ON DELETE CASCADE" for parent->child->child scenario
 | |
| #   3) KEY on a virtual column with ON UPDATE CASCADE
 | |
| #   4) Define triggers on master, their results should be replicated
 | |
| #      as part of row events and they should be applied on slave with
 | |
| #      the default slave_run_triggers_for_rbr=NO
 | |
| #   5) Define triggers + Foreign Keys on master, their results should be
 | |
| #      replicated as part of row events and master and slave should be in sync.
 | |
| #   6) Triggers are present only on slave and 'slave_run_triggers_for_rbr=NO'
 | |
| #   7) Triggers are present only on slave and 'slave_run_triggers_for_rbr=YES'
 | |
| #   8) Triggers and Foreign Keys are present only on slave and
 | |
| #      'slave_run_triggers_for_rbr=NO'
 | |
| #   9) Triggers are Foreign Keys are present only on slave and
 | |
| #      'slave_run_triggers_for_rbr=YES'
 | |
| #
 | |
| # ==== References ====
 | |
| #
 | |
| # MDEV-23033: All slaves crash once in ~24 hours and loop restart with signal 11
 | |
| #
 | |
| 
 | |
| --source include/have_binlog_format_row.inc
 | |
| --source include/have_innodb.inc
 | |
| --source include/master-slave.inc
 | |
| 
 | |
| 
 | |
| --echo #
 | |
| --echo # Test case 1: KEY on a virtual column with ON DELETE CASCADE
 | |
| --echo #
 | |
| CREATE TABLE t1 (id INT NOT NULL PRIMARY KEY) ENGINE=InnoDB;
 | |
| INSERT INTO t1 VALUES (1),(2),(3);
 | |
| 
 | |
| CREATE TABLE t2 (id INT NOT NULL PRIMARY KEY,
 | |
|   t1_id INT NOT NULL,
 | |
|   v_col INT AS (t1_id+1) VIRTUAL, KEY (v_col), KEY (t1_id),
 | |
|   CONSTRAINT a FOREIGN KEY (t1_id) REFERENCES t1 (id) ON DELETE CASCADE
 | |
| ) ENGINE=InnoDB;
 | |
| 
 | |
| INSERT INTO t2 VALUES (90,1,NULL);
 | |
| INSERT INTO t2 VALUES (91,2,default);
 | |
| 
 | |
| # Following query results in an assert on slave
 | |
| DELETE FROM t1 WHERE id=1;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --echo #
 | |
| --echo # Verify data consistency on slave
 | |
| --echo #
 | |
| --let $diff_tables= master:test.t1, slave:test.t1
 | |
| --source include/diff_tables.inc
 | |
| --let $diff_tables= master:test.t2, slave:test.t2
 | |
| --source include/diff_tables.inc
 | |
| 
 | |
| --connection master
 | |
| DROP TABLE t2,t1;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --echo #
 | |
| --echo # Test case 2: Verify "ON DELETE CASCADE" for parent->child->child scenario
 | |
| --echo #              Parent table: users
 | |
| --echo #              Child tables: matchmaking_groups, matchmaking_group_users
 | |
| --echo #              Parent table: matchmaking_groups
 | |
| --echo #              Child tables: matchmaking_group_users, matchmaking_group_maps
 | |
| --echo #
 | |
| --echo #              Deleting a row from parent table should be reflected in
 | |
| --echo #              child tables.
 | |
| --echo #              matchmaking_groups->matchmaking_group_users->matchmaking_group_maps
 | |
| --echo #              users->matchmaking_group_users->matchmaking_group_maps
 | |
| --echo #
 | |
| 
 | |
| --connection master
 | |
| CREATE TABLE users (id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
 | |
|   name VARCHAR(32) NOT NULL DEFAULT ''
 | |
| ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
 | |
| 
 | |
| CREATE TABLE matchmaking_groups (
 | |
|   id BIGINT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
 | |
|   host_user_id INT UNSIGNED NOT NULL UNIQUE,
 | |
|   v_col INT AS (host_user_id+1) VIRTUAL, KEY (v_col),
 | |
|   CONSTRAINT FOREIGN KEY (host_user_id) REFERENCES users (id)
 | |
|   ON DELETE CASCADE ON UPDATE CASCADE
 | |
| ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
 | |
| 
 | |
| CREATE TABLE matchmaking_group_users (
 | |
|   matchmaking_group_id BIGINT UNSIGNED NOT NULL,
 | |
|   user_id INT UNSIGNED NOT NULL,
 | |
|   v_col1 int as (user_id+1) virtual, KEY (v_col1),
 | |
|   PRIMARY KEY (matchmaking_group_id,user_id),
 | |
|   UNIQUE KEY user_id (user_id),
 | |
|   CONSTRAINT FOREIGN KEY (matchmaking_group_id)
 | |
|   REFERENCES matchmaking_groups (id) ON DELETE CASCADE ON UPDATE CASCADE,
 | |
|   CONSTRAINT FOREIGN KEY (user_id)
 | |
|   REFERENCES users (id) ON DELETE CASCADE ON UPDATE CASCADE
 | |
| ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
 | |
| 
 | |
| CREATE TABLE matchmaking_group_maps (
 | |
|   matchmaking_group_id BIGINT UNSIGNED NOT NULL,
 | |
|   map_id TINYINT UNSIGNED NOT NULL,
 | |
|   v_col2 INT AS (map_id+1) VIRTUAL, KEY (v_col2),
 | |
|   PRIMARY KEY (matchmaking_group_id,map_id),
 | |
|   CONSTRAINT FOREIGN KEY (matchmaking_group_id)
 | |
|   REFERENCES matchmaking_groups (id) ON DELETE CASCADE ON UPDATE CASCADE
 | |
| ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --connection master
 | |
| INSERT INTO users VALUES (NULL,'foo'),(NULL,'bar');
 | |
| INSERT INTO matchmaking_groups VALUES (10,1,default),(11,2,default);
 | |
| INSERT INTO matchmaking_group_users VALUES (10,1,default),(11,2,default);
 | |
| INSERT INTO matchmaking_group_maps VALUES (10,55,default),(11,66,default);
 | |
| 
 | |
| DELETE FROM matchmaking_groups WHERE id = 10;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --echo #
 | |
| --echo # No rows should be returned as ON DELETE CASCASE should have removed
 | |
| --echo # corresponding rows from child tables. There should not any mismatch
 | |
| --echo # of 'id' field between parent->child.
 | |
| --echo #
 | |
| SELECT * FROM matchmaking_group_users WHERE matchmaking_group_id NOT IN (SELECT id FROM matchmaking_groups);
 | |
| SELECT * FROM matchmaking_group_maps WHERE matchmaking_group_id NOT IN (SELECT id FROM matchmaking_groups);
 | |
| 
 | |
| --echo #
 | |
| --echo # Rows with id=11 should be present
 | |
| --echo #
 | |
| SELECT * FROM matchmaking_group_users;
 | |
| SELECT * FROM matchmaking_group_maps;
 | |
| 
 | |
| --connection master
 | |
| DELETE FROM users WHERE id = 2;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --echo #
 | |
| --echo # No rows should be present in both the child tables
 | |
| --echo #
 | |
| SELECT * FROM matchmaking_group_users;
 | |
| SELECT * FROM matchmaking_group_maps;
 | |
| 
 | |
| --connection master
 | |
| DROP TABLE matchmaking_group_maps, matchmaking_group_users, matchmaking_groups, users;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --echo #
 | |
| --echo # Test case 3: KEY on a virtual column with ON UPDATE CASCADE
 | |
| --echo #
 | |
| 
 | |
| --connection master
 | |
| CREATE TABLE t1 (a INT NOT NULL PRIMARY KEY, b INT NOT NULL) ENGINE=InnoDB;
 | |
| INSERT INTO t1 VALUES (1, 80);
 | |
| 
 | |
| CREATE TABLE t2 (a INT KEY, b INT,
 | |
|   v_col int as (b+1) virtual, KEY (v_col),
 | |
|   CONSTRAINT b FOREIGN KEY (b) REFERENCES t1(a) ON UPDATE CASCADE
 | |
| ) ENGINE=InnoDB;
 | |
| INSERT INTO t2 VALUES (51, 1, default);
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --connection master
 | |
| UPDATE t1 SET a = 50 WHERE a = 1;
 | |
| 
 | |
| --echo #
 | |
| --echo # Master: Verify that ON UPDATE CASCADE works fine
 | |
| --echo # old_row: (51, 1, 2) ON UPDATE New_row: (51, 50, 51)
 | |
| --echo #
 | |
| SELECT * FROM t2 WHERE b=50;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --echo #
 | |
| --echo # Slave: Verify that ON UPDATE CASCADE works fine
 | |
| --echo # old_row: (51, 1, 2) ON UPDATE New_row: (51, 50, 51)
 | |
| --echo #
 | |
| SELECT * FROM t2 WHERE b=50;
 | |
| 
 | |
| --connection master
 | |
| DROP TABLE t2, t1;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --echo #
 | |
| --echo # Test case 4: Define triggers on master, their results should be 
 | |
| --echo #              replicated as part of row events and they should be
 | |
| --echo #              applied on slave with the default
 | |
| --echo #              slave_run_triggers_for_rbr=NO
 | |
| --echo #
 | |
| 
 | |
| # In row-based replication, the binary log contains row changes. It will have
 | |
| # both the changes made by the statement itself, and the changes made by the
 | |
| # triggers that were invoked by the statement. Slave server(s) do not need to
 | |
| # run triggers for row changes they are applying.  Hence verify that this
 | |
| # property remains the same and data should be available as if trigger was
 | |
| # executed. Please note by default slave_run_triggers_for_rbr=NO.
 | |
| 
 | |
| --connection master
 | |
| CREATE TABLE t1 (id INT NOT NULL PRIMARY KEY) ENGINE=InnoDB;
 | |
| CREATE TABLE t2 (count INT NOT NULL) ENGINE=InnoDB;
 | |
| CREATE TRIGGER trg AFTER INSERT ON t1 FOR EACH ROW INSERT INTO t2 VALUES (1);
 | |
| INSERT INTO t1 VALUES (2),(3);
 | |
| --sync_slave_with_master
 | |
| 
 | |
| SHOW GLOBAL VARIABLES LIKE 'slave_run_triggers_for_rbr';
 | |
| --echo #
 | |
| --echo # As two rows are inserted in table 't1', two rows should get inserted
 | |
| --echo # into table 't2' as part of trigger.
 | |
| --echo #
 | |
| --let $assert_cond= COUNT(*) = 2 FROM t2
 | |
| --let $assert_text= Table t2 should have two rows.
 | |
| --source include/assert.inc
 | |
| 
 | |
| --connection master
 | |
| DROP TABLE t1,t2;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --echo #
 | |
| --echo # Test case 5: Define triggers + Foreign Keys on master, their results
 | |
| --echo #              should be replicated as part of row events and master
 | |
| --echo #              and slave should be in sync.
 | |
| --echo #
 | |
| --connection master
 | |
| CREATE TABLE t1 (id INT NOT NULL PRIMARY KEY) ENGINE=InnoDB;
 | |
| CREATE TABLE t2 (t1_id INT NOT NULL,
 | |
|   v_col INT AS (t1_id+1) VIRTUAL, KEY (v_col), KEY (t1_id),
 | |
|   CONSTRAINT a FOREIGN KEY (t1_id) REFERENCES t1 (id) ON DELETE CASCADE
 | |
| ) ENGINE=InnoDB;
 | |
| CREATE TABLE t3 (count INT NOT NULL) ENGINE=InnoDB;
 | |
| CREATE TRIGGER trg AFTER INSERT ON t1 FOR EACH ROW INSERT INTO t3 VALUES (1);
 | |
| 
 | |
| INSERT INTO t1 VALUES (2),(3);
 | |
| INSERT INTO t2 VALUES (2, default), (3, default);
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --echo #
 | |
| --echo # As two rows are inserted in table 't1', two rows should get inserted
 | |
| --echo # into table 't3' as part of trigger.
 | |
| --echo #
 | |
| --let $assert_cond= COUNT(*) = 2 FROM t3
 | |
| --let $assert_text= Table t3 should have two rows.
 | |
| --source include/assert.inc
 | |
| 
 | |
| --echo #
 | |
| --echo # Verify ON DELETE CASCASE correctness
 | |
| --echo #
 | |
| --connection master
 | |
| DELETE FROM t1 WHERE id=2;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| --connection master
 | |
| --let $diff_tables= master:test.t1, slave:test.t1
 | |
| --source include/diff_tables.inc
 | |
| --let $diff_tables= master:test.t2, slave:test.t2
 | |
| --source include/diff_tables.inc
 | |
| --let $diff_tables= master:test.t3, slave:test.t3
 | |
| --source include/diff_tables.inc
 | |
| 
 | |
| DROP TABLE t3,t2,t1;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| #
 | |
| # Test case: Triggers only on slave
 | |
| #
 | |
| --write_file $MYSQLTEST_VARDIR/tmp/trig_on_slave.inc PROCEDURE
 | |
|   if ($slave_run_triggers_for_rbr == '') {
 | |
|     --die !!!ERROR IN TEST: you must set $slave_run_triggers_for_rbr
 | |
|   }
 | |
| 
 | |
| --connection slave
 | |
| SET @save_slave_run_triggers_for_rbr= @@GLOBAL.slave_run_triggers_for_rbr;
 | |
| --eval SET GLOBAL slave_run_triggers_for_rbr= $slave_run_triggers_for_rbr;
 | |
| SHOW GLOBAL VARIABLES LIKE '%slave_run_triggers_for_rbr%';
 | |
| 
 | |
| --connection master
 | |
| CREATE TABLE t1 (id INT NOT NULL PRIMARY KEY) ENGINE=InnoDB;
 | |
| CREATE TABLE t2 (t1_id INT NOT NULL,
 | |
|   v_col INT AS (t1_id+1) VIRTUAL, KEY (v_col),
 | |
|   KEY (t1_id), CONSTRAINT a FOREIGN KEY (t1_id) REFERENCES t1 (id) ON DELETE CASCADE
 | |
| ) ENGINE=InnoDB;
 | |
| CREATE TABLE t3 (count INT NOT NULL) ENGINE=InnoDB;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| CREATE TRIGGER trg AFTER INSERT ON t2 FOR EACH ROW INSERT INTO t3 VALUES (1);
 | |
| 
 | |
| --connection master
 | |
| INSERT INTO t1 VALUES (2),(3);
 | |
| INSERT INTO t2 VALUES (2, default), (3, default);
 | |
| --sync_slave_with_master
 | |
| 
 | |
| if ($slave_run_triggers_for_rbr == 'NO') {
 | |
| --echo #
 | |
| --echo # Count must be 0
 | |
| --echo #
 | |
| --let $assert_cond= COUNT(*) = 0 FROM t3
 | |
| --let $assert_text= Table t3 should have zero rows.
 | |
| --source include/assert.inc
 | |
| }
 | |
| if ($slave_run_triggers_for_rbr == 'YES') {
 | |
| --echo #
 | |
| --echo # Count must be 2
 | |
| --echo #
 | |
| --let $assert_cond= COUNT(*) = 2 FROM t3
 | |
| --let $assert_text= Table t3 should have two rows.
 | |
| --source include/assert.inc
 | |
| }
 | |
| 
 | |
| --connection master
 | |
| DELETE FROM t1 WHERE id=2;
 | |
| --sync_slave_with_master
 | |
| SET GLOBAL slave_run_triggers_for_rbr= @save_slave_run_triggers_for_rbr;
 | |
| 
 | |
| --echo #
 | |
| --echo # Verify t1, t2 are consistent on slave.
 | |
| --echo #
 | |
| --let $diff_tables= master:test.t1, slave:test.t1
 | |
| --source include/diff_tables.inc
 | |
| --let $diff_tables= master:test.t2, slave:test.t2
 | |
| --source include/diff_tables.inc
 | |
| 
 | |
| --connection master
 | |
| DROP TABLE t3,t2,t1;
 | |
| --sync_slave_with_master
 | |
| #END OF
 | |
| PROCEDURE
 | |
| 
 | |
| --echo #
 | |
| --echo # Test case 6: Triggers are present only on slave and
 | |
| --echo #              'slave_run_triggers_for_rbr=NO'
 | |
| --echo #
 | |
| --let $slave_run_triggers_for_rbr=NO
 | |
| --source $MYSQLTEST_VARDIR/tmp/trig_on_slave.inc
 | |
| 
 | |
| --echo #
 | |
| --echo # Test case 7: Triggers are present only on slave and
 | |
| --echo #              'slave_run_triggers_for_rbr=YES'
 | |
| --echo #
 | |
| --let $slave_run_triggers_for_rbr=YES
 | |
| --source $MYSQLTEST_VARDIR/tmp/trig_on_slave.inc
 | |
| --remove_file $MYSQLTEST_VARDIR/tmp/trig_on_slave.inc
 | |
| 
 | |
| #
 | |
| # Test case: Trigger and Foreign Key are present only on slave
 | |
| #
 | |
| --write_file $MYSQLTEST_VARDIR/tmp/trig_fk_on_slave.inc PROCEDURE
 | |
|   if ($slave_run_triggers_for_rbr == '') {
 | |
|     --die !!!ERROR IN TEST: you must set $slave_run_triggers_for_rbr
 | |
|   }
 | |
| 
 | |
| --connection slave
 | |
| SET @save_slave_run_triggers_for_rbr= @@GLOBAL.slave_run_triggers_for_rbr;
 | |
| --eval SET GLOBAL slave_run_triggers_for_rbr= $slave_run_triggers_for_rbr;
 | |
| SHOW GLOBAL VARIABLES LIKE '%slave_run_triggers_for_rbr%';
 | |
| 
 | |
| --connection master
 | |
| CREATE TABLE t1 (id INT NOT NULL PRIMARY KEY) ENGINE=InnoDB;
 | |
| SET sql_log_bin=0;
 | |
| CREATE TABLE t2 (t1_id INT NOT NULL,v_col INT AS (t1_id+1) VIRTUAL) ENGINE=INNODB;
 | |
| SET sql_log_bin=1;
 | |
| CREATE TABLE t3 (count INT NOT NULL) ENGINE=InnoDB;
 | |
| --sync_slave_with_master
 | |
| 
 | |
| # Have foreign key and trigger on slave.
 | |
| CREATE TABLE t2 (t1_id INT NOT NULL,
 | |
|   v_col INT AS (t1_id+1) VIRTUAL, KEY (v_col), KEY (t1_id),
 | |
|   CONSTRAINT a FOREIGN KEY (t1_id) REFERENCES t1 (id) ON DELETE CASCADE
 | |
| ) ENGINE=InnoDB;
 | |
| CREATE TRIGGER trg AFTER INSERT ON t2 FOR EACH ROW INSERT INTO t3 VALUES (1);
 | |
| 
 | |
| --connection master
 | |
| INSERT INTO t1 VALUES (2),(3);
 | |
| INSERT INTO t2 VALUES (2, default), (3, default);
 | |
| --sync_slave_with_master
 | |
| 
 | |
| if ($slave_run_triggers_for_rbr == 'NO') {
 | |
| --echo #
 | |
| --echo # Count must be 0
 | |
| --echo #
 | |
| --let $assert_cond= COUNT(*) = 0 FROM t3
 | |
| --let $assert_text= Table t3 should have zero rows.
 | |
| --source include/assert.inc
 | |
| }
 | |
| if ($slave_run_triggers_for_rbr == 'YES') {
 | |
| --echo #
 | |
| --echo # Count must be 2
 | |
| --echo #
 | |
| --let $assert_cond= COUNT(*) = 2 FROM t3
 | |
| --let $assert_text= Table t3 should have two rows.
 | |
| --source include/assert.inc
 | |
| }
 | |
| 
 | |
| --connection master
 | |
| DELETE FROM t1 WHERE id=2;
 | |
| --echo # t1: Should have one row
 | |
| SELECT * FROM t1;
 | |
| --echo # t2: Should have two rows
 | |
| SELECT * FROM t2;
 | |
| --sync_slave_with_master
 | |
| --echo # t1: Should have one row
 | |
| SELECT * FROM t1;
 | |
| --echo # t2: Should have one row on slave due to ON DELETE CASCASE
 | |
| SELECT * FROM t2;
 | |
| SET GLOBAL slave_run_triggers_for_rbr= @save_slave_run_triggers_for_rbr;
 | |
| 
 | |
| --connection master
 | |
| DROP TABLE t3,t2,t1;
 | |
| --sync_slave_with_master
 | |
| #END OF
 | |
| PROCEDURE
 | |
| 
 | |
| --echo #
 | |
| --echo # Test case 8: Triggers and Foreign Keys are present only on slave and
 | |
| --echo #              'slave_run_triggers_for_rbr=NO'
 | |
| --echo #
 | |
| --let $slave_run_triggers_for_rbr=NO
 | |
| --source $MYSQLTEST_VARDIR/tmp/trig_fk_on_slave.inc
 | |
| 
 | |
| --echo #
 | |
| --echo # Test case 9: Triggers are Foreign Keys are present only on slave and
 | |
| --echo #              'slave_run_triggers_for_rbr=YES'
 | |
| --echo #
 | |
| --let $slave_run_triggers_for_rbr=YES
 | |
| --source $MYSQLTEST_VARDIR/tmp/trig_fk_on_slave.inc
 | |
| --remove_file $MYSQLTEST_VARDIR/tmp/trig_fk_on_slave.inc
 | |
| 
 | |
| --source include/rpl_end.inc
 |