売上集計をリアルタイムでなく、バッチモードで行うとタイムアウトしないという
ただのタイムアウト回避方法。
デフォルトはリアルタイム集計法。
リアルタイムで集計したいのもわかるけど。
データが大量だとタイムアウトが発生する。
商品別集計したときの画面イメージはこんな感じ。
ちなみに、リアルタイムは
1. 「月度で集計する」や「期間で集計する」ボタンが押されたときにその場で集計→集計データ格納→集計データ取得→表示、と行う方法。
バッチモードだと
2. あらかじめ売上データを集計しておき、「月度で集計する」や「期間で集計する」ボタンが押されたときには、すでに集計して格納済みのデータを取得→表示、と行う方法。
グラフ表示も同じロジックを通るため、これまたタイムアウトで表示できなかったり。
というわけで。
先にバッチモードで集計しておいたデータを表示するようにカスタマイズする。
パラメータの変更
まずは管理画面から
管理画面>システム設定>パラメータ設定>DAILY_BATCH_MODE
をfalseからtrueに変更して登録する。
売上集計バッチの作成
次に、売上集計を定期的におこなうバッチを新規作成する。
/data/script/execute_batch_daily.php
#!/usr/bin/env php
<?php
/**
* デイリーバッチ起動スクリプト
* @author y-nakajima
*/
require_once( dirname(__FILE__) . "/../install.php"); define( "CLASS_PATH", DATA_PATH
. 'class/'); define( "CLASS_EX_PATH", DATA_PATH
. 'class_extends/'); define( "CACHE_PATH" . DATA_PATH
. 'cache/');
require_once(CLASS_EX_PATH . "util_extends/GC_Utils_Ex.php");
require_once(CLASS_EX_PATH . "util_extends/SC_Utils_Ex.php");
require_once(CLASS_EX_PATH . "db_extends/SC_DB_MasterData_Ex.php");
require_once(CLASS_EX_PATH . "db_extends/SC_DB_DBFactory_Ex.php");
require_once(CLASS_PATH . "SC_DBConn.php");
require_once(CLASS_PATH . "SC_Query.php");
require_once(CLASS_PATH . "SC_SelectSql.php");
// アプリケーション初期化処理
$objInit = new SC_Initial_Ex();
$objInit->init();
$usage = <<<USAGE
php -f execute_batch_daily.php start term
集計期間を指定して売上集計します。
start 集計開始日。何日前の売上から集計するか。1から365以内で指定。
term 集計対象期間。何日分の売上を集計をするか。1から365以内で指定。
例:昨日の売上を集計します。
php -f execute_batch_daily.php 1 1
USAGE;
if( $argc != 3){
echo($usage);
}
$start = $argv[1];
$term = $argv[2];
echo($usage);
}
if( $start < 1 || $start > 365 || $term < 1 || $term > 365){
echo($usage);
}
if($term > $start ){
echo($usage);
}
$objBatch = new SC_Batch_Daily_Ex();
$objBatch->lfStartDailyTotal( $term, $start, true);
?>
バッチの修正
/data/class/batch/SC_Batch_Daily.php
のlfStartDailyTotal()を修正する。
修正したのは3点。
- キャンセル、削除になった受注詳細情報を削除しているのをコメントアウト。
- 最後のバッチ実行からLOAD_BATCH_PASS秒経過していないと実行しない。のログがうまく出力されないので修正。
- 集計のforループ
/**
* バッチ処理を実行する.
*
* @param mixed $argv コマンドライン引数
* @return void
*/
function execute($argv = "") {
$term = 0;
$start = 1; // 集計期間は、$start~$termの間となる。通常前日分から。
$command = false;
// 集計対象期間の取得(指定日分さかのぼる)
if (SC_Utils_Ex::sfIsInt($argv[1]) && $argv[1] <= 365) {
$term = $argv[1];
$command = true;
}
// 集計開始日
if (SC_Utils_Ex::sfIsInt($argv[2]) && $argv[2] <= 365) {
$start = $argv[2];
$command = true;
}
if($term > 0) {
// 集計の開始
$this->lfStartDailyTotal($term, $start, $command);
}
}
// 集計の開始
function lfStartDailyTotal($term, $start, $command = false) {
// グラフ画像の削除
$path = GRAPH_DIR . "*.png";
// 削除された受注データの受注詳細情報の削除
$objQuery = new SC_Query();
/*
EC-CUBE全体をgrepしてみたが、なぜ受注詳細情報だけを削除してしまうのか不明なので削除しないでおく。
商品別集計もdel_flg=0でキャンセルステータスではない注文から集計してるので影響はなさそう。
$where = "order_id IN (SELECT order_id FROM dtb_order WHERE del_flg = 1)";
$objQuery->delete("dtb_order_detail", $where);
*/
// 最後に更新された日付を取得
$ret = $objQuery->max("dtb_bat_order_daily", "create_date");
// 最後のバッチ実行からLOAD_BATCH_PASS秒経過していないと実行しない。
if($pass < LOAD_BATCH_PASS) {
// GC_Utils_Ex::gfPrintLog("LAST BATCH " . $arrRet[0]['create_date'] . " > " . $batch_pass . " -> EXIT BATCH $batch_date");
$batch_next = date( "Y-m-d H:i:s", strtotime($batch_last) + LOAD_BATCH_PASS
); GC_Utils_Ex::gfPrintLog("LAST BATCH " . $batch_last . " NEXT BATCH AFTER $batch_next -> EXIT BATCH");
return;
}
// 集計
// for ($i = $start; $i < $term; $i++) {
for ($i = $start; $k = 0; $i--, $k++ ) {
// 基本時間から$i日分さかのぼる
$tmp_time = $now_time - ($i * 24 * 3600);
$batch_date = date("Y/m/d", $tmp_time); GC_Utils_Ex::gfPrintLog("LOADING BATCH $batch_date");
$this->lfBatOrderDaily($tmp_time);
$this->lfBatOrderDailyHour($tmp_time);
$this->lfBatOrderAge($tmp_time);
// タイムアウトを防ぐ
SC_Utils_Ex::sfFlush();
}
}
サーバへcronをセット。
サーバ(UNIX系OS)にcronをセットする。毎晩AM2:00に前日分の売上を集計するようにするには
以下のようにする。
$ crontab -e
0 2 * * * /usr/local/bin/php -f /path/to/eccube/data/script/execute_batch_daily.php 1 1 2>&1 > /dev/null
リアルタイム集計でもタイムアウトを防ぐようにする
SC_Batch_Daily::lfRealTimeDailyTotal()
のforループ中の最後に
SC_Utils_Ex::sfFlush();
を追加した。
// リアルタイムで集計を実施する。集計が終了しているレコードは実施しない。
/*
$sdate:YYYY-MM-DD hh:mm:ss形式の日付
$edate:YYYY-MM-DD hh:mm:ss形式の日付
*/
function lfRealTimeDailyTotal($sdate, $edate) {
$loop = intval($pass / 86400);
for($i = 0; $i <= $loop; $i++) {
$tmp_time = strtotime($sdate) + ($i * 86400); $batch_date = date("Y/m/d H:i:s", $tmp_time); $objQuery = new SC_Query();
$arrRet = $objQuery->select("order_date, create_date", "dtb_bat_order_daily", "order_date = ?", array($batch_date)); // すでにバッチ処理が終了しているかチェックする。
if( $count > 0 ) {
list($create_date) = split("\.", $arrRet[0]['create_date']); list($order_date) = split("\.", $arrRet[0]['order_date']); // オーダー開始日より一日以上後に集計されている場合は集計しなおさない
if($order_time + 86400 < $create_time || $tmp_time > time()) { GC_Utils_Ex
::gfPrintLog("EXIT BATCH $batch_date $tmp_time" . " " . time()); continue;
}
}
GC_Utils_Ex::gfPrintLog("LOADING BATCH $batch_date");
$this->lfBatOrderDaily($tmp_time);
$this->lfBatOrderDailyHour($tmp_time);
$this->lfBatOrderAge($tmp_time);
// タイムアウトを防ぐ 2009.05.01 追加
SC_Utils_Ex::sfFlush();
}
}
最終更新:2009年05月01日 18:54