1. 程式人生 > 資料庫 >mysql 如何動態修改複製過濾器

mysql 如何動態修改複製過濾器

MySQL動態修改複製過濾器

說說今天遇到的問題吧,今天在處理一個業務方的需求,比較變態,我大概描述一下:

1、線上的阿里雲rds上面有個遊戲的日誌庫,裡面的表都是日表的形式,資料量比較大了,每次備份的時候,都會導致線上的rds報警,報警內容是IO資源佔用過多。

2、這個rds上有一個本地的ECS只讀從庫,這個只讀從庫會實時同步線上的rds資料庫中的資料,這個只讀從庫供業務方查詢使用

3、業務方說這些資料都還有用,只讀從庫上的資料必須有,線上rds上的資料可以刪除,保留兩個星期即可。

場景就是這麼個場景,DBA想要解決報警這個問題,業務方想要保證擁有完整的資料。請問,怎麼解決?

當時看到這個問題,我想罵人,這需求一看就不合理,哪兒有刪除一個庫,另外一個庫上還保留的道理,況且都是些日誌資料,不直接搞個冷備份,然後刪除線上,搞這麼一出幹啥啊。但是啊,怎麼說也沒有緩和的餘地,於是就開始思考這個問題應該怎麼解決。我想到的解決辦法有以下幾個:

1、擴容,提升效能。資料量大,擴磁碟唄,IO使用率高,提升效能麼,這是最直接的解決辦法,也是最貴的解決辦法,首先被砍掉。

2、先備份再刪除再還原。rds主庫上提前備份日表資料,然後刪除資料,此時從庫會同步刪除資料,然後再將第一步備份的資料還原到從庫上。這個辦法從可行性上來講是可以的,因為保證了沒有資料丟失。但是操作起來比較麻煩,手續太多,不夠方便。

3、使用replicate-ignore-table引數進行對於指定的表進行過濾。設定了這個引數,可以讓你過濾指定資料表的所有操作。我們看看官方文件對這個引數的描述,給個連結:https://dev.mysql.com/doc/refman/5.7/en/replication-options-slave.html#option_mysqld_replicate-wild-ignore-table

描述如下:

Creates a replication filter which keeps the slave thread from replicating a statement in which any table matches the given wildcard pattern. To specify more thanone table to ignore,use this option multiple times,

上面的意思是你可以使用這個引數建立一個過濾器,從而過濾掉匹配你制定的規則的特定表的操作(聽著很繞口),就是說你可以制定過濾規則,加入規則中制定了表a,那麼表a的操作就不會同步到從庫中了。

這和我們的需求符合,也就是我們如果設定了要過濾的表,那麼當我們進行刪除表操作的時候,從庫中不會對錶進行刪除,就實現了我們想要的結果。測試一下這個功能吧:

首先我們建立資料庫test_ignore,然後在其中建立表:

主庫上操作:

mysql:test_ignore>>showtables;
Emptyset(0.00sec)

mysql:test_ignore>>createtableaaa(idintnotnull);
QueryOK,0rowsaffected(0.19sec)

mysql:test_ignore>>createtableaab(idintnotnull);
QueryOK,0rowsaffected(0.01sec)

mysql:test_ignore>>createtableaac(idintnotnull);
QueryOK,0rowsaffected(0.00sec)

mysql:test_ignore>>createtableaad(idintnotnull);
QueryOK,0rowsaffected(0.01sec)

mysql:test_ignore>>createtableaae(idintnotnull);
QueryOK,0rowsaffected(0.01sec)

從庫上檢視:

mysql:test_ignore>>showtables;
+-----------------------+
|Tables_in_test_ignore|
+-----------------------+
|aaa|
|aab|
|aac|
|aad|
|aae|
+-----------------------+
5rowsinset(0.00sec)

發現已經同步過來了。此時是處於主從同步狀態,如果現在我們在主庫上刪除表,那麼從庫上的表一定會刪除,這不是我們想要的結果。

很顯然,接下來的一步是配置replicate-wild-ignore-table這個引數了,一般情況下,我們需要通過停止從庫的服務進行my.cnf檔案的配置,如果我們要配置多個表,則需要在my.cnf檔案中寫多條通配的記錄。例如,在本例子中,需要配置該引數的值為test_ignore.aa%,其中%代表萬用字元,也就是說,test_ignore資料庫中形如aa%這種格式的表操作都會被過濾掉。而我們建立的表aaa、aab、aac、aad、aae都是形如這種的,所以針對這幾個表的操作一定不會同步到從庫了,我們測試一下:

首先檢視當前的複製狀態:

mysql 如何動態修改複製過濾器

雙Yes狀態,說明覆制關係沒有問題

主庫進行操作:

mysql:test_ignore>>droptableaaa;
QueryOK,0rowsaffected(0.01sec)

mysql:test_ignore>>droptableaab;
QueryOK,0rowsaffected(0.00sec)

從庫上進行檢視:

mysql:test_ignore>>showtables;
+-----------------------+
|Tables_in_test_ignore|
+-----------------------+
|aaa|
|aab|
|aac|
|aad|
|aae|
+-----------------------+
5rowsinset(0.00sec)

從庫上的表還在,說明主庫上的操作沒有被同步到從庫,我們配置的引數

replicate-wild-ignore-table=test_ignore.aa%

起作用了。此時,如果我們在主庫上建立一個表:

`主庫`
mysql:test_ignore>>createtableaaf(idint);
QueryOK,0rowsaffected(0.00sec)

`從庫`
mysql:test_ignore>>showtables;
+-----------------------+
|Tables_in_test_ignore|
+-----------------------+
|aaa|
|aab|
|aac|
|aad|
|aae|
+-----------------------+
5rowsinset(0.00sec)

發現從庫並沒有同步主庫的表aaf,因為aaf也匹配了test_ignore.aa%這條規則。

利用這個特性,我們能夠很好的解決這個業務場景,也就是主庫刪除,從庫保留資料。但是,這裡要說但是了,這個方法有一個比較嚴重的問題,就是每次都需要重啟從庫,如果我們需要配置第二條規則,第三條規則,則需要重啟從庫2次,3次,這個過程中,從庫對於業務方是不可見的,如果無法訪問,很可能造成程式報錯,這是我們不能忍受的。

這個過程肯定是要解決的,怎麼解決呢?能不能找到不停機就能修改複製過濾器的方法?找找官方文件。

果然,停機是不可能停機的,這輩子都不可能停機。官方文件中有這麼一句話:

You can also create such a filter by issuing aCHANGE REPLICATION FILTER REPLICATE_WILD_IGNORE_TABLEstatement.

我去,這是個啥語句,表示從來沒有用過,可以通過線上變更復制過濾器的方法來對過濾器進行修改,看看官方文件中的介紹:

mysql 如何動態修改複製過濾器

mysql 如何動態修改複製過濾器

看到了一個神奇的語句,趕緊來試試:

mysql:test_ignore>>changereplicationfilterreplicate_wild_ignore_table=('test_ig%.aa%');
ERROR3017(HY000):Thisoperationcannotbeperformedwitharunningslavesqlthread;runSTOPSLAVESQL_THREADfirst

mysql:test_ignore>>stopslave;
QueryOK,0rowsaffected(0.00sec)

mysql:test_ignore>>changereplicationfilterreplicate_wild_ignore_table=('test_ig%.aa%');
QueryOK,0rowsaffected(0.00sec)

mysql:test_ignore>>startslave;
QueryOK,0rowsaffected(0.01sec)

直接使用,提示需要stop slave sql_thread,想想也能理解,不停止複製直接修改複製的規則好像有點不妥,索性停止了整個複製,然後重新修改複製過濾器,妥了,成功執行,開啟複製,一套操作行雲流水。

再來看看複製關係中的狀態:

mysql 如何動態修改複製過濾器

忽略的表規則已經變成了test_ig%.aa%,也就是說,以test_ig開頭的資料庫中以aa開頭的表的操作,都不會被同步到從庫,包括對錶的alter和drop以及create操作。

但這裡,方案就出來了,我們知道,日表一般是YYYYMMDD這種形式的,我們只要過濾YYYYMM%這種格式的日表,然後在主庫上對它進行刪除,這個操作將不會被同步到從庫,那麼這個問題就可以順利解決了。

當然,除了這個方案之外,還有一些方案,例如:

如果業務容忍部分資料丟失,我們還可以使用關閉binlog---刪表---開啟binlog的方式使得從庫不會同步主庫的drop操作;

線上所有的日表操作都配置成ignore,然後利用觸發器將日表中的更新同步到從庫中;

這一系列的操作,其實不是從本質上解決問題,本質上還是業務設計的問題,日表中的打點日誌太多,可以適當減少這些打點日誌,對於打點日誌,需要確定保留週期,過期的日誌,需要及時清理,保證伺服器的指標和效能。

以上就是mysql 如何動態修改複製過濾器的詳細內容,更多關於MySQL動態修改複製過濾器的資料請關注我們其它相關文章!