この記事で ターゲット ソリューション リファレンス   適用範囲: 5.6 MySQLサーババージョン4.1[5.6に4.1をリリース] この資料の情報は、すべてのプラットフォームに適用されます。   目標 Undropforinnodbでこわれたテーブルからデータを抽出する。   解決策 オープンソースUsing the opensource UnDROP tool for InnoDB from twindb.comで、時々innodb_force_recoveryで読み取ったテーブルからリカバリできる。   Undropツールは使われていないDBサーバが使っているibdataファイル。   一般的に、そのツールはibdataファイルあるいは独立したInnoDB tablespaceファイル(innodb_file_per_table使っている.ibdファイル)でインディクスページを抽出する。Blobページが利用できるサブディレクトリへ抽出する。   もしデータがインディクスページに抽出されたら、次のステップはデータディレクトリから主キー、一般的なクラスタリング索引IDをリカバリする。そして、データをLOAD DATA INFILEが使えるファイルに抽出する。   もしよければ、リカバリしたいデータベースのschema dumpで起動して、必要なときに、innodb_force_recoveryを使ってください。たとえ古いバックアップであっても、何もないよりまだましである。時にUnDROPでibdataファイルから有効なテーブル定義を抽出できる。何のバックアップもなければ、.frmファイルでテーブル定義を再作成できる。.frmもなければ、最後の手はUnDROPでidbataからテーブル定義を抽出して、せめて一部だけのデータをリカバリできる。   UnDROPツールは以下の通り。データを抽出する順で配列する: stream_parser   stream_parserはbdataからページを抽出するツールである。使い方もすごく便利である: ./stream_parser f <path_to_ibdata>   ページはディフォルトで"pages-<ibdata_file_name>"に抽出する。インディクスページはサブディスプレイに格納される。 FIL_PAGE_INDEX,且blob页被储存在子目录FIL_PAGE_TYPE_BLOB。   テーブルにすべてのデータを抽出したいであれば、テーブルの主キーのデータディレクトリインディクスIDを識別する必要がある。これはUnDROPツールの"recover_dictionary.sh"スクリプトで、抽出されたインディクスページに抽出されたディクショナリーデータはサーバの'test' schemaに移す。このように: $ ./recover_dictionary.sh Generating dictionary tables dumps... OK Creating test database ... OK Creating dictionary tables in database test: SYS_TABLES ... OK SYS_COLUMNS ... OK SYS_INDEXES ... OK SYS_FIELDS ... OK All OK Loading dictionary tables data: SYS_TABLES ... 1845 recs OK SYS_COLUMNS ... 22029 recs OK SYS_INDEXES ... 4994 recs OK SYS_FIELDS ... 6070 recs OK All OK   どんなインディクスIDもディクショナリーを検索して探し出せる。 例はmoodle2 schemaのテーブルmdl2_user: mysql> SELECT SYS_TABLES.NAME TABLE_NAME, SYS_TABLES.ID TABLE_ID, SYS_INDEXES.NAME INDEX_NAME, SYS_INDEXES.ID INDEX_ID FROM SYS_TABLES LEFT JOIN SYS_INDEXES ON SYS_TABLES.ID = SYS_INDEXES.TABLE_ID WHERE SYS_INDEXES.NAME LIKE '%PRIMARY%' AND SYS_TABLES.NAME LIKE 'moodle2/mdl2_user' AND SYS_INDEXES.NAME IN ('PRIMARY', 'GENERAL_CLUSTERED_INDEX'); +‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐+‐‐‐‐‐‐‐‐‐‐+‐‐‐‐‐‐‐ ‐‐‐‐‐+‐‐‐‐‐‐‐‐‐‐+ | TABLE_NAME | TABLE_ID | INDEX_NAME | INDEX_ID | +‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐+‐‐‐‐‐‐‐‐‐‐+‐‐‐‐‐‐‐‐‐‐‐‐+‐‐‐‐ ‐‐‐‐‐‐+ | moodle2/mdl2_user | 646 | PRIMARY | 1867 | +‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐+‐‐‐‐‐‐‐‐‐‐+‐‐‐‐‐‐‐‐‐‐‐‐ +‐‐‐‐‐‐‐‐‐‐+ 1 row in set (0.00 sec)   INDEX_IDは名1に該当する抽出ページファイルを返す: $ ls pages‐ibdata1/FIL_PAGE_INDEX/*1867.page 2015/12/14 Document 2056049.1 https://support.oracle.com/epmos/faces/DocumentDisplay?_adf.ctrlstate= scibdopz6_382&id=2056049.1 3/6 pages‐ibdata1/FIL_PAGE_INDEX/0000000000001867.page   c_parser もし、テーブル定義があれば、データがc_parserでリカバリできる。このように、mdl2_user.sqlにはテーブル定義を含んでいる: $ ./c_parser ‐b "./pages‐ibdata1/FIL_PAGE_TYPE_BLOB" ‐p "dumps/moodle2" ‐l dumps/moodle/mdl2_user.load ‐5f pagesibdata1/ FIL_PAGE_INDEX/0000000000001867.page ‐t mdl2_user.sql   各schemaに対して、以下のスクリプトとダンプディレクトリのテーブル定義で、データディレクトリで見つけ出したテーブルを抽出して、LOAD DATA INFILEにふさわしいファイルにロードする。もし、もとのテーブル定義がないあるいは.frmファイルで抽出出来ない場合に、最後の手として、sys_parser行をアンチノートしてください。元のテーブル定義があれば 分けられたsqlファイルとしてdumps/<schema>/<table>.sqlに格納してください: #!/bin/bash RECOVERY_DB="test" USER="root" PASS="somepass" DUMPS="dumps" # Create schema echo > ${DUMPS}/schema.sql for DB in `mysql ‐‐user=${USER} ‐‐password=${PASS} ‐NBe "select name from ${RECOVERY_DB}.sys_tables" | sed ‐r "s/^(.*)\/.*$/\1/" | grep ‐v SYS_ | sort ‐u ` do mkdir ‐p ${DUMPS}/${DB} echo "Creating schema for $DB..." echo >> ${DUMPS}/schema.sql echo "CREATE DATABASE IF NOT EXISTS $DB;" >> ${DUMPS}/schema.sql for TABLE in `mysql ${RECOVERY_DB} ‐‐user=${USER} ‐‐password=${PASS} ‐NBe "SELECT NAME FROM SYS_TABLES WHERE NAME LIKE '${DB}/%'"` do echo $TABLE # ./sys_parser ‐u${USER} ‐p${PASS} ‐d ${RECOVERY_DB} ${TABLE} | tee ${DUMPS}/${TABLE}.sql >> ${DUMPS}/schema.sql PKEY=`mysql ${RECOVERY_DB} ‐BNe "SELECT SYS_INDEXES.ID FROM SYS_TABLES LEFT JOIN SYS_INDEXES ON (SYS_TABLES.ID = SYS_INDEXES.TABLE_ID) WHERE SYS_TABLES.NAME = \"${TABLE}\" AND SYS_INDEXES.NAME=\"PRIMARY\""` echo "pkey = $PKEY" PAGE="pages‐ibdata1/FIL_PAGE_INDEX/`printf '%016u' ${PKEY}`.page" echo "PAGE = $PAGE" ./c_parser ‐b "./pages‐ibdata1/FIL_PAGE_TYPE_BLOB" ‐p "./${DUMPS}/${DB}" ‐ l ${DUMPS}/${TABLE}.load ‐5f ${PAGE} ‐t ${DUMPS}/${TABLE}.sql > ${DUMPS}/${TABLE} 2015/12/14 Document 2056049.1 https://support.oracle.com/epmos/faces/DocumentDisplay?_adf.ctrlstate= scibdopz6_382&id=2056049.1 4/6 done done   sys_parser 前に言ってたとおりに、sys_parserはibdataファイルからテーブル定義を抽出するときに使える。けどこれはせいぜい最後の手段で、テーブル定義がなければ、ibdataから使用可能データを抽出することはありえない。 .frmファイルから有効なテーブル定義を抽出する MySQL Utilitiesパッケージのmysqlfrmで壊れていない.frmファイルから有効な定義を抽出できる。診断モードで、mysqlfrmを使うと、sys_parserでibdataからデータを抽出するときに、同じようなトラブルになるから、mysqlfrmを使うこととサーバ標識は大切である。インポートは調整されて、c_parserと運用する。 所以使用mysqlfrm与服务器标识是很重要的。输出必须被调整来与c_parser运作,コマンドライン、警告、またはデフォルトの文字セット情報は、テーブル定義で発見された場合ので、c_parserが死ぬ。これはテーブル定義を抽出し、一つのschema.sqlがディレクトリから見つけ出した.frmファイルでschema(s)を作成する方法である: #!/bin/bash for FRM in `find ../datadir/ ‐type f ‐wholename "*frm" | sort` do mysqlfrm ‐‐server=root:somepass@localhost:../datadir/mysql.sock ‐‐port=33307 $FRM | extract_schema.pl if [ ${PIPESTATUS[0]} ‐ne 0 ]; then echo "$FRM is corrupt" fi done   extract_schema.plこのように: !/usr/bin/perl open (SCHEMAFILE, '>>', "schema.sql") or die "Can not write to schema.sql $!"; $schema = ""; $table = ""; 2015/12/14 Document 2056049.1 https://support.oracle.com/epmos/faces/DocumentDisplay?_adf.ctrlstate= scibdopz6_382&id=2056049.1 5/6 while (<STDIN>) { $origline = $ _; chomp; if (/^ CREATE TABLE.*$/) { m/.*CREATE TABLE \`(. *)\`\.\`(.*)\`[[:space:]]\(/; $schema = $1; $table = $2; print "Creat ing $schema.$table\n"; unless (‐e $schema or mkdir $schem a) { die "Unable to create dir for sche ma $schema"; } p rint SCHEMAFILE "CREATE DATABASE IF NOT EXISTS $schema;\n"; print SCHEMAFILE "USE $schema;\n"; print SCHEMAFILE "$origline"; open (TABLEFILE, '>', "$schem a/$table.sql") or die "Can not write to $schema/$table.sql"; print TABLEF ILE "CREATE TABLE $table (\n"; } else { s/ENGINE=(.*?)[[:space:]].*/ENGINE=\1;/; s/`PRIMARY`//; s/^#.*$//; s/^WARNING.*$//; $origline =~ s/(.*ENGINE.*$)/\1;/; $origline =~ s/^#.*$//; $origline =~ s/^WARNING.*$//; print SCHEMAFILE "$origline" if (!/^\s*$/); print TABLEFILE "$_\n" if (!/^\s*$/); } }   幸い、UnDROPはまだ使用可能だが、これを使うこととは状況はとってもひどくなった。データベースがこわれたときに、どんなデータをリカバリする保障もない。このような状況を避けるために、MySQL Enterprise Backupで定期的に、バックアップを作成してください。そして、せめて一つのslaveをコピしてください。