<?php
/*
 * Copyright © 2000-2006 LOCKON CO.,LTD. All Rights Reserved.
 *
 * http://www.lockon.co.jp/
 */
 
require_once("../require.php");
require_once("../../../data/module/Tar.php");

//ページ管理クラス
class LC_Page {
	//コンストラクタ
	function LC_Page() {
		//メインテンプレートの指定
		$this->tpl_mainpage = 'system/bkup.tpl';
		$this->tpl_subnavi = 'system/subnavi.tpl';
		$this->tpl_mainno = 'system';		
		$this->tpl_subno = 'bkup';
		$this->tpl_subtitle = 'バックアップ管理';
		
//		$this->bkup_dir = ROOT_DIR . USER_DIR . "bkup/";
		$this->bkup_dir = ROOT_DIR . "html/test/" . "bkup/";
	}
}

$objPage = new LC_Page();
$objView = new SC_AdminView();
$objQuery = new SC_Query();

// セッションクラス
$objSess = new SC_Session();
// 認証可否の判定
sfIsSuccess($objSess);

// バックアップテーブルがなければ作成する
lfCreateBkupTable();

switch($_POST['mode']) {
// バックアップを作成する
case 'bkup':
	// 入力文字列の変換
	$arrData = lfConvertParam($_POST);

	// エラーチェック
	$arrErr = lfCheckError($arrData);

	// エラーがなければバックアップ処理を行う	
	if (count($arrErr) <= 0) {
		// バックアップファイル作成
		$arrErr = lfCreateBkupData($arrData['bkup_name']);
		
		// DBにデータ更新
		if (count($arrErr) <= 0) {
			lfUpdBkupData($arrData);
		}else{
			$arrForm = $arrData;
		}
		
		$objPage->tpl_onload = "alert('バックアップ完了しました');";
	}else{
		$arrForm = $arrData;
	}

	break;
	
// リストア
case 'restore':
case 'restore_config':
	if ($_POST['mode'] == 'restore_config') {
		$objPage->mode = "restore_config";
	}

	lfRestore($_POST['list_name']);

	break;
	
// 削除
case 'del':
	// ファイルの削除
	unlink($objPage->bkup_dir.$_POST['list_name'] . ".tar.gz");

	// DBから削除
	$delsql = "DELETE FROM dtb_bkup WHERE bkup_name = ?";
	$objQuery->query($delsql, array($_POST['list_name']));

	break;
default:
	break;
}

// バックアップリストを取得する
$arrBkupList = lfGetBkupData("ORDER BY create_date DESC");

// テンプレートファイルに渡すデータをセット
$objPage->arrErr = $arrErr;
$objPage->arrForm = $arrForm;
$objPage->arrBkupList = $arrBkupList;

$objView->assignobj($objPage);		//変数をテンプレートにアサインする
$objView->display(MAIN_FRAME);		//テンプレートの出力

//-------------------------------------------------------------------------------------------------------
/* 取得文字列の変換 */
function lfConvertParam($array) {
	/*
	 *	文字列の変換
	 *	K :  「半角(ﾊﾝｶｸ)片仮名」を「全角片仮名」に変換
	 *	C :  「全角ひら仮名」を「全角かた仮名」に変換
	 *	V :  濁点付きの文字を一文字に変換。"K","H"と共に使用します	
	 *	n :  「全角」数字を「半角(ﾊﾝｶｸ)」に変換
	 *  a :  全角英数字を半角英数字に変換する
	 */
	$arrConvList['bkup_name'] = "a";
	$arrConvList['bkup_memo'] = "KVa";
	
	// 文字変換
	foreach ($arrConvList as $key => $val) {
		// POSTされてきた値のみ変換する。
		if(isset($array[$key])) {
			$array[$key] = mb_convert_kana($array[$key] ,$val);
		}
	}
	return $array;
}

// エラーチェック
function lfCheckError($array){
	$objErr = new SC_CheckError($array);
	
	$objErr->doFunc(array("バックアップ名", "bkup_name", STEXT_LEN), array("EXIST_CHECK","MAX_LENGTH_CHECK","NO_SPTAB","ALNUM_CHECK"));
	$objErr->doFunc(array("バックアップメモ", "bkup_memo", MTEXT_LEN), array("MAX_LENGTH_CHECK"));
	
	// 重複チェック
	$ret = lfGetBkupData("WHERE bkup_name = ?", array($array['bkup_name']));
	if (count($ret) > 0) {
		$objErr->arrErr['bkup_name'] = "バックアップ名が重複しています。別名を入力してください。";
	}

	return $objErr->arrErr;
}

// バックアップファイル作成
function lfCreateBkupData($bkup_name){
	global $objPage;
	$objQuery = new SC_Query();
	$csv_data = "";
	$csv_autoinc = "";
	$err = true;
	
	$bkup_dir = $objPage->bkup_dir;
	$bkup_dir = $bkup_dir . $bkup_name . "/";

	// 全テーブル取得
	$arrTableList = lfGetTableList();
	
	// 各テーブル情報を取得する
	foreach($arrTableList as $key => $val){
		
		if ($val != "dtb_bkup") {
			
			// 自動採番型の構成を取得する
			$csv_autoinc .= lfGetAutoIncrement($val);
			
			// 全データを取得
			if ($val == "dtb_pagelayout"){
				$arrData = $objQuery->getAll("SELECT * FROM $val ORDER BY page_id");
			}else{
				$arrData = $objQuery->getAll("SELECT * FROM $val");
			}
			
			// CSVデータ生成
			if (count($arrData) > 0) {
				
				// カラムをCSV形式に整える
				$arrKyes = sfGetCommaList(array_keys($arrData[0]), false);
				
				// データをCSV形式に整える
				$data = "";
				foreach($arrData as $data_key => $data_val){
					$data .= lfGetCSVList($arrData[$data_key]);
				}
				// CSV出力データ生成
				$csv_data .= $val . "\n";
				$csv_data .= $arrKyes . "\n";
				$csv_data .= $data;
				$csv_data .= "\n";
			}	
		}
	}

	$csv_file = $bkup_dir . "bkup_data.csv";
	$csv_autoinc_file = $bkup_dir . "autoinc_data.csv";
	// CSV出力
	// ディレクトリが存在していなければ作成する		
	if (!is_dir(dirname($csv_file))) {
		$err = mkdir(dirname($csv_file));
	}
	if ($err) {
		// dataをCSV出力
		$fp = fopen($csv_file,"w");
		if($fp) {
			if($csv_data != ""){
				$err = fwrite($fp, $csv_data);
			}
			fclose($fp);
		}
		
		// 自動採番をCSV出力
		$fp = fopen($csv_autoinc_file,"w");
		if($fp) {
			if($csv_autoinc != ""){
				$err = fwrite($fp, $csv_autoinc);
			}
			fclose($fp);
		}
	}

	// 各種ファイルコピー
	if ($err) {
		// 商品画像ファイルをコピー
		// ディレクトリが存在していなければ作成する		
		$image_dir = $bkup_dir . "save_image/";
		if (!is_dir(dirname($image_dir))) $err = mkdir(dirname($image_dir));		
		$copy_mess = "";
		$copy_mess = sfCopyDir("../../upload/save_image/",$image_dir, $copy_mess);
		
		// テンプレートファイルをコピー
		// ディレクトリが存在していなければ作成する		
		$templates_dir = $bkup_dir . "templates/";
		if (!is_dir(dirname($templates_dir))) $err = mkdir(dirname($templates_dir));		
		$copy_mess = "";
		$copy_mess = sfCopyDir("../../user_data/templates/",$templates_dir, $copy_mess);
		
		// インクルードファイルをコピー
		// ディレクトリが存在していなければ作成する		
		$inc_dir = $bkup_dir . "include/";
		if (!is_dir(dirname($inc_dir))) $err = mkdir(dirname($inc_dir));		
		$copy_mess = "";
		$copy_mess = sfCopyDir("../../user_data/include/",$inc_dir, $copy_mess);
	
		// CSSファイルをコピー
		// ディレクトリが存在していなければ作成する		
		$css_dir = $bkup_dir . "css/";
		if (!is_dir(dirname($css_dir))) $err = mkdir(dirname($css_dir));		
		$copy_mess = "";
		$copy_mess = sfCopyDir("../../user_data/css/",$css_dir, $copy_mess);

		//圧縮フラグTRUEはgzip圧縮をおこなう
		$tar = new Archive_Tar($objPage->bkup_dir . $bkup_name.".tar.gz", TRUE);

		//bkupフォルダに移動する
		chdir($objPage->bkup_dir);

		//圧縮をおこなう
		$zip = $tar->create("./" . $bkup_name . "/");

		// バックアップデータの削除
		if ($zip) sfDelFile($bkup_dir);
	}

	if (!$err) {
		$arrErr['bkup_name'] = "バックアップに失敗しました。";
		// バックアップデータの削除
		sfDelFile($bkup_dir);
	}
	
	return $arrErr;
}

/* 配列の要素をCSVフォーマットで出力する。*/
function lfGetCSVList($array) {
	if (count($array) > 0) {
		foreach($array as $key => $val) {
			if ($val == "") {
				$line .= "'',";
			}else{
				$line .= "'".$val."',";
			}
		}
		$line = ereg_replace(",$", "\n", $line);
		return $line;
	}else{
		return false;
	}
}

// 全テーブルリストを取得する
function lfGetTableList(){
	$objQuery = new SC_Query();
	
	if(DB_TYPE == "pgsql"){
		$sql = "SELECT tablename FROM pg_tables WHERE tableowner = ? ORDER BY tablename ; ";
		$arrRet = $objQuery->getAll($sql, array(DB_USER));
		$arrRet = sfSwapArray($arrRet);
		$arrRet = $arrRet['tablename'];
	}else if(DB_TYPE == "mysql"){
		$sql = "SHOW TABLES;";
		$arrRet = $objQuery->getAll($sql);
		$arrRet = sfSwapArray($arrRet);
		$arrRet = $arrRet['Tables_in_eccube_db'];
	}
	
	return $arrRet;
}

// 自動採番型をCSV出力形式に変換する
function lfGetAutoIncrement($table_name){
	$arrColList = lfGetColumnList($table_name);
	$ret = "";
	
	if(DB_TYPE == "pgsql"){
		$match = 'nextval(\'';
	}else if(DB_TYPE == "mysql"){
		$match = "auto_incr";
	}

	foreach($arrColList['col_def'] as $key => $val){
		
		if (substr($val,0,9) == $match) {
			$col = $arrColList['col_name'][$key];
			$autoVal = lfGetAutoIncrementVal($table_name, $col);
			$ret .= "$table_name,$col,$autoVal\n";
		}
	}
	
	return $ret;
}

// テーブル構成を取得する
function lfGetColumnList($table_name){
	$objQuery = new SC_Query();

	if(DB_TYPE == "pgsql"){
		$sql = "SELECT 
					a.attname, t.typname, a.attnotnull, d.adsrc as defval, a.atttypmod,	a.attnum as fldnum,	e.description 
				FROM 
					pg_class c,
					pg_type t,
					pg_attribute a left join pg_attrdef d on (a.attrelid=d.adrelid and a.attnum=d.adnum)
								   left join pg_description e on (a.attrelid=e.objoid and a.attnum=e.objsubid)
				WHERE (c.relname=?) AND (c.oid=a.attrelid) AND (a.atttypid=t.oid) AND a.attnum > 0
				ORDER BY fldnum";
		$arrColList = $objQuery->getAll($sql, array($table_name));
		$arrColList = sfSwapArray($arrColList);
		
		$arrRet['col_def'] = $arrColList['defval'];
		$arrRet['col_name'] = $arrColList['attname'];
	}else if(DB_TYPE == "mysql"){
		$sql = "SHOW COLUMNS FROM $table_name";
		$arrColList = $objQuery->getAll($sql);
		$arrColList = sfSwapArray($arrColList);
		
		$arrRet['col_def'] = $arrColList['Extra'];
		$arrRet['col_name'] = $arrColList['Field'];
	}
	return $arrRet;
}

// 自動採番型の値を取得する
function lfGetAutoIncrementVal($table_name , $colname = ""){
	$objQuery = new SC_Query();
	$ret = "";

	if(DB_TYPE == "pgsql"){
		$ret = $objQuery->nextval($table_name, $colname);
	}else if(DB_TYPE == "mysql"){
		$sql = "SHOW TABLE STATUS LIKE ?";
		$arrData = $objQuery->getAll($sql, array($table_name));
		$ret = $arrData[0]['Auto_increment'];
	}
	return $ret;
}

// バックアップテーブルにデータを更新する
function lfUpdBkupData($data){
	$objQuery = new SC_Query();
	
	$sql = "INSERT INTO dtb_bkup (bkup_name,bkup_memo,create_date) values (?,?,now())";
	$objQuery->query($sql, array($data['bkup_name'],$data['bkup_memo']));
}

// バックアップテーブルからデータを取得する
function lfGetBkupData($where = "", $data = array()){
	$objQuery = new SC_Query();
	
	$sql = "SELECT bkup_name, bkup_memo, create_date FROM dtb_bkup ";
	if ($where != "")	$sql .= $where;
	
	$ret = $objQuery->getall($sql,$data);
	
	return $ret;
}

// バックアップファイルをリストアする
function lfRestore($bkup_name){
	global $objPage;
	$objQuery = new SC_Query("", false);
	$csv_data = "";
	$err = true;
	
	$bkup_dir = $objPage->bkup_dir . $bkup_name . "/";
	
	//バックアップフォルダに移動する
	chdir($objPage->bkup_dir);
	
	//圧縮フラグTRUEはgzip解凍をおこなう
	$tar = new Archive_Tar($bkup_name . ".tar.gz", TRUE);
	
	//指定されたフォルダ内に解凍する
	$err = $tar->extract("./");
	
	// 無事解凍できれば、リストアを行う
	if ($err) {
		
		// トランザクション開始
		$objQuery->begin();
		
		// DBをクリア
		$err = lfDeleteAll($objQuery);
		
		// INSERT実行
		if ($err) $err = lfExeInsertSQL($objQuery, $bkup_dir . "bkup_data.csv");

		// 自動採番の値をセット
		if ($err) lfSetAutoInc($objQuery, $bkup_dir . "autoinc_data.csv");

		// 各種ファイルのコピー
		if ($err) {
			// 画像のコピー
			$image_dir = $bkup_dir . "save_image/";
			$copy_mess = "";
			$copy_mess = sfCopyDir($image_dir, "../../upload/save_image/", $copy_mess, true);		
	
			// テンプレートのコピー
			$tmp_dir = $bkup_dir . "templates/";
			$copy_mess = "";
			$copy_mess = sfCopyDir($tmp_dir, "../../user_data/templates/", $copy_mess, true);		
			
			// インクルードファイルのコピー
			$inc_dir = $bkup_dir . "include/";
			$copy_mess = "";
			$copy_mess = sfCopyDir($inc_dir, "../../user_data/include/", $copy_mess, true);		
			
			// CSSのコピー
			$css_dir = $bkup_dir . "css/";
			$copy_mess = "";
			$copy_mess = sfCopyDir($css_dir, "../../user_data/css/", $copy_mess, true);		

			// バックアップデータの削除
			sfDelFile($bkup_dir);
		}

		// リストア成功ならコミット失敗ならロールバック
		if ($err) {
			$objQuery->commit();
			$objPage->restore_msg = "リストア終了しました。";
			$objPage->restore_err = true;
		}else{
			$objQuery->rollback();
			$objPage->restore_msg = "リストアに失敗しました。";
			$objPage->restore_name = $bkup_name;
			$objPage->restore_err = false;
		}

	}
}

// CSVファイルからインサート実行
function lfExeInsertSQL($objQuery, $csv){
	global $objPage;
	// csvファイルからデータの取得
	$arrCsvData = file($csv);

	$sql = "";
	$base_sql = "";
	$tbl_flg = false;
	$col_flg = false;
	$ret = true;
	$pagelayout_flg = false;
	$mode = $objPage->mode;
		
	foreach($arrCsvData as $key => $val){
		$data = trim($val);
		
		//空白行のときはテーブル変更
		if ($data == "") {
			$base_sql = "";
			$tbl_flg = false;
			$col_flg = false;
			continue;
		}
		
		// テーブルフラグがたっていない場合にはテーブル名セット
		if (!$tbl_flg) {
			$base_sql = "INSERT INTO $data ";
			$tbl_flg = true;
			
			if($data == "dtb_pagelayout"){
				$pagelayout_flg = true;
			}
			
			continue;
		}
		
		// カラムフラグがたっていない場合にはカラムセット
		if (!$col_flg) {
			if ($mode != "restore_config"){
				$base_sql .= " ($data) ";
			}
			$col_flg = true;
			continue;
		}

		// インサートする値をセット
		$sql = $base_sql . "VALUES ( ?";
		$arrInsdata = split(",",$data);
		for($i = 0; $i < count($arrInsdata); $i++){
			$sql .= ",?";
		}
		$sql .= ");\n";
		$objQuery->query($sql, $arrInsdata);
		$objQuery->getlastquery();
		
		sfprintr($arrInsdata);
		// エラーがあれば終了
		if ($err->message != ""){
			$objQuery->getlastquery();
			return false;
		}
		
		if ($pagelayout_flg) {
			// dtb_pagelayoutの場合には最初のデータはpage_id = 0にする
			$sql = "UPDATE dtb_pagelayout SET page_id = 0";
			$objQuery->query($sql);
			$pagelayout_flg = false;
		}
		
	}
	return $ret;
}

// 自動採番をセット
function lfSetAutoInc($objQuery, $csv){
	// csvファイルからデータの取得
	$arrCsvData = file($csv);

	foreach($arrCsvData as $key => $val){
		$arrData = split(",", trim($val));
		$objQuery->setval($arrData[0], $arrData[1], $arrData[2]);
	}
}

// DBを全てクリアする
function lfDeleteAll($objQuery){
	$ret = true;

	$arrTableList = lfGetTableList();
	
	foreach($arrTableList as $key => $val){
		// バックアップテーブルは削除しない
		if ($val != "dtb_bkup") {
			$trun_sql = "DELETE FROM $val;";
			$ret = $objQuery->query($trun_sql);
			
			if (!$ret) return $ret;
		}
	}
	
	return $ret;
}

// バックアップテーブルを作成する
function lfCreateBkupTable(){
	$objQuery = new SC_Query();
	
	// テーブルの存在チェック
	$arrTableList = lfGetTableList();
	
	if(!in_array("dtb_bkup", $arrTableList)){
		// 存在していなければ作成
		$cre_sql = "
			create table dtb_bkup
			(
				bkup_name	text,
				bkup_memo	text,
				create_date	timestamp
			);
		";
		
		$objQuery->query($cre_sql);
	}
}

?>