Skip to content

Instantly share code, notes, and snippets.

@lf-uraku-yuki
Created November 7, 2017 04:41
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save lf-uraku-yuki/519038e8e5b87bb8c17c3feb0b03238b to your computer and use it in GitHub Desktop.
Save lf-uraku-yuki/519038e8e5b87bb8c17c3feb0b03238b to your computer and use it in GitHub Desktop.
Codeigniter 3.x Database Connect Retry Sample
<?php
function reconnectDb()
{
if ($this->db->conn_id !== false) {
return;
}
log_message('error', 'DB接続失敗。再接続開始');
// スクリプト実行可能時間を延ばしておく
ini_set('max_execution_time', '180');
for ($retry_count = 0; $retry_count < 8; $retry_count ++) {
sleep(15);
$this->load->database();
if ($this->db->conn_id !== false) {
log_message('debug', 'DB再接続処理OK');
return;
}
log_message('debug', 'DB再接続処理NG');
}
}
@lf-uraku-yuki
Copy link
Author

目的

  • CodeigniterのDB接続不可時に処理をエラーにせずなんとか再接続したい。

前提・用途

  • DBに繋がらないからといってクラッシュされては困るようなクリティカルな処理が有る。
  • Amazon RDSのマルチAZ等のように、冗長化されてはいるが数秒~2分程度のダウンタイムは発生するような構成であり、それに耐えたい。
  • Web APIを提供しているなど、瞬間的に大量のアクセスが発生しDBのmax_connectionを使い切るような状況が起こり得る。が、普段のアクセスは少ない。
  • Codeigniter標準のDBフェイルオーバー機能が使えるほど冗長化された構成ではない(=フェイルオーバー用のDBが無い)。
  • この記事はMySQLにmysqliを使って接続する前提になっています。

注意点

  • あくまでDBへの再接続チャンスを得るための処理なので万能ではありません。上記例の場合、15秒ごとの8回のリトライに失敗すればDBには接続できないままです。
  • かなりの時間を待たせる処理になるので、クリティカルな処理、バッチ的な処理、API的な処理でのみ保険的に利用するべきでしょう。
  • バッチ的な処理など、それなりに時間がかかる処理の中で何度もDBにアクセスするような場合は、最初に接続確認を行っても途中でDBがダウンして接続出来なくなるような事も考えられます。$this->db->recconect() メソッドで conn_id プロパティを更新したうえで再度、再接続処理を実行する事もできるはずですが、運悪くDB処理の真っ最中にダウンしたような場合に対しては無力です。
  • CodeigniterのDB接続設定(database.php)にて db_debug を trueにしている場合、DB周りでのError・Warningはそのまま処理の即時停止となり、この再接続処理にたどり着きません。db_debug を false にしている環境でのみ動作します。
  • 上記と同様にディスプレイにPHPのエラーを表示する設定も、DB接続失敗時にWarningが発生しているため非推奨です。

@lf-uraku-yuki
Copy link
Author

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment