2023年11月2日 星期四

Impdp halted due to a fatal error , 出現錯誤

 

impdp 出現  error , process 直接跳掉, 且系統log也沒有出現錯誤


. . exported "DBA"."DBA_RECORD_S1":"SYS_P495"      18.14 KB       3 rows

. . exported "DBA"."DBA_RECORD_S1":"SYS_P496"      30.74 KB      81 rows

. . exported "DBA"."DBA_RECORD_S1":"SYS_P497"     16.91 KB       2 rows

. . exported "DBA"."DBA_RECORD_S1":"SYS_P498"      29.72 KB      73 rows


The job "SYSTEM". "SYS_IMPORT_SCHEMA_01" was halted due to a fatal error at 11:52:41



這次的錯誤並沒有明顯的錯誤訊息, impdp 直接被中斷中止, 先查一下 系統跟alert.log 並沒有相關錯誤, 記憶體跟空間都足夠


其實可以去看 trc 看看 process 錯誤 是什麼, 但 trc, log  檔案數量太多 , 無法一一去查看, 時間也不夠


看impdp log 是直接斷在 imp table 中間, 就直接先判斷 是某張table 出現問題導致無法匯入, 並考慮 EXCLUDE 參數。


要是你Table 數量過多無法直接排除的話, 可以先參考 expdp log , 因為 expdp impdp 匯出導入都是按照 table size 順序處理, 就可以知道 impdp 是停在哪一個table


impdp system/ATTIdYPtyMt77uLY DIRECTORY=DBBACKUP \

DUMPFILE=expdp_dba01.dmp LOGFILE=expdp_dba01.log \

EXCLUDE=TABLE:\"LIKE \'DBA_TEST_OTHER\%\'\"



排除後就可正常導入












2023年5月25日 星期四

oracle 19c 清除 SYS.WRI$_ADV_OBJECTS


因為在12.2中,引入了新的特性:optimizer statistics advisor。

優化器統計信息顧問每天都會在維護窗口運行,auto_stats_advisor_task多次運行,

會消耗大量sysaux表空間,要是你有大量 partition table 這個會更消耗空間。


WRI$_ADV_OBJECTS為 AUTO_STATS_ADVISOR_TASK 或INDIVIDUAL_STATS_ADVISOR_TASK 保留的大量舊記錄會導致佔據大量 SYSAUX 空間。



AUTO_STATS_ADVISOR_TASK 用於自動統計顧問任務

INDIVIDUAL_STATS_ADVISOR_TASK 用於手動統計顧問任務



可以透過 oracle 自動排成去清掉舊資料,這邊可以看到預設的舊資料天數是多少


select TASK_NAME,parameter_name, parameter_value FROM DBA_ADVISOR_PARAMETERS WHERE task_name='AUTO_STATS_ADVISOR_TASK' and PARAMETER_NAME='EXECUTION_DAYS_TO_EXPIRE';


把舊資料修改為10天

EXEC DBMS_ADVISOR.SET_TASK_PARAMETER(task_name=> 'AUTO_STATS_ADVISOR_TASK', parameter=> 'EXECUTION_DAYS_TO_EXPIRE', value => 10);


自動刪除: 正常情況下AUTO_STATS_ADVISOR_TASK 的 EXECUTION_DAYS_TO_EXPIRE 參數設置為 30(預設情況下)。因此,超過 30 天的舊記錄被標記為過期。

手動刪除: 可以使用以下命令手動清除過期的記錄,而不是依賴自動。這將清除超過 30 天保留期的舊記錄。


手動刪除指令:

exec prvt_advisor.delete_expired_tasks;


要是筆數已經太多會產生過多的undo , 可以直接truncate 

*參考SYSAUX Tablespace Grows Rapidly After Upgrading Database to 12.2.0.1 or Above Due To Statistics Advisor (Doc ID 2305512.1)



看一下 等等要刪除的筆數

SELECT TASK_NAME, COUNT(*) CNT FROM DBA_ADVISOR_OBJECTS GROUP BY TASK_NAME ORDER BY CNT DESC;


把要保留的資料先備分

CREATE TABLE WRI$_ADV_OBJECTS_NEW AS  SELECT * FROM WRI$_ADV_OBJECTS WHERE TASK_ID !=(SELECT DISTINCT ID FROM WRI$_ADV_TASKS WHERE NAME='AUTO_STATS_ADVISOR_TASK'); SQL> SELECT COUNT(*) FROM WRI$_ADV_OBJECTS_NEW;

 

刪除

TRUNCATE TABLE WRI$_ADV_OBJECTS;

 

還原 

INSERT INTO WRI$_ADV_OBJECTS (ID,

                              TYPE,

                              TASK_ID,

                              EXEC_NAME,

                              ATTR1,

                              ATTR2,

                              ATTR3,

                              ATTR4,

                              ATTR5,

                              ATTR6,

                              ATTR7,

                              ATTR8,

                              ATTR9,

                              ATTR10,

                              ATTR11,

                              ATTR12,

                              ATTR13,

                              ATTR14,

                              ATTR15,

                              ATTR16,

                              ATTR17,

                              ATTR18,

                              ATTR19,

                              ATTR20,

                              OTHER,

                              SPARE_N1,

                              SPARE_N2,

                              SPARE_N3,

                              SPARE_N4,

                              SPARE_C1,

                              SPARE_C2,

                              SPARE_C3,

                              SPARE_C4)

   SELECT ID,

          TYPE,

          TASK_ID,

          EXEC_NAME,

          ATTR1,

          ATTR2,

          ATTR3,

          ATTR4,

          ATTR5,

          ATTR6,

          ATTR7,

          ATTR8,

          ATTR9,

          ATTR10,

          ATTR11,

          ATTR12,

          ATTR13,

          ATTR14,

          ATTR15,

          ATTR16,

          ATTR17,

          ATTR18,

          ATTR19,

          ATTR20,

          OTHER,

          SPARE_N1,

          SPARE_N2,

          SPARE_N3,

          SPARE_N4,

          SPARE_C1,

          SPARE_C2,

          SPARE_C3,

          SPARE_C4

     FROM WRI$_ADV_OBJECTS_NEW;

commit; 

 

index rebuild 

 ALTER INDEX WRI$_ADV_OBJECTS_IDX_01 REBUILD online ;

 ALTER INDEX WRI$_ADV_OBJECTS_PK REBUILD  online ;







2022年10月25日 星期二

Oracle 19c lsnrctl 監聽器無法動態註冊

 

該主機更改了 hostname及ip , 且也一併修正 /etc/hosts 、tnsnames.ora及listener.ora 內的設定

確定都是新IP新的hostname, 但在重啟時DB卻不會動態註冊, 使用靜態註冊是可以



但因為這是生產要使用的DB,還是想要把它修好


在想可能是 local_listener 參數問題 


這邊先測試一下問題來源


把值設默認, 重新動態註冊就可以了

alter system set local_listener='';  

alter system register;


表示真的是 local_listener 問題 重新設定 local_listener 看看 

設定跟 tnsnames.ora 一樣



alter system set local_listener='LISTENER_ORA';


出現錯誤

*

ERROR at line 1:

ORA-32017: failure in updating SPFILE

ORA-00119: invalid specification for system parameter LOCAL_LISTENER

ORA-00141: all addresses specified for parameter LOCAL_LISTENER are invalid

ORA-00132: syntax error or unresolved network name 'LISTENER_ORA'


重新檢查 tnsnames.ora 檔案  , 發現有多一個符號, 哈 烏龍





清除完, 重啟完成






2022年5月17日 星期二

data guard 備庫切成主庫

 

-------------- 備切換成主庫,  在備庫上操作

--停止同步

alter database recover managed standby database disconnect from session;

--查看狀態

select database_role, switchover_status from v$database;

select open_mode from v$database;


---

在備庫 failover 操作

SQL > ALTER DATABASE RECOVER MANAGED STANDBY DATABASE FINISH FORCE;


---假如 主庫已死, 可以直接操作 . 備庫直接切換成主庫

SQL> ALTER DATABASE COMMIT TO SWITCHOVER TO PRIMARY;



這時要注意一下, spfile , pfile 的 內容需不需要改變


---- 備庫可以上下一次 看看能否正常

SQL> SHUTDOWN IMMEDIATE;

SQL> STARTUP;


此時備庫






備庫 spfile 沒變




2022年1月6日 星期四

Postgres Logical Replication 邏輯複寫 10 to 14

一.邏輯複寫 介紹

邏輯複寫(Logical Replication)
是一種依據複寫指標(通常是主鍵)複製資料物件及其更新的方法。

我們使用術語邏輯與物理複寫相對比,
物理複寫使用確切的區塊位址進行每一個字元組的複寫。
所以 subscribe 訂閱方 是真的會有資料在上頭, 可建立index 


邏輯複寫使用 publish 發布(來源)和 subscribe 訂閱(目標)模式,
你把它想成 publish 是主,subscribe 是備
一個 publish 可以多個 subscribe 


邏輯複寫設定好後,發佈者的變化就會即時發送給訂閱者。
訂閱者按照與發佈者所執行的SQL順序依序執行SQL
以確保單個訂閱內的發佈的交易事務一致性。
這種資料複寫方法有時被稱為交易事務複寫。

(有點像mysql 的那種主從設定, 所以會有資料衝突發生)


邏輯複寫  是從 Postgres 10以上開始支援 , 通常使用的時機為

1.將多個資料庫合併成一個資料庫(例如為了分析的需求)。
2.在不同的 PostgreSQL 版本之間複寫。
3.在不同平台(例如 Linux 到 Windows)上的 PostgreSQL 伺服器之間的複寫
4.將複寫的資料給予不同的使用者群組存取權限。
5.在多個資料庫之間共享資料庫的一部份。


所以說 想要升級Postgres版本也可以使用 Logical Replication 做資料移轉的動作
downtime 會比使用 pg_upgrade 更短


以下是 10 (publish ) to 14  (subscribe)







postgres=# select version();
                                                 version                                                  
----------------------------------------------------------------------------------------------------------
 PostgreSQL 10.19 on x86_64-pc-linux-gnu, compiled by gcc (GCC) 4.8.5 20150623 (Red Hat 4.8.5-44), 64-bit
(1 row)

postgres=# 



二.publish 配置相關設定

---- publish 主庫新增同步用帳號

CREATE USER logical_repl REPLICATION LOGIN CONNECTION LIMIT 8 ENCRYPTED PASSWORD 'logical_repl'; 

---- 可以不需要SUPERUSER,但要logical_repl帳號對這些表有Select授權


----修改 postgresql.conf

wal_level = logical
max_wal_senders = 10
max_replication_slots = 8


----修改 pg_hba.conf

host    all             logical_repl    10.10.10.2/32           md5




------ 只同步指定的表

 CREATE PUBLICATION pub1 FOR TABLE dba_test; 
 
 CREATE PUBLICATION pub1 FOR TABLE dba_test WITH (publish = 'insert, update, delete');

 
---- 同步全表
 CREATE PUBLICATION pub1 FOR ALL TABLES ;


--- 授權
GRANT USAGE ON SCHEMA public TO logical_repl;
GRANT SELECT ON dba_test TO logical_repl; 


 ---  同步狀態
   SELECT * FROM pg_publication;


dba=# 
dba=# SELECT * FROM pg_publication;
 pubname | pubowner | puballtables | pubinsert | pubupdate | pubdelete 
---------+----------+--------------+-----------+-----------+-----------
 pub1    | 18397765 | f            | t         | t         | t
(1 row)

dba=# 
dba=# 


pubname: 指釋出的名稱
pubowner: 指釋出的屬主,可以和pg_user檢視的usesysid欄位關聯查詢得到屬主具體資訊
puballtables:是否釋出資料庫中的所有表,t表示釋出資料庫中所有已存在的表和以後新建的表
pubinsert: t表示僅釋出表上的INSERT操作
pubupdate: t表示僅釋出表上的UPDATE操作
pubdelete: t表示僅釋出表上的DELETE操作




 ------   查詢 訂閱的表
  SELECT * FROM pg_publication_tables;
  

dba=# SELECT * FROM pg_publication_tables;
 pubname | schemaname | tablename 
---------+------------+-----------
 pub1    | public     | dba_test
(1 row)

dba=# 
dba=# 




三.subscribe 配置相關設定





dba=# select version();
                                                 version                                                 
---------------------------------------------------------------------------------------------------------
 PostgreSQL 14.1 on x86_64-pc-linux-gnu, compiled by gcc (GCC) 4.8.5 20150623 (Red Hat 4.8.5-44), 64-bit
(1 row)

dba=# 
dba=# 
dba=# 


----修改 postgresql.conf

wal_level = logical
max_replication_slots = 8
max_logical_replication_workers = 8


** 備庫不一定要設定  wal_level = logical , 只是這邊預防先做設定
** max_logical_replication_workers  設定邏輯複製程序數,應大於訂閱節點的數量,並且給表同步預留一些程序數量



這邊 subscribe 已經有建立與 publish  相同的table 


 
------------------- 在備庫上 開啟 訂閱同步

CREATE SUBSCRIPTION sub1 CONNECTION 'host=10.10.10.1 port=5432user=logical_repl dbname=dba password=logical_repl' PUBLICATION pub1;





---- 測試 , 只要在主庫 publish   新增資料 ,備庫 subscribe  就會立刻出現資料



--- 查詢目前在 subscribe 備庫 已經有設定的資訊

dba=# 
dba=# SELECT * FROM pg_subscription;

  oid  | subdbid | subname | subowner | subenabled | subbinary | substream |                                  subconninfo                                   | subslotname | subsynccommit | subpublications 
-------+---------+---------+----------+------------+-----------+-----------+--------------------------------------------------------------------------------+-------------+---------------+-----------------
 16464 |   16452 | sub2    |       10 | t          | f         | f         | host=10.10.10.1 port=5432user=logical_repl dbname=dba password=logical_repl | sub2        | off           | {pub1}
(1 rows)

dba=# 

要特別注意 這邊有出現 logical_repl  的密碼 @@"


--- 查詢目前在 subscribe 備庫,同步的狀態


dba=# 
dba=# select * from pg_stat_subscription ;

 subid | subname |  pid  | relid | received_lsn  |      last_msg_send_time       |     last_msg_receipt_time     | latest_end_lsn |        latest_end_time        
-------+---------+-------+-------+---------------+-------------------------------+-------------------------------+----------------+-------------------------------
 16464 | sub2    | 20693 |       | 57BC/3894A220 | 2022-01-07 12:07:01.734512+08 | 2022-01-07 12:06:36.672786+08 | 57BC/3894A220  | 2022-01-07 12:07:01.734512+08
(1 rows)

dba=# 

包含最後的 lsn 及 最後一次同步時間 ,可以看出它有沒有在正常同步




四.未來要新增同步的表



 --- 主庫後續新增 要同步的表
 
 GRANT SELECT ON dba_test2 TO logical_repl; 
 ALTER PUBLICATION pub1 ADD TABLE dba_test2 ; 
 
 
 ----- 新增完後 ,subscribe  備庫要在執行指令 更新資料
 
 ALTER SUBSCRIPTION sub1 REFRESH PUBLICATION;


四.衝突解決

subscribe 其實是 write read 狀態下, 所以可以對同步過來的 table 進行異動
當同步中 subscribe 有出現 違反任何限制 pk uni key constraints , 同步就會停止

但當同步 UPDATE 或 DELETE 操作時 , 但subscribe 沒有該資料時, 並不會產生衝突,而這些操作將會簡單地跳過。


衝突會產生錯誤並停止複寫;它必須由使用者手動解決。
有關衝突的詳細訊息可以在使用者的伺服器日誌中找到。


解決方案有二

1.跳過與現有資料衝突的交易事務

透過呼叫 pg_replication_origin_advance() 函數以及與訂閱名稱和位置相對應的 node_name,可以跳過該交易事務。可以在 pg_replication_origin_status 系統檢視表中看到開始的目前位置。


2.修正subscribe 方table內的資料

---- 先關閉 訂閱 ,   在備庫 先刪除資料  然後再開啟訂閱

alter subscription sub1 disable;

delete from t1 where id=3;

alter subscription sub1 enable ;


類似這種處理方式




五.限制及備注


1.資料庫結構和 DDL 指令不會被複寫。後續的結構變更需要手動保持同步。(但是,請注意,兩側的結構沒有必要完全相同。)當主要資料庫中的結構定義變更時,邏輯複寫是沒問題的:當發佈者上的結構産生變更並且複寫的資料開始到達訂閱戶但不符合資料表結構,複寫將産生錯誤,直到結構更新。在許多情況下,可以透過先將預定的架構變更套用於訂閱戶來避免間歇性的錯誤。


2.支援同步 TRUNCATE 指令,使用必須格外小心。


3. TABLE(包括 PARTITION TABLE)支援同步。
其他例如檢視表 VIEW,具體化檢視圖 MATERIALIZED VIEW 或外部資料表 FOREIGN TABLE,將會導致錯誤。


4.初期快照:執行同步指令後,table會先進行一次完整的資料同步,同步完後就會進行同步模式

初次同步subscribe 方會額外出現 process 開始同步

postgres: logical replication worker for subscription 16405 sync 16468




5.由於subscribe方是真實的table,可以額外新增index及數據修改








2021年2月8日 星期一

mysql 誤刪資料舊回

mysql 誤刪資料舊回


mysql 版本 :8.0.23


這邊使用 binlog 一筆一筆查看然後把資料找出, 刪除大量的資料時就不太適合了


工具是 mysql 內建的 mysqlbinlog


mysqlbinlog --no-defaults --base64-output=decode-rows -v -v /mysql/binlog.000027 | grep -A 200 'account_table' > record_sql.txt


輸入關鍵字找回


cat  record_sql.txt


### DELETE FROM `ab`.`account_table`

### WHERE

###   @1='73057daa246f46f388205c3ce86baab5' /* VARSTRING(200) meta=200 nullable=0 is_null=0 */

###   @2='87fda75a85b24900b2224a4e8948818a' /* VARSTRING(200) meta=200 nullable=1 is_null=0 */

###   @3='af374573bcc041559d68362d5cf399a9' /* VARSTRING(200) meta=200 nullable=1 is_null=0 */

###   @4='bb4d95daa4224e14a9e80c2fb14cd829' /* VARSTRING(200) meta=200 nullable=1 is_null=0 */

###   @5=1 /* INT meta=0 nullable=1 is_null=0 */

###   @6=1 /* INT meta=0 nullable=1 is_null=0 */

###   @7=1 /* TINYINT meta=0 nullable=1 is_null=0 */

###   @8='2020-12-23 17:08:34' /* DATETIME(0) meta=0 nullable=1 is_null=0 */

###   @9='81412b09582643daa8b357f6c94cd698' /* VARSTRING(200) meta=200 nullable=1 is_null=0 */

###   @10=NULL /* DATETIME(0) meta=0 nullable=1 is_null=1 */

###   @11=NULL /* VARSTRING(200) meta=200 nullable=1 is_null=1 */

###   @12=0 /* TINYINT meta=0 nullable=1 is_null=0 */



透過這種方式 找回被誤刪的資料

2020年10月13日 星期二

mysql 備庫 同步停止 , Waiting for table metadata lock

 

主庫執行 truncate table 指令

這時備庫卻停止同步

show slave status\G;

出現

Seconds_Behind_Master: 1080

Slave_SQL_Running_State: Waiting for table metadata lock


表示備庫這邊有卡住  , 備庫的 show slave status 的 Seconds_Behind_Master 越來越多

查詢

select * from information_schema.innodb_trx

看到備庫是否還有未完成的session , kill session 就可以了


恢復完成,  變成

Seconds_Behind_Master: 0

Slave_SQL_Running_State: Slave has read all relay log; waiting for more updates