/*********************************************************************
* All Rights reserved,Copyright (c) K-Opticom
**********************************************************************
*＜プログラム内容＞
*   システム名      ：eo顧客基幹システム
*   モジュール名    ：JKKRzkzmAnkenChkCC
*   ソースファイル名：JKKRzkzmAnkenChkCC.java
*   作成者          ：富士通
*   日付            ：2021年9月13日
*＜機能概要＞
*   ローゼット化設置済案件判定CCです。
*＜修正履歴＞
*   バージョン  修正日       修正者      修正内容
*   v56.00.00   2021/09/13   FJ)平野     【ANK-4038-00-00】【eo定期】光ローゼット化対応
*   v60.00.00   2022/07/08   FJ)吉村     【ANK-4244-00-00】【eo定期】MT既設賃貸対応
*   v61.00.00	2022/12/14	 FJ)澤田	 【ANK-4315-00-00】【eo定期】 eoホームゲートウェイ導入対応
*   v61.00.01	2023/09/01	 FJ)藤本涼	 【ANK-4431-00-00】【ｅｏ定期】 ONU計画交換対応
*   v73.00.00	2024/12/02	 FJ)辻中	 【ANK-4427-00-00】NTT卸対応
**********************************************************************/
package com.fujitsu.futurity.bp.custom.common;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

import com.fujitsu.futurity.bp.x21.bpm.ServiceComponentRequestInvoker;
import com.fujitsu.futurity.bp.x21.bpm.db.SessionHandle;
import com.fujitsu.futurity.bp.x21.bpm.exception.RequestParameterException;
import com.fujitsu.futurity.bp.x21.bpm.parameter.IRequestParameterReadWrite;
import com.fujitsu.futurity.bp.x21.cc.AbstractCommonComponent;
import com.fujitsu.futurity.bp.x21.cc.exception.SCCallException;
import com.fujitsu.futurity.common.JCMConstants;
import com.fujitsu.futurity.common.x01.sc.SCControlMapKeys;
import com.fujitsu.futurity.mapping.bp.common.TemplateErrorUtil;
import com.fujitsu.futurity.model.base.CAANMsg;
import com.fujitsu.futurity.model.common.JCMAPLConstMgr;

import eo.common.constant.JKKStrConst;
import eo.common.util.JKKStringUtil;
import eo.ejb.cbs.cbsmsg.EKK0081A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0081A010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0251A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0251A010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0861A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0861A010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0891A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0891A010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKU0281A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKU0281A010CBSMsg1List;


/**
 * ローゼット化設置済案件判定部品です。<p>
 * <br>
 * @author FJ
 */
public class JKKRzkzmAnkenChkCC extends AbstractCommonComponent
{
	/** エラーメッセージ */
	public static final String ERR_MSG = "INVALID_RETURN_MESSAGE";

	/** SC呼び出し部品 */
	ServiceComponentRequestInvoker scCall = null;
	
	/**
	 * テンプレートID(EKK0251A010)
	 */
	private static final String TEMPLATE_ID_EKK0251A010 = "EKK0251A010";

	/**
	 * テンプレートID(EKK0861A010)
	 */
	private static final String TEMPLATE_ID_EKK0861A010 = "EKK0861A010";
	
	/**
	 * テンプレートID(EKK0081A010)
	 */
	private static final String TEMPLATE_ID_EKK0081A010 = "EKK0081A010";
	
	/**
	 * テンプレートID(EKK0891A010)
	 */
	private static final String TEMPLATE_ID_EKK0891A010 = "EKK0891A010";
	
	/**
	 * テンプレートID(EKU0281A010)
	 */
	private static final String TEMPLATE_ID_EKU0281A010 = "EKU0281A010";
		
	/** 
	 * チェック結果：判定不可
	 */
	private static final String CHK_JDGEFUKA = "99";
	
	/** 
	 * チェック結果：ローゼット化設置済案件対象
	 */
	private static final String CHK_RZKZM = "3";

	/** 
	 * チェック結果：ローゼット化設置済案件対象外
	 */
	private static final String CHK_NOTRZKZM = "0";
	
	/** 
	 * チェック結果：工事スキップ可
	 */
	private static final String CHK_KOJISKIP_KA = "1";
	
	/** 
	 * チェック結果：工事スキップ不可
	 */
	private static final String CHK_KOJISKIP_FUKA = "2";
	
	/** CBSMsgのテンプレートID項目 */
	private static final String TEMPLATEID = "templateID";

	/** CBSMsgの機能コード項目 */
	private static final String FUNC_CODE = "func_code";

	/** CBSMsgのステータス項目 */
	private static final String STATUS = "status";

	/**
	 * ローゼット化設置済案件判定処理メイン
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param fixedText サービスメッセージ
	 * @return リクエストパラメータ
	 * @throws Throwable スロー可能オブジェクト
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite jdgeRzkzmAnken(SessionHandle handle, IRequestParameterReadWrite param, String fixedText) throws Throwable
	{
		
		HashMap<String, Object> ccMsg = (HashMap<String, Object>)param.getData(fixedText);

		// サービス契約番号
		String svcKeiNo = getMapValue(ccMsg, "svc_kei_no");
		// サービス契約回線内訳番号
		String svcKaisenUcwkNo = getMapValue(ccMsg, "svc_kei_kaisen_ucwk_no");
		// 異動区分
		String idoDiv = getMapValue(ccMsg, "ido_div");
		
		// SC呼び出し部品
		ServiceComponentRequestInvoker scCall = new ServiceComponentRequestInvoker();
		
		// 入力パラメータチェック
		if ((svcKeiNo == null||"".equals(svcKeiNo)) || (svcKaisenUcwkNo == null||"".equals(svcKaisenUcwkNo))
				||(idoDiv == null||"".equals(idoDiv)))
		{
			// チェック結果を"99"で返却
			ccMsg.put("chk_result", CHK_JDGEFUKA);
			return param;
		}
		else
		{
			// ローゼット化設置済み案件チェックの結果をマップに格納
			HashMap<String, Object> rzResultMap = rzkzmAnkenChk(handle, param, scCall, fixedText,ccMsg);
			// チェック結果を取得
			String rzkzmAnkenResult = (String)rzResultMap.get("rozett_result");
			// チェック結果が"3"(ローゼット化設置済案件対象)の場合
			if(CHK_RZKZM.equals(rzkzmAnkenResult))
			{
				// 工事スキップ可否チェックを実行し結果を格納
				String kojiSkipKhResult = kojiSkipJdge(handle, param, scCall, fixedText,rzResultMap,ccMsg);
				// チェック結果は工事スキップ可否判定の結果を返却
				ccMsg.put("chk_result", kojiSkipKhResult);
			// チェック結果が"3"(ローゼット化設置済案件対象)以外の場合
			}else{
				// ローゼット化設置済案件チェックの結果を格納
				ccMsg.put("chk_result", rzkzmAnkenResult);
			}
		}
		return param;
	}
		
	/**
	 * エラー情報マッピング処理
	 * 
	 * <br>
	 * @param param リクエストパラメータ
	 * @param templates テンプレート
	 * @param returnCode リターンコード
	 * @return リクエストパラメータ
	 * @throws RequestParameterException リクエストパラメータ操作時の例外
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite editErrorInfo(IRequestParameterReadWrite param, 
													CAANMsg[] templates,
													String fixedText,
													int returnCode) throws RequestParameterException
	{
		for (int i = 0; i < templates.length; i++)
		{
			CAANMsg template = templates[i];
			int templateStatus = template.getInt(STATUS);
			if (returnCode != 0)
			{
				templateStatus = 9000;
			}
			if (JCMAPLConstMgr.getString("RETURN_MESSAGE_" + String.format("%1$04d", templateStatus)) == null)
			{
				templateStatus = 0;
			}

			int bpStatus = 0;
			Object obj = param.getControlMapData(SCControlMapKeys.RETURN_CODE);
			if (obj == null)
			{
				bpStatus = -1;
			}
			else
			{
				bpStatus = Integer.parseInt((String)param.getControlMapData(SCControlMapKeys.RETURN_CODE));
			}

			if (templateStatus > bpStatus)
			{
				// BPにサービスコンポーネントのステータスを設定する。
				String formatStatus = String.format("%1$04d", templateStatus);
				String message = JCMAPLConstMgr.getString("RETURN_MESSAGE_" + formatStatus);
				param.setControlMapData(SCControlMapKeys.RETURN_CODE, formatStatus);
				param.setControlMapData(SCControlMapKeys.RETURN_MESSAGE, message);
			}

			// ユーザデータ情報
			HashMap<String, Object> inMap = (HashMap<String, Object>)param.getData(fixedText);

			HashMap mp = template.getHashMap();

			Iterator it = mp.keySet().iterator();
			while (it.hasNext())
			{
				String key = (String)it.next();
				if (key.endsWith("_err"))
				{
					int keyIdx = key.lastIndexOf("_err");
					if (inMap.containsKey(key.substring(0, keyIdx)))
					{
						inMap.put(key, mp.get(key));
					}
				}
			}
		}
		return param;
	}
	
	/**
	 * ローゼット化設置済案件チェック
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param scCall SC呼出部品のインスタンス
	 * @param ccMsg CCの入出力領域
	 * @return paramMap チェック結果を格納したマップ
	 * @throws Exception SC実行時の例外
	 */
	@SuppressWarnings("unchecked")
	private HashMap<String, Object> rzkzmAnkenChk(SessionHandle handle, 
			IRequestParameterReadWrite param, 
			ServiceComponentRequestInvoker scCall,
			String fixedText,
			HashMap ccMsg
			) throws Exception
	{
		CAANMsg template = null;

		// チェック結果_初期値は"0"(ローゼット化設置済み案件対象外）
		String rzkzmResult = CHK_NOTRZKZM;
		// サービス契約回線内訳番号
		String svcKeiKaisenUcwkNo = (String)ccMsg.get("svc_kei_kaisen_ucwk_no");
		// サービス契約番号
		String svcKeiNo = (String)ccMsg.get("svc_kei_no");
		// 工事スキップ可否判定処理に渡すマップを作成
		HashMap<String, Object> paramMap = new HashMap<String, Object>();
		// サービス契約回線内訳一意照会を実行
		Object[][] ekk0251a010IN = {
				{EKK0251A010CBSMsg.KEY_SVC_KEI_KAISEN_UCWK_NO, svcKeiKaisenUcwkNo}
		};
		// ◇ SC呼出実行
		CAANMsg ekk0251a010Msg = callSC (handle, param, scCall, TEMPLATE_ID_EKK0251A010, "2",fixedText,  ekk0251a010IN);
		// 処理結果
		CAANMsg[] ekk0251a010MsgList = ekk0251a010Msg.getCAANMsgList(EKK0251A010CBSMsg.EKK0251A010CBSMSG1LIST);
		if (ekk0251a010MsgList != null && 0 < ekk0251a010MsgList.length)
		{
			template = ekk0251a010MsgList[0];
			// マンション物件番号を取得
			String mansionBukkenNo = template.getString(EKK0251A010CBSMsg1List.MANSION_BUKKEN_NO);
			// ローゼット設置状態コードを取得
			String rzStcStatCd = template.getString(EKK0251A010CBSMsg1List.ROZETT_STC_STAT_CD);
			// マップに取得したローゼット設置状態コードを格納
			paramMap.put("rozett_stat_cd",rzStcStatCd);
			// マンション物件番号が設定済みの場合
			if (mansionBukkenNo != null && !"".equals(mansionBukkenNo))
			{
				// マンション物件一意照会を実行
				Object[][] ekk0861a010IN = {
						{EKK0861A010CBSMsg.KEY_MANSION_BUKKEN_NO, mansionBukkenNo}
				};
				// ◇ SC呼出実行
				CAANMsg ekk0861a010Msg = callSC (handle, param, scCall, TEMPLATE_ID_EKK0861A010, "1",fixedText,ekk0861a010IN);
				// 処理結果
				CAANMsg[] ekk0861a010MsgList = ekk0861a010Msg.getCAANMsgList(EKK0861A010CBSMsg.EKK0861A010CBSMSG1LIST);

				if (ekk0861a010MsgList != null && 0 < ekk0861a010MsgList.length)
				{
					template = ekk0861a010MsgList[0];
					// マンション物件コードを取得
					String mansionBukkenCd = template.getString(EKK0861A010CBSMsg1List.MANSION_BUKKEN_CD);
					// マンション物件コードが取得できた場合
					if (mansionBukkenCd != null && !"".equals(mansionBukkenCd))
					{
						// 取得したコードが"002"(メゾン)であれば
						if(JKKStrConst.MANSION_BUKKEN_CD_MEZON.equals(mansionBukkenCd))
						{
							// チェック結果は"3"(ローゼット化設置済案件対象)
							rzkzmResult = CHK_RZKZM;
						}
						// 取得したコードが"001"(マンション)の場合
						else if (JKKStrConst.MANSION_BUKKEN_CD_MANSION.equals(mansionBukkenCd))
						{
							// サービス契約一意照会を実行
							Object[][] ekk0081a010IN = {
									{EKK0081A010CBSMsg.KEY_SVC_KEI_NO, svcKeiNo},
									{EKK0081A010CBSMsg.KEY_RSV_APLY_YMD, JPCBPCommon.getOpeDate(null)}
							};
							// ◇ SC呼出実行
							CAANMsg ekk0081a010Msg = callSC (handle, param, scCall, TEMPLATE_ID_EKK0081A010, "2",fixedText,ekk0081a010IN);
							// 処理結果
							CAANMsg[] ekk0081a010MsgList = ekk0081a010Msg.getCAANMsgList(EKK0081A010CBSMsg.EKK0081A010CBSMSG1LIST);
							if (ekk0081a010MsgList != null && 0 < ekk0081a010MsgList.length)
							{
								template = ekk0081a010MsgList[0];
								// 提供方式契約番号を取得
								String teikyoHosikiKeiNo= template.getString(EKK0081A010CBSMsg1List.TK_HOSHIKI_KEI_NO);
								if(teikyoHosikiKeiNo != null && !"" .equals(teikyoHosikiKeiNo))
								{
									// 提供方式契約一意照会を実行
									Object[][] ekk0891a010IN = {
											{EKK0891A010CBSMsg.KEY_TK_HOSHIKI_KEI_NO, teikyoHosikiKeiNo}
									};
									// ◇ SC呼出実行
									CAANMsg ekk0891a010Msg = callSC (handle, param, scCall, TEMPLATE_ID_EKK0891A010, "1",fixedText,ekk0891a010IN);
									// 処理結果
									CAANMsg[] ekk0891a010MsgList = ekk0891a010Msg.getCAANMsgList(EKK0891A010CBSMsg.EKK0891A010CBSMSG1LIST);
									if (ekk0891a010MsgList != null && 0 < ekk0891a010MsgList.length)
									{
										template = ekk0891a010MsgList[0];
										// 提供方式コードを取得
										String teikyoHosikiCd= template.getString(EKK0891A010CBSMsg1List.TK_HOSHIKI_CD);
										// ANK-4244-00-00 MOD START
										// 加入契約支払方法コードを取得
									//	String addKeiPayWayCd= template.getString(EKK0891A010CBSMsg1List.KANYU_KEI_PAY_HOSHIKI_CD);
										// 提供方式コードが"008"(光配線)の場合
									//	if(JKKStrConst.CD00590_OPTICAL.equals(teikyoHosikiCd) && JKKStrConst.CD01216_KOBETSU.equals(addKeiPayWayCd))
										if(JKKStrConst.CD00590_OPTICAL.equals(teikyoHosikiCd))
										// ANK-4244-00-00 MOD END
										{
											// チェック結果は"3"(ローゼット化設置済案件対象)
											rzkzmResult = CHK_RZKZM;
										}else{
											// チェック結果は"0"(ローゼット化設置済案件対象外)
											rzkzmResult = CHK_NOTRZKZM;
										}
									}
								}
							}
						}
					// マンション物件コードが取得できなかった場合
					}else{
						// チェック結果は"0"(ローゼット化設置済案件対象外)
						rzkzmResult = CHK_NOTRZKZM;
					}
				}
			// マンション物件番号が未設定の場合
			}else{
				// 【ANK-4427-00-00】NTT卸対応 ADD START
				// サービス契約一意照会を実行
				Object[][] ekk0081a010IN = {
						{EKK0081A010CBSMsg.KEY_SVC_KEI_NO, svcKeiNo},
						{EKK0081A010CBSMsg.KEY_RSV_APLY_YMD, JPCBPCommon.getOpeDate(null)}
				};
				// ◇ SC呼出実行
				CAANMsg ekk0081a010Msg = callSC (handle, param, scCall, TEMPLATE_ID_EKK0081A010, "2",fixedText,ekk0081a010IN);
				// 処理結果
				CAANMsg[] ekk0081a010MsgList = ekk0081a010Msg.getCAANMsgList(EKK0081A010CBSMsg.EKK0081A010CBSMSG1LIST);
				if (ekk0081a010MsgList != null && 0 < ekk0081a010MsgList.length)
				{
					template = ekk0081a010MsgList[0];
					String nttOrsDiv= template.getString(EKK0081A010CBSMsg1List.NTTORS_DIV);
					//nttOrsDivに値が設定されていなければ、通常のホームタイプ
					if (JKKStringUtil.isNullEmpty(nttOrsDiv)){
				//【ANK-4427-00-00】NTT卸対応 ADD END
					// チェック結果は"3"(ローゼット化設置済案件対象)
						rzkzmResult = CHK_RZKZM;
				// 【ANK-4427-00-00】NTT卸対応 ADD START
					} else {
						//nttOrsDivに値が設定されていれば、NTT卸契約
						rzkzmResult = CHK_NOTRZKZM;
					}
				} 
				// 【ANK-4427-00-00】NTT卸対応 ADD END
			}
		}
		// マップにチェック結果を格納
		paramMap.put("rozett_result",rzkzmResult);
		return paramMap;	
	}
	
	/**
	 * 工事スキップ可否判定処理
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param scCall SC呼出部品のインスタンス
	 * @param rzResultMap ローゼット化設置済案件検索結果マップ
	 * @param ccMsg CCの入出力領域
	 * @return 工事案件スキップ可否結果
	 * @throws Exception SC実行時の例外
	 */
	@SuppressWarnings("unchecked")
	private String kojiSkipJdge(SessionHandle handle, 
			IRequestParameterReadWrite param, 
			ServiceComponentRequestInvoker scCall,
			String fixedText,
			HashMap rzResultMap,
			HashMap ccMsg
			) throws Exception
	{
		// サービス契約回線内訳番号
		String inSvcKeiKaisenUcwkCd = (String)ccMsg.get("svc_kei_kaisen_ucwk_no");
		// 異動区分
		String inIdoDiv = (String)ccMsg.get("ido_div");
		// ローゼット設置状態コードをローゼット案件判定結果より取得
		String rozzetStatCd = (String)rzResultMap.get("rozett_stat_cd");
		CAANMsg template = null;
		// 工事スキップチェック結果_初期値は"2"(工事スキップ不可）
		String kojiSkipResult = CHK_KOJISKIP_FUKA;
		// 異動区分が"00006"(撤去(解約))または"00019"(住所変更)の場合
		if (JKKStrConst.CD00576_TK_DSL.equals(inIdoDiv) || JKKStrConst.CD00576_ADCHG_ADD.equals(inIdoDiv))
		{
			// ローゼット設置状態コードが"1"(設置済)であれば
			if (JKKStrConst.ROZETT_STAT_CD_SETZUMI.equals(rozzetStatCd))
			{
				// 工事スキップ可否チェック結果は"1"(工事スキップ可）
				kojiSkipResult = CHK_KOJISKIP_KA;
			}
			// "1"(設置済)以外は"2"(工事スキップ不可)
			else
			{
				kojiSkipResult = CHK_KOJISKIP_FUKA;
			}
		}
		// ANK-4315-00-00 MOD START
//		// 異動区分が"00009"(コース変更)の場合
//		else if(JKKStrConst.CD00576_00009.equals(inIdoDiv))
		// ANK-4431-00-00 MOD START
//		// 異動区分が"00009"(コース変更)または異動区分が"00041"(光電話・番号追加)または"00026"(機器契約変更)または"00002"(サービス追加)の場合
//		else if(JKKStrConst.CD00576_00009.equals(inIdoDiv) || JKKStrConst.CD00576_00041.equals(inIdoDiv) || JKKStrConst.CD00576_00026.equals(inIdoDiv) || JKKStrConst.CD00576_00002.equals(inIdoDiv))
		// 異動区分が"00009"(コース変更)または異動区分が"00041"(光電話・番号追加)または"00026"(機器契約変更)または"00002"(サービス追加)または"00093"(ONU計画交換)の場合
		else if(JKKStrConst.CD00576_00009.equals(inIdoDiv) || JKKStrConst.CD00576_00041.equals(inIdoDiv) || JKKStrConst.CD00576_00026.equals(inIdoDiv) || JKKStrConst.CD00576_00002.equals(inIdoDiv) || JKKStrConst.CD00576_00093.equals(inIdoDiv))
		// ANK-4431-00-00 MOD END
		// ANK-4315-00-00 MOD END
		{
			// ローゼット設置状態コードが"1"(設置済)の場合
			if (JKKStrConst.ROZETT_STAT_CD_SETZUMI.equals(rozzetStatCd))
			{
				// 申込収容位置空き情報一意照会を実行
				Object[][] eku0281a010IN = {
						{EKU0281A010CBSMsg.KEY_SVC_KEI_KAISEN_UCWK_NO, inSvcKeiKaisenUcwkCd}
				
				};
				// ◇ SC呼出実行
				CAANMsg eku0281a010Msg = callSC (handle, param, scCall, TEMPLATE_ID_EKU0281A010, "2",fixedText,  eku0281a010IN);
				// 処理結果
				CAANMsg[] eku0281a010MsgList = eku0281a010Msg.getCAANMsgList(EKU0281A010CBSMsg.EKU0281A010CBSMSG1LIST);
				if (eku0281a010MsgList != null && 0 < eku0281a010MsgList.length)
				{
					template = eku0281a010MsgList[0];
					// 親機種別コード取得
					String oyaKikiSbtCd = template.getString(EKU0281A010CBSMsg1List.OYA_KIKI_SBT_CD);
					// ANK-4431-00-00 MOD START
//					// CT空き有無取得
//					String ctAkiUm = template.getString(EKU0281A010CBSMsg1List.CT_AKI_UM);
//					// 親機種別コードが"04"(10G-EPON)かつCT空き有無が"1"(あり)の場合は
//					if (JKKStrConst.OYAK_SBT_CD_10G_EPON.equals(oyaKikiSbtCd) && JKKStrConst.CT_AKI_UM_ARI.equals(ctAkiUm))
					// 親機種別コードが"04"(10G-EPON)の場合は
					if (JKKStrConst.OYAK_SBT_CD_10G_EPON.equals(oyaKikiSbtCd))
					// ANK-4431-00-00 MOD END
					{
						// 工事スキップ可否チェック結果は"1"(工事スキップ可）
						kojiSkipResult = CHK_KOJISKIP_KA;
					}
					// それ以外は"2"(工事スキップ不可)
					else
					{
						kojiSkipResult = CHK_KOJISKIP_FUKA;
					}
				}
			}
			// ローゼット設置状態コードが"1"(設置済)以外の場合
			else
			{
				// チェック結果は"2"(工事スキップ不可)
				kojiSkipResult = CHK_KOJISKIP_FUKA;
			}
		// 上記異動区分以外が渡されていた場合は判定不可とする。
		}else{
			kojiSkipResult = CHK_JDGEFUKA;
		}
		return kojiSkipResult;
	}
	/**
	 * マップから値を取得
	 * 
	 * @param map 値を取り出すマップ
	 * @param key キー
	 * @return 取り出した値
	 */
	private String getMapValue(HashMap<String, Object> map, String key)
	{
		String str = (String)map.get(key);

		if (str == null)
		{
			return "";
		}

		return str;
	}
	
	/**
	 * SC呼出処理
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param scCall SC呼出部品のインスタンス
	 * @param svcIfId サービスIFのID
	 * @param funcCode 機能コード
	 * @param svcIfMapData 上り設定用データ
	 * @return メッセージ
	 * @throws Exception SC実行時の例外
	 */
	@SuppressWarnings("unchecked")
	private CAANMsg callSC(SessionHandle handle, 
							IRequestParameterReadWrite param, 
							ServiceComponentRequestInvoker scCall,
							String svcIfId,
							String funcCode,
							String fixedText,
							Object[][] svcIfMapData) throws Exception
	{
		// ◇ 上りマッピング処理
		// 上りマッピングの処理結果を取得するためのマップを生成
		HashMap<String, Object> paramMap = editInMsg(param, svcIfId, funcCode, svcIfMapData);

		// SC呼出実行
		Map<?, ?> result = scCall.run(paramMap, handle);

		// 処理結果の判定
		CAANMsg[] templates = (CAANMsg[])result.get(JCMConstants.TEMPLATE_LIST_KEY);

		// 取得したリターンコード、ステータスの内容を見て異常かどうかの判断をする。
		String rtnCode = result.get(JCMConstants.RET_CD_INT_KEY).toString();

		// エラー情報の転記
		editErrorInfo(param, templates, fixedText, (Integer)result.get(JCMConstants.RET_CD_INT_KEY));

		// エラー情報のマップを取得
		ArrayList<Object> errList = (ArrayList<Object>)param.getControlMapData(SCControlMapKeys.ERROR_INFO);
		if (errList == null)
		{
			errList = new ArrayList<Object>();
		}
		// コントロールマップに設定
		param.setControlMapData(SCControlMapKeys.ERROR_INFO, TemplateErrorUtil.getErrorInfo(result, errList));

		int status = templates[0].getInt(STATUS);
		// 異常の場合、SCCallExceptionを生成してスローする
		if (!("0".equals(rtnCode) && 0 == status))
		{
			throw new SCCallException("戻り値不正", rtnCode, status);
		}

		return ((CAANMsg[])result.get(JCMConstants.TEMPLATE_LIST_KEY))[0];
	}

	/**
	 * 上りマッピング処理
	 * 
	 * @param param リクエストパラメータ
	 * @param svcIfId サービスIFのID
	 * @param funcCode 機能コード
	 * @param mapData 上り設定用データ
	 * @return 上りを設定したマップ
	 * @throws RequestParameterException リクエストパラメータ操作時の例外
	 */
	private HashMap<String, Object> editInMsg(IRequestParameterReadWrite param, 
												String svcIfId, 
												String funcCode,
												Object[][] mapData) throws RequestParameterException
	{
		HashMap<String, Object> paramMap = new HashMap<String, Object>();

		// 【取得元：電文ヘッダ(ヘッダ)】
		// 電文ID
		paramMap.put(JCMConstants.TRANZACTION_ID_KEY, param.getTelegramID());
		// ユースケースID
		paramMap.put(JCMConstants.USECASE_ID_KEY, param.getUsecaseID());
		// オペレーションID
		paramMap.put(JCMConstants.OPERATION_ID_KEY, param.getOperationID());
		// サービス呼び出し区分
		paramMap.put(JCMConstants.CALL_TYPE_KEY, param.getCallType());

		// 【取得元：ユーザエリア(コントロールマップ)】
		// 依頼先ホスト名
		paramMap.put(JCMConstants.CLIENT_HOST_NAME_KEY, param.getControlMapData(SCControlMapKeys.REQ_HOSTNAME));
		// 依頼元IPアドレス
		paramMap.put(JCMConstants.CLIENT_IP_ADDRESS_KEY, param.getControlMapData(SCControlMapKeys.REQ_HOSTIP));
		// 依頼元画面ID
		paramMap.put(JCMConstants.INVOKE_GAMEN_ID_KEY, param.getControlMapData(SCControlMapKeys.REQ_VIEWID));
		// オペレータID
		paramMap.put(JCMConstants.OPERATOR_ID_KEY, param.getControlMapData(SCControlMapKeys.OPERATOR_ID));

		CAANMsg template = new CAANMsg("eo.ejb.cbs.cbsmsg." + svcIfId + "CBSMsg");

		// テンプレートID
		template.set(TEMPLATEID, svcIfId);

		template.set(FUNC_CODE, funcCode);

		// オペレータID
		Object operatorId = param.getControlMapData(SCControlMapKeys.OPERATOR_ID);
		template.set(JCMConstants.OPERATOR_ID_KEY, operatorId);

		// 運用日付
		Object operateDate = param.getControlMapData(SCControlMapKeys.OPE_DATE);
		template.set(JCMConstants.OPERATE_DATE_KEY, operateDate);

		// 運用日時
		Object operateDateTime = param.getControlMapData(SCControlMapKeys.OPE_TIME);
		template.set(JCMConstants.OPERATE_DATETIME_KEY, operateDateTime);

		for (int i = 0; i < mapData.length; i++)
		{
			Object[] mapVal = mapData[i];
			if (mapVal[1] instanceof byte[])
			{
				template.set((String)mapVal[0], (byte[])mapVal[1]);
			}
			else
			{
				if ("".equals(mapVal[1]))
				{
					template.setNull((String)mapVal[0]);
				}
				else
				{
					template.set((String)mapVal[0], (String)mapVal[1]);
				}
			}
		}

		CAANMsg[] templates = new CAANMsg[1];
		templates[0] = template;
		paramMap.put(JCMConstants.TEMPLATE_LIST_KEY, templates);

		return paramMap;
	}

}
