/*******************************************************************************
*	All Rights reserved,Copyright (c) K-Opticom
********************************************************************************
*＜プログラム内容＞
*	システム名		：eo顧客基幹システム
*	モジュール名	：JKK0341TPMA
*	ソースファイル名：JKK0341TPMA.java
*	作成者			：富士通
*	日付			：2011年11月22日
*＜機能概要＞
*	機器提供サービス契約のサービスIF顧客独自処理部品
*＜修正履歴＞
*	バージョン	修正日		修正者		修正内容
*	v1.00.00	2011/11/22	富士通		新規作成
 *  v5.00.00    2013/10/03  FJ）寺園    OM-2013-0002733
*  v19.00.00    2015/10/19  FJ）谷口    【OM-2015-0001224】ECS-SODの緊急通報住所と利用場所住所が不一致
*
********************************************************************************/

package eo.ejb.cbs.mainproc;

import java.util.ArrayList;
import java.util.HashMap;

import com.fujitsu.futurity.common.JCMConstants;
import com.fujitsu.futurity.model.base.CAANMsg;
import com.fujitsu.futurity.model.ejb.common.JSYejbLog;
import com.fujitsu.futurity.model.ejb.common.StatusCodes;
import com.fujitsu.futurity.model.ejb.common.fw.AgentDispatchContext;
import com.fujitsu.futurity.model.ejb.common.fw.TemplateMainHandler;

import eo.ejb.cbm.entity.KK0341ETMsg;
import eo.ejb.common.JKKModelCommon;
import eo.ejb.common.JKKejbCallTypeChecker;
import eo.ejb.common.db.JKKejbExclusiveProcKK0251;
import eo.ejb.common.db.JKKejbExclusiveProcKK0491;
import eo.ejb.common.db.JKKejbExclusiveProcKK2111;
import eo.ejb.common.db.JKKejbKK0341DBABase;

/**
 * <p>
 * 機器提供サービス契約のサービスIF顧客独自処理部品クラスです。
 * </p>
 * @author 富士通
 */
public class JKK0341TPMA implements TemplateMainHandler
{

	/** CBS層 機器提供サービス契約番号キー項目名 */
	private static final String CBS_PK_NAME = "kktk_svc_kei_no";

	/** CBS層 世代登録年月日時分秒キー項目名 */
	private static final String CBS_GENE_NAME = "gene_add_dtm";

	/** CBS層 世代登録年月日時分秒エラーキー項目名 */
	private static final String CBS_GENE_NAME_ERR = "gene_add_dtm_err";

	/** CBS層 請求契約番号キー項目名 */
	private static final String CBS_SEIKY_NO = "seiky_kei_no";

	/** CBS層 更新年月日時分秒（更新前）キー項目名 */
	private static final String CBS_UPD_BF_NAME = "upd_dtm_bf";

	/** CBS層 親契約識別コードキー項目名 */
	private static final String CBS_OYA_KEI_NAME = "oya_kei_skbt_cd";

	/** CBS層 ロックルートキー */
	private static final ArrayList<String> CBS_TAG_KEYS = new ArrayList<String>();

	// OM-2015-0001224 ADD START
	/** CBS層 予約適用年月日キー項目名 */
	private static final String CBS_RSV_APLY_NAME = "rsv_aply_ymd";
	// OM-2015-0001224 ADD END

	static
	{
		CBS_TAG_KEYS.add("svc_kei_no");
		CBS_TAG_KEYS.add("svc_kei_kaisen_ucwk_no");
	}

	/** 親契約識別コード（サービス契約） */
	private static final String OYA_KEI_CD_SVC = "01";

	/** 親契約識別コード（サービス契約回線内訳） */
	private static final String OYA_KEI_CD_KAIS = "02";

	/** 親契約識別コード（サービス契約内訳） */
	private static final String OYA_KEI_CD_UCWK = "03";

	/** 親契約識別コード（オプションサービス契約） */
	private static final String OYA_KEI_CD_OP = "04";

	/**
	 * 複数処理の対象
	 */
	private static final HashMap<String, String> SEC_LOCK_MAP = new HashMap<String, String>();

	static
	{
		SEC_LOCK_MAP.put("EKK0341D010", "EKK0341D010");
	}

	/** エラーフラグ */
	private static final String ERR_FLG = "EA";

	/** エラーフラグ設定項目名 */
	private static final String ERR_CONTENTS = "upd_dtm_bf_err";
	
	/** 機器送付先住所手動入力フラグ：手動入力でない */
	private static final String KIKI_SHS_AD_MAN_INPUT_FLG_AUTO = "0";

	/**
	 * <p>
	 * 機器提供サービス契約のスキーマのロック、タイムスタンプチェック、およびタイムスタンプ更新を行います。
	 * 請求契約のスキーマのロックを行います。
	 * </p>
	 * @param inCBSMsg 処理対象のメッセージキャリア
	 * @param inContext Agentから渡されたAgentDispatchContext
	 */
	public void execExclusiveProc(CAANMsg inCBSMsg, AgentDispatchContext inContext)
	{
		// 必要な項目を保持したETMsgを作成する
		CAANMsg inETMsg = new CAANMsg(KK0341ETMsg.class.getName());

		// プライマリキーを設定
		inETMsg.set(KK0341ETMsg.KKTK_SVC_KEI_NO, inCBSMsg.getString(CBS_PK_NAME));

		// 親契約識別コードを設定
		inETMsg.set(KK0341ETMsg.OYA_KEI_SKBT_CD, inCBSMsg.getString(CBS_OYA_KEI_NAME));

		// キー情報を設定
		for (int i = 0; i < CBS_TAG_KEYS.size(); i++)
		{
			inETMsg.set(CBS_TAG_KEYS.get(i).toUpperCase(), inCBSMsg.getString(CBS_TAG_KEYS.get(i)));
		}

		// 更新情報の設定
		inETMsg.set(KK0341ETMsg.UPD_DTM, inCBSMsg.getString(JCMConstants.OPERATE_DATETIME_KEY));
		inETMsg.set(KK0341ETMsg.UPD_OPEACNT, inCBSMsg.getString(JCMConstants.OPERATOR_ID_KEY));

		// メイン処理の呼び出し
		boolean ret = this.execExclusiveMain(inETMsg, inCBSMsg.getString(CBS_UPD_BF_NAME), inCBSMsg.getString(JCMConstants.FUNC_CODE_KEY));

		// メイン処理が正常だった場合
		if (ret)
		{
			// 請求契約ロック限定処理部品の呼び出しを行う
			ret = this.execSecLock(inCBSMsg, inCBSMsg.getString(JCMConstants.TEMPLATE_ID_KEY));
		}

		// 処理結果がfalseの場合は関連制約エラーとする
		if (!ret)
		{
			inCBSMsg.set(ERR_CONTENTS, ERR_FLG);
			inCBSMsg.set(JCMConstants.STATUS_INT_KEY, StatusCodes.RELATION_ERR);
			return;
		}
	}

	/**
	 * <p>
	 * 機器提供サービス契約のスキーマのレコード検索、ロック、タイムスタンプチェック、およびタイムスタンプ更新を行います。
	 * 請求契約のスキーマのロックを行います。
	 * </p>
	 * @param inCBSMsg 処理対象のメッセージキャリア
	 * @param inContext Agentから渡されたAgentDispatchContext
	 */
	public void execExclusiveProcRefTgt(CAANMsg inCBSMsg, AgentDispatchContext inContext)
	{
		CAANMsg retMsg = null;

		// 処理対象のメッセージに世代登録年月日時分秒が入力項目として存在する場合はプライマリ検索を行う
		if(inCBSMsg.containsKeyOfSchema(CBS_GENE_NAME_ERR))
		{
			// プライマリ検索用のメッセージを作成
			CAANMsg inETMsg = new CAANMsg(KK0341ETMsg.class.getName());
			inETMsg.set(KK0341ETMsg.KKTK_SVC_KEI_NO, inCBSMsg.getString(CBS_PK_NAME));
			inETMsg.set(KK0341ETMsg.GENE_ADD_DTM, inCBSMsg.getString(CBS_GENE_NAME));

			// プライマリ検索を行う
			retMsg = new JKKejbKK0341DBABase().findByPrimaryKey(inETMsg);
		}
		else
		{
			// カレント検索用のメッセージを作成
			CAANMsg inETMsg = new CAANMsg(KK0341ETMsg.class.getName());
			inETMsg.set(KK0341ETMsg.KKTK_SVC_KEI_NO, inCBSMsg.getString(CBS_PK_NAME));
			// OM-2015-0001224 MOD START
			// 入力の予約適用年月日で検索を行なう必要がある。
			// 未来日(20991231)レコードに対しての更新時に排他エラーになってしまう。
//			inETMsg.set(KK0341ETMsg.RSV_APLY_YMD, JKKModelCommon.getOpeDate(inCBSMsg));
			String templateID = inCBSMsg.getString(JCMConstants.TEMPLATE_ID_KEY);
			// EKK0341C232_機器提供サービス契約情報変更（住所変更）
			if("EKK0341C232".equals(templateID))
			{
				inETMsg.set(KK0341ETMsg.RSV_APLY_YMD, inCBSMsg.getString(CBS_RSV_APLY_NAME));
			}
			// 上記S-IF以外は従来通り運用日で取得する
			else
			{
				inETMsg.set(KK0341ETMsg.RSV_APLY_YMD, JKKModelCommon.getOpeDate(inCBSMsg));
			}
			// OM-2015-0001224 MOD END

			// カレント検索を行う
			retMsg = new JKKejbKK0341DBABase().findByCurrent(inETMsg);
		}

		// レコードが取得できなかった場合、処理を終了する
		if (null == retMsg)
		{
			return;
		}

		// 更新情報の設定
		retMsg.set(KK0341ETMsg.UPD_DTM, inCBSMsg.getString(JCMConstants.OPERATE_DATETIME_KEY));
		retMsg.set(KK0341ETMsg.UPD_OPEACNT, inCBSMsg.getString(JCMConstants.OPERATOR_ID_KEY));

		// メイン処理の呼び出し
		boolean ret = this.execExclusiveMain(retMsg, inCBSMsg.getString(CBS_UPD_BF_NAME), inCBSMsg.getString(JCMConstants.FUNC_CODE_KEY));

		// メイン処理が正常だった場合
		if (ret)
		{
			// ロック限定処理部品の呼び出しを行う
			ret = this.execSecLock(inCBSMsg, inCBSMsg.getString(JCMConstants.TEMPLATE_ID_KEY));
		}

		// 処理結果がfalseの場合は関連制約エラーとする
		if (!ret)
		{
			inCBSMsg.set(ERR_CONTENTS, ERR_FLG);
			inCBSMsg.set(JCMConstants.STATUS_INT_KEY, StatusCodes.RELATION_ERR);
			return;
		}
	}

	/**
	 * <p>
	 * 機器提供サービス契約のスキーマのロック、タイムスタンプチェック、およびタイムスタンプ更新を行います。
	 * </p>
	 * @param inETMsg 処理対象のメッセージキャリア（ET）
	 * @param updDtmBf 更新年月日時分秒（更新前）
	 * @param funcCd 機能コード
	 * @return スキーマのロック、タイムスタンプチェック、およびタイムスタンプ更新が正常終了の場合はtrue。
	 */
	private boolean execExclusiveMain(CAANMsg inETMsg, String updDtmBf, String funcCd)
	{
		// ログの出力
		outLog(inETMsg, updDtmBf);

		boolean ret = true;

		// 親契約識別コードによる処理の呼び分け
		if (OYA_KEI_CD_SVC.equals(inETMsg.getString(KK0341ETMsg.OYA_KEI_SKBT_CD))
				|| OYA_KEI_CD_UCWK.equals(inETMsg.getString(KK0341ETMsg.OYA_KEI_SKBT_CD))
				|| OYA_KEI_CD_OP.equals(inETMsg.getString(KK0341ETMsg.OYA_KEI_SKBT_CD)))
		{
			ret = new JKKejbExclusiveProcKK2111().isExProcSvcKeiNo(inETMsg, inETMsg.getString(KK0341ETMsg.SVC_KEI_NO), updDtmBf, funcCd);
		}
		else if (OYA_KEI_CD_KAIS.equals(inETMsg.getString(KK0341ETMsg.OYA_KEI_SKBT_CD)))
		{
			ret = new JKKejbExclusiveProcKK0251().isExProcKaisenUcwkNo(inETMsg, inETMsg.getString(KK0341ETMsg.SVC_KEI_KAISEN_UCWK_NO), updDtmBf, funcCd);
		}

		return ret;
	}

	/**
	 * <p>
	 * ログの出力を行います。
	 * </p>
	 * @param inETMsg 処理対象のメッセージキャリア（ET）
	 * @param updDtmBf 更新年月日時分秒（更新前）
	 */
	private void outLog(CAANMsg inETMsg, String updDtmBf)
	{
		// プライマリキーの出力
		JSYejbLog.println(JSYejbLog.DEBUG, getClass(), "JKK0341TPMA.kktk_svc_kei_no=" + inETMsg.getString(KK0341ETMsg.KKTK_SVC_KEI_NO));

		// 親契約識別コードの出力
		JSYejbLog.println(JSYejbLog.DEBUG, getClass(), "JKK0341TPMA.oya_kei_skbt_cd=" + inETMsg.getString(KK0341ETMsg.OYA_KEI_SKBT_CD));

		// キー情報の出力
		for (int i = 0; i < CBS_TAG_KEYS.size(); i++)
		{
			JSYejbLog.println(JSYejbLog.DEBUG, getClass(), "JKK0341TPMA." + CBS_TAG_KEYS.get(i) + "=" + inETMsg.getString(CBS_TAG_KEYS.get(i).toUpperCase()));
		}

		// 更新年月日時分秒（更新前）の出力
		JSYejbLog.println(JSYejbLog.DEBUG, getClass(), "JKK0341TPMA.upd_dtm_bf=" + updDtmBf);
	}

	/**
	 * <p>
	 * ロック限定処理の呼び出し判定を行い、呼出し対象インターフェイスの場合は処理の呼び出しを行います。
	 * </p>
	 * @param inCBSMsg 処理対象のCBSメッセージ
	 * @param templateID サービスインターフェイスID
	 * @return ロックに失敗した場合はfalse。
	 */
	private boolean execSecLock(CAANMsg inCBSMsg, String templateID)
	{
		// ロック限定処理呼出し対象のインターフェイス以外は呼出を行わない
		if (!SEC_LOCK_MAP.containsKey(templateID))
		{
			return true;
		}		

		// 請求契約番号がCBSメッセージに存在しない場合は処理を行わない
		if (!inCBSMsg.containsKeyOfMsgData(CBS_SEIKY_NO))
		{
			return true;
		}

		// ロック限定処理の呼び出しを行う
		return new JKKejbExclusiveProcKK0491().isExLockSeikyKeiNo(inCBSMsg, inCBSMsg.getString(CBS_SEIKY_NO));
	}

	/**
	 * <p>
	 * 住所コード相関チェック（機器送付先）を行います。
	 * </p>
	 * @param inCBSMsg	CBSMsgメッセージ
	 * @param inContext	Agent
	 */
	public void kikiSohusAdChk(CAANMsg inCBSMsg, AgentDispatchContext inContext) {

		JSYejbLog.println(JSYejbLog.DEBUG,this.getClass(),"invoke#Call");

		// チェックを行う
		int stRet = 0;
		String templateID = inCBSMsg.getString(JCMConstants.TEMPLATE_ID_KEY);
		String errCD = "E3";
        if ("EKK0341C480".equals(templateID)) 
        {
    		errCD = "E1";
        }

		// 機器送付先住所が変更されている場合のみチェックを行う
		if(JKKModelCommon.isKikiSohusAdChgKK0341(inCBSMsg, inContext, inCBSMsg.getString("kktk_svc_kei_no"), inCBSMsg.getString("kiki_sohus_ad_cd"), inCBSMsg.getString("kiki_sohus_pcd")))
		{
			//チェック条件：機器提供サービス契約情報変更_単関連制約実施条件_住所コード相関チェック(機器送付先)の戻り値がtrueと等しい
			if(isAddCdSoknKikiChk(inCBSMsg, inContext, inCBSMsg.getString("kiki_sohus_ad_cd"), inCBSMsg.getString("kiki_shs_ad_man_input_flg")) == true){
				//住所入力チェック（契約）の戻り値がfalseと等しい ：エラー
				if(JKKModelCommon.isKKAddressInputCheck(inCBSMsg, inContext , errCD, "kiki_sohus_ad_cd", "kiki_sohus_state_nm", "kiki_sohus_city_nm", "kiki_sohus_oaztsu_nm", "kiki_sohus_azcho_nm", "kiki_sohus_bnchigo" ) == false){
					inCBSMsg.set("kiki_sohus_ad_cd_err", errCD);
					JSYejbLog.println(JSYejbLog.DEBUG,this.getClass(),"kiki_sohus_ad_cd_err:" + errCD);
					stRet = StatusCodes.ITEM_RELATION_ERR;
				}
			}
		}

		JSYejbLog.println(JSYejbLog.DEBUG,this.getClass(),"invoke#End");

		//チェック結果によりステータスを設定する
		if (stRet > inCBSMsg.getInt("status")) {
			inCBSMsg.set("status",stRet);
		}
	}
	
	/**
	 * <p>
	 * 単関連制約 実施条件_住所コード相関チェック(機器送付先)
	 * </p>
	 * @param inCBSMsg CBSメッセージ
	 * @param inContext Agentから渡されたAgentDispatchContext
	 * @return true：実施条件を満たしている false：実施条件を満たしていない
	 */
	private boolean isAddCdSoknKikiChk(CAANMsg inCBSMsg, AgentDispatchContext inContext, String kikiSohusAdCd, String kikiShsAdManInputFlg)
	{
		String templateID = inCBSMsg.getString(JCMConstants.TEMPLATE_ID_KEY);
        if ("EKK0341C230".equals(templateID) ||
        		"EKK0341C231".equals(templateID) ||
        		"EKK0341C232".equals(templateID)) 
        {
            // チェック条件@：サービスの呼出元が「バッチ」以外であること
    		if (JKKejbCallTypeChecker.isCallTypeBatch(inContext))
    		{
    			return false;
    		}
        }

		// チェック条件A：「機器送付先住所コード」が入力されている
		if (kikiSohusAdCd == null)
		{
			return false;
		}

		// チェック条件B：「機器送付先住所手動入力フラグ」が"0"である
		if (!KIKI_SHS_AD_MAN_INPUT_FLG_AUTO.equals(kikiShsAdManInputFlg))
		{
			return false;
		}
		return true;
	}
}