/*********************************************************************
* All Rights reserved,Copyright (c) K-Opticom
**********************************************************************
*＜プログラム内容＞
*	システム名		：eo顧客基幹システム
*	モジュール名	：JKKNinshoIdInfoCC
*	ソースファイル名：JKKNinshoIdInfoCC.java
*	作成者			：富士通
*	日付			：2011年12月13日
*＜機能概要＞
*	料金コース情報照会CC
*＜修正履歴＞
*	バージョン	修正日		修正者		修正内容
*	v1.00.00	2011/12/13	FAP)古内		新規作成
*	v3.00.00	2012/08/05	FJ)団		【ANK-0545-00-00】サービスコード対応
*	v3.00.01	2012/08/20	FJ)団		【ANK-0546-02-00：イオスマートリンク対応】
*	v3.00.02	2012/08/29	FJ)団		 ST1-2012-0000462対応
*	v3.01.00	2012/09/26	FJ)強矢		 ST1-2012-0000606対応
*
**********************************************************************/
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.ejb.cbs.cbsmsg.EKK0161B004CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0161B004CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0171A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0171A010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0351A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0351A010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0351B001CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0351B001CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0361A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0361A010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0811A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0811A010CBSMsg1List;


/**
 * 認証ID情報の照会を行う部品。
 * 
 * @author 909141
 *
 */
public class JKKNinshoIdInfoCC extends AbstractCommonComponent
{

	/** CBSMsgのテンプレートID項目 */
	private static final String TEMPLATEID = "templateID";
	
	/** CBSMsgの機能コード項目 */
	private static final String FUNC_CODE = "func_code";
	
	/** CBSMsgのステータス項目 */
	private static final String STATUS = "status";
	
	/** テンプレートID(EKK0811A010) */
	private static final String TEMPLATE_ID_EKK0811A010 = "EKK0811A010";
	
	/** テンプレートID(EKK0161B004) */
	private static final String TEMPLATE_ID_EKK0161B004 = "EKK0161B004";
	
	/** テンプレートID(EKK0171A010) */
	private static final String TEMPLATE_ID_EKK0171A010 = "EKK0171A010";
	
	/** テンプレートID(EKK0351B001) */
	private static final String TEMPLATE_ID_EKK0351B001 = "EKK0351B001";
	
	/** テンプレートID(EKK0351A010) */
	private static final String TEMPLATE_ID_EKK0351A010 = "EKK0351A010";
	
	/** テンプレートID(EKK0361A010) */
	private static final String TEMPLATE_ID_EKK0361A010 = "EKK0361A010";
	
	/** サービスID */
	private static final String SERVICE_ID = "KKSV033408CC";
	
	
	/**
	 * 認証ID情報の照会を行う。
	 * 
	 * @param handle
	 * @param param
	 * @param fixedText
	 * @return IRequestParameterReadWrite
	 * @throws Exception
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite searchNinshoIdInfo(SessionHandle handle, 
															IRequestParameterReadWrite param, 
															String fixedText) throws Exception
	{
		HashMap<String, Object> work02Map = new HashMap<String, Object>();
		param.setData("KKSV0334WORK02", work02Map);
		
		// ◇ SC呼出実行
		// SC呼び出し部品のインスタンス生成（引数にはログに出力するクラス名を渡す。空文字を設定した場合はログに出力されない）
		ServiceComponentRequestInvoker scCall = new ServiceComponentRequestInvoker();
		
		HashMap<String, Object> ccMsg = (HashMap<String, Object>)param.getData(SERVICE_ID);
		HashMap<String, Object> work03Map = (HashMap<String, Object>)param.getData("KKSV0334WORK03");
		
		// *** EKK0811A010(料金コース_オプションサービス一意照会) の実行 ***
		Object[][] ekk0811a010IN = {
				// CC.ＫＥＹ＿料金コースコード => ＫＥＹ＿料金コースコード
				{EKK0811A010CBSMsg.KEY_PCRS_CD, work03Map.get("pcrs_cd")},
				// "B015"(マルチセッション) => ＫＥＹ＿オプションサービスコード
				{EKK0811A010CBSMsg.KEY_OP_SVC_CD, "B015"},
				// "" => ＫＥＹ＿世代登録年月日時分秒
				{EKK0811A010CBSMsg.KEY_GENE_ADD_DTM, ""},
				// 運用日 => ＫＥＹ＿予約適用年月日
				{EKK0811A010CBSMsg.KEY_RSV_APLY_YMD, JPCBPCommon.getOpeDate(null)}
		};
		
		// ◇ SC呼出実行
		CAANMsg ekk0811a010Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0811A010, "2", ekk0811a010IN);
		
		// 処理結果判定
		CAANMsg[] ekk0811a010MsgList = ekk0811a010Msg.getCAANMsgList(EKK0811A010CBSMsg.EKK0811A010CBSMSG1LIST);
		
		// 最大オプションサービス数
		String max_op_svc_cnt = "0";
		
		if (ekk0811a010MsgList != null && 0 < ekk0811a010MsgList.length)
		{
			if (ekk0811a010MsgList[0].getString(EKK0811A010CBSMsg1List.MAX_OP_SVC_CNT) != null
					&& !"".equals(ekk0811a010MsgList[0].getString(EKK0811A010CBSMsg1List.MAX_OP_SVC_CNT)))
			{
				max_op_svc_cnt = ekk0811a010MsgList[0].getString(EKK0811A010CBSMsg1List.MAX_OP_SVC_CNT);
			}
		}
		
		// *** EKK0161B004(サービス契約内訳一覧照会) の実行 ***
		Object[][] ekk0161b004IN = {
			// cc.ＫＥＹ＿サービス契約番号 => ＫＥＹ＿サービス契約番号
			{EKK0161B004CBSMsg.KEY_SVC_KEI_NO, (String)ccMsg.get("key_svc_kei_no")},
			// "" => ＫＥＹ＿料金コースコード
			{EKK0161B004CBSMsg.KEY_PCRS_CD, ""},
			// "" => ＫＥＹ＿サービス期間中フラグ
			{EKK0161B004CBSMsg.KEY_SVC_PRD_FLG, ""}
		};
		
		// ◇ SC呼出実行
		CAANMsg ekk0161b004Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0161B004, "1", ekk0161b004IN);
		
		// 処理結果判定
		CAANMsg[] ekk0161b004MsgList = ekk0161b004Msg.getCAANMsgList(EKK0161B004CBSMsg.EKK0161B004CBSMSG1LIST);
		
		// 有効な認証ＩＤ数
		int numOfNinshoId = 0;
		
		ArrayList<CAANMsg> ispNinshoIdList = new ArrayList<CAANMsg>();
		
		for (int i = 0; i < ekk0161b004MsgList.length; i++)
		{
			// 料金コースコードがパラメータの料金コースコードと違う場合は対象としない。
			if (!work03Map.get("pcrs_cd").equals(ekk0161b004MsgList[i].getString(EKK0161B004CBSMsg1List.PCRS_CD)))
			{
				continue;
			}
			
			String svcKeiUcwkStat = ekk0161b004MsgList[i].getString(EKK0161B004CBSMsg1List.SVC_KEI_UCWK_STAT);
			// "020"(照査済)"030"(締結済)"100"(サービス提供中)"210"(休止・中断中)"220"(停止中)以外は対象データとしない。
			if (!("020".equals(svcKeiUcwkStat) 
					|| "030".equals(svcKeiUcwkStat) 
					|| "100".equals(svcKeiUcwkStat) 
					|| "210".equals(svcKeiUcwkStat)
					|| "220".equals(svcKeiUcwkStat)))
			{
				continue;
			}
			
			// *** EKK0171A010(サービス契約内訳＜eo光ネット＞一意照会) の実行 ***
			Object[][] ekk0171a010IN = {
				// EKK0161B004.サービス契約内訳番号 => ＫＥＹ＿サービス契約内訳番号
				{EKK0171A010CBSMsg.KEY_SVC_KEI_UCWK_NO, ekk0161b004MsgList[i].getString(EKK0161B004CBSMsg1List.SVC_KEI_UCWK_NO)},
				// EKK0161B004.世代登録年月日時分秒 => ＫＥＹ＿世代登録年月日時分秒
				{EKK0171A010CBSMsg.KEY_GENE_ADD_DTM, ekk0161b004MsgList[i].getString(EKK0161B004CBSMsg1List.GENE_ADD_DTM)}
			};
			
			// ◇ SC呼出実行
			CAANMsg ekk0171a010Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0171A010, "1", ekk0171a010IN);
			
			// 処理結果判定
			CAANMsg[] ekk0171a010MsgList = ekk0171a010Msg.getCAANMsgList(EKK0171A010CBSMsg.EKK0171A010CBSMSG1LIST);
			
			// 検索結果なし、または、認証IDが設定されていない場合はカウントしない。
			if (ekk0171a010MsgList.length == 0) 
			{
				continue;
			}
			if (ekk0171a010MsgList[0].isNull(EKK0171A010CBSMsg1List.ISP_NINSHO_ID))
			{
				continue;
			}
			
			// 認証ＩＤの取得件数をカウントアップする。
			numOfNinshoId++;
			
			ispNinshoIdList.add(ekk0171a010MsgList[0]);
		}
		
		ArrayList<CAANMsg> multiSessionList = new ArrayList<CAANMsg>();
		
		// 認証ＩＤの取得件数が１件の場合
		if (numOfNinshoId == 1)
		{
			// *** EKK0351B001(オプションサービス契約一覧照会（サービス契約内訳番号）) の実行 ***
			Object[][] ekk0351b001IN = {
				// EKK0171A010.サービス契約内訳番号 => ＫＥＹ＿サービス契約内訳番号
				{EKK0351B001CBSMsg.KEY_SVC_KEI_UCWK_NO, ispNinshoIdList.get(0).getString(EKK0171A010CBSMsg1List.SVC_KEI_UCWK_NO)}
			};
			
			// ◇ SC呼出実行
			CAANMsg ekk0351b001Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0351B001, "1", ekk0351b001IN);
			
			// 処理結果判定
			CAANMsg[] ekk0351b001MsgList = ekk0351b001Msg.getCAANMsgList(EKK0351B001CBSMsg.EKK0351B001CBSMSG1LIST);
			
			for (int j = 0; j < ekk0351b001MsgList.length; j++)
			{
				String op_svc_kei_stat = ekk0351b001MsgList[j].getString(EKK0351B001CBSMsg1List.OP_SVC_KEI_STAT);
				// "020"(照査済)"030"(締結済)"100"(サービス提供中)"210"(休止・中断中)"220"(停止中)以外は対象データとしない。
				if (!("020".equals(op_svc_kei_stat) 
						|| "030".equals(op_svc_kei_stat) 
						|| "100".equals(op_svc_kei_stat) 
						|| "210".equals(op_svc_kei_stat)
						|| "220".equals(op_svc_kei_stat)))
				{
					continue;
				}
				
				// *** EKK0351A010(オプションサービス契約一意照会) の実行 ***
				Object[][] ekk0351a010IN = {
					// EKK0351B001.オプションサービス契約番号 => ＫＥＹ＿オプションサービス契約番号
					{EKK0351A010CBSMsg.KEY_OP_SVC_KEI_NO, ekk0351b001MsgList[j].getString(EKK0351B001CBSMsg1List.OP_SVC_KEI_NO)},
					// EKK0351B001.世代登録年月日時分秒 => ＫＥＹ＿世代登録年月日時分秒
					{EKK0351A010CBSMsg.KEY_GENE_ADD_DTM, ekk0351b001MsgList[j].getString(EKK0351B001CBSMsg1List.GENE_ADD_DTM)},
					// "" => ＫＥＹ＿予約適用年月日
					{EKK0351A010CBSMsg.KEY_RSV_APLY_YMD, ""}
				};
				
				// ◇ SC呼出実行
				CAANMsg ekk0351a010Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0351A010, "1", ekk0351a010IN);
				
				// 処理結果判定
				CAANMsg[] ekk0351a010MsgList = ekk0351a010Msg.getCAANMsgList(EKK0351A010CBSMsg.EKK0351A010CBSMSG1LIST);
				
				// ※ 一覧照会(カレント)に対する一意照会(PK指定)であるため、必ずデータが存在する
				
				// オプションサービスコードが「B015(マルチセッション）」以外は対象としない。
				if (!"B015".equals(ekk0351a010MsgList[0].getString(EKK0351A010CBSMsg1List.OP_SVC_CD)))
				{
					continue;
				}
				
				// ST1-2012-0000462対応
				//// 料金コースコードがパラメータの料金コースコードと違う場合は対象としない。
				//if (!work03Map.get("pcrs_cd").equals(ekk0351a010MsgList[0].getString(EKK0351A010CBSMsg1List.PCRS_CD)))
				//{
				//	continue;
				//}
				
				// *** EKK0361A010(オプションサービス契約<ISP>一意照会) の実行 ***
				Object[][] ekk0361a010IN = {
					// EKK0351A010.オプションサービス契約番号 => ＫＥＹ＿オプションサービス契約番号
					{EKK0361A010CBSMsg.KEY_OP_SVC_KEI_NO, ekk0351a010MsgList[0].getString(EKK0351A010CBSMsg1List.OP_SVC_KEI_NO)},
					// EKK0351A010.世代登録年月日時分秒 => ＫＥＹ＿世代登録年月日時分秒
					{EKK0361A010CBSMsg.KEY_GENE_ADD_DTM, ekk0351a010MsgList[0].getString(EKK0351A010CBSMsg1List.GENE_ADD_DTM)}
				};
				
				// ◇ SC呼出実行
				CAANMsg ekk0361a010Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0361A010, "1", ekk0361a010IN);
				
				// 処理結果判定
				CAANMsg[] ekk0361a010MsgList = ekk0361a010Msg.getCAANMsgList(EKK0361A010CBSMsg.EKK0361A010CBSMSG1LIST);
				
				// ※ オプションサービス契約一覧照会結果を元に検索しているため、必ずデータは存在する。
				multiSessionList.add(ekk0361a010MsgList[0]);
				
			}
		}
		
		ArrayList<HashMap<String, Object>> ninshoIdList = new ArrayList<HashMap<String, Object>>();
		
		// *** 下りデータの作成 ***
		// 認証ＩＤの取得件数が１件の場合
		if (numOfNinshoId == 1)
		{
			// (マルチセッションの)オプションサービス契約のレコード件数
			int multiSessionCount = multiSessionList.size();
			
			// 認証ＩＤ情報出力フラグ
			ccMsg.put("login_id_inf_flg", "1");
			// 認証ＩＤ情報数
			ccMsg.put("login_id_inf_num", String.valueOf(numOfNinshoId));
			// マルチセッション現在取得数
			ccMsg.put("get_multisession_num", String.valueOf(multiSessionCount));
			// マルチセッション最大上限数
			ccMsg.put("max_get_multisession", max_op_svc_cnt);
			
			// ISP認証ID
			String isp_ninsho_id = ispNinshoIdList.get(0).getString(EKK0171A010CBSMsg1List.ISP_NINSHO_ID);
			
			// オプションサービス契約が１件以上の場合
			if (multiSessionCount > 0)
			{
				// マルチセクタ[M9]出力は、オプションサービス契約の件数分
				for (int i = 0; i < multiSessionList.size(); i++)
				{
					
					HashMap<String, Object> map = new HashMap<String, Object>();
					
					// ISP認証ID
					map.put("isp_ninsho_id", isp_ninsho_id);
					// マルチセッションサービス数
					map.put("sshon_su", multiSessionList.get(i).getString(EKK0361A010CBSMsg1List.MLTISE_SESSION_CNT));
					
					String multi_nnshou_id = multiSessionList.get(i).getString(EKK0361A010CBSMsg1List.MLTISE_NINSHO_ID);
					if (multi_nnshou_id != null && !"".equals(multi_nnshou_id))
					{
						// マルチセッションご利用認証ID
						// マルチセッション認証IDに値が設定されている場合、マルチセッション認証IDを設定
						map.put("multi_nnshou_id", multi_nnshou_id);
					}
					else
					{
						// マルチセッションご利用認証ID
						// マルチセッション認証IDに値が設定されていない場合、ISP認証IDを設定
						map.put("multi_nnshou_id", isp_ninsho_id);
					}
					
					// マルチセッション申込ボタン出力フラグ
					// 最大オプションサービス数(マルチセッション) - マルチセッション現在取得数 が１以上の場合、"1"を設定
					// 上記以外の場合は、"0"を設定
					map.put("multi_session_flg", (Integer.parseInt(max_op_svc_cnt) - multiSessionCount) > 0 ? "1" : "0");
					
					ninshoIdList.add(map);
				}
			}
			// オプションサービス契約が０件の場合
			else
			{
				// マルチセクタ[M9]出力は、サービス契約内訳の１件目を出力
				
				HashMap<String, Object> map = new HashMap<String, Object>();
				
				// ISP認証ID
				map.put("isp_ninsho_id", isp_ninsho_id);
				// マルチセッションサービス数
				map.put("sshon_su", "0");
				// マルチセッションご利用認証ID に ISP認証IDを設定
				map.put("multi_nnshou_id", isp_ninsho_id);
				// マルチセッション申込ボタン出力フラグ
				map.put("multi_session_flg", "0");
				
				ninshoIdList.add(map);
			}
			ccMsg.put("ninshoIdList", ninshoIdList);
		}
		// 認証ＩＤの取得件数が複数件の場合
		else if (numOfNinshoId > 1)
		{
			// 認証ＩＤ情報出力フラグ
			ccMsg.put("login_id_inf_flg", "1");
			// 認証ＩＤ情報数
			ccMsg.put("login_id_inf_num", String.valueOf(numOfNinshoId));
			// マルチセッション現在取得数
			ccMsg.put("get_multisession_num", "0");
			// マルチセッション最大上限数
			ccMsg.put("max_get_multisession", "0");
			
			// マルチセクタ[M9]出力は、サービス契約内訳の件数分
			for (int i = 0; i < ispNinshoIdList.size(); i++)
			{
				HashMap<String, Object> map = new HashMap<String, Object>();
				
				// ISP認証ID
				map.put("isp_ninsho_id", ispNinshoIdList.get(i).getString(EKK0171A010CBSMsg1List.ISP_NINSHO_ID));
				// マルチセッションサービス数
				map.put("sshon_su", "0");
				// マルチセッションご利用認証ID
				map.put("multi_nnshou_id", ispNinshoIdList.get(i).getString(EKK0171A010CBSMsg1List.ISP_NINSHO_ID));
				// マルチセッション申込ボタン出力フラグ
				map.put("multi_session_flg", "0");
				
				ninshoIdList.add(map);
			}
			ccMsg.put("ninshoIdList", ninshoIdList);
		}
		// 認証ＩＤが取得できなかった場合
		else
		{
			// 認証ＩＤ情報出力フラグ
			ccMsg.put("login_id_inf_flg", "0");
			// 認証ＩＤ情報数
			ccMsg.put("login_id_inf_num", "0");
			// マルチセッション現在取得数
			ccMsg.put("get_multisession_num", null);
			// マルチセッション最大上限数
			ccMsg.put("max_get_multisession", null);
		}
		work02Map.put("return_result_cd", "000000");
		
		return param;
	}
	
	/**
	 * チェック処理用のサービスインターフェイスを作成し、返却する。
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param fixedText サービスメッセージ
	 * @return HashMap<String, Object> サービスインターフェイスマップ
	 * @throws RequestParameterException
	 */
	@SuppressWarnings("unchecked")
	public HashMap<String, Object> getInvokeCBS(SessionHandle handle, 
												IRequestParameterReadWrite param, 
												String fixedText) throws RequestParameterException
	{
		
		ArrayList<CAANMsg> templateList = new ArrayList<CAANMsg>();
		
		HashMap<String, Object> ccMsg = (HashMap<String, Object>)param.getData(SERVICE_ID);
		HashMap<String, Object> work03Map = (HashMap<String, Object>)param.getData("KKSV0334WORK03");
		
		// *** EKK0811A010(料金コース_オプションサービス一意照会) の実行 ***
		Object[][] ekk0811a010IN = {
				// CC.ＫＥＹ＿料金コースコード => ＫＥＹ＿料金コースコード
				{EKK0811A010CBSMsg.KEY_PCRS_CD, work03Map.get("pcrs_cd")},
				// "B015"(マルチセッション) => ＫＥＹ＿オプションサービスコード
				{EKK0811A010CBSMsg.KEY_OP_SVC_CD, "B015"},
				// "" => ＫＥＹ＿世代登録年月日時分秒
				{EKK0811A010CBSMsg.KEY_GENE_ADD_DTM, ""},
				// 運用日 => ＫＥＹ＿予約適用年月日
				{EKK0811A010CBSMsg.KEY_RSV_APLY_YMD, JPCBPCommon.getOpeDate(null)}
		};
		
		// 上りマッピングの処理結果を取得するためのマップを生成
		HashMap<String, Object> paramMap = editInMsg(param, TEMPLATE_ID_EKK0811A010, "2", ekk0811a010IN);
		templateList.add(((CAANMsg[])paramMap.get(JCMConstants.TEMPLATE_LIST_KEY))[0]);
		
		// *** EKK0161B004(サービス契約内訳一覧照会) の実行 ***
		Object[][] ekk0161b004IN = {
				// cc.ＫＥＹ＿サービス契約番号 => ＫＥＹ＿サービス契約番号
				{EKK0161B004CBSMsg.KEY_SVC_KEI_NO, (String)ccMsg.get("key_svc_kei_no")},
				// "" => ＫＥＹ＿料金コースコード
				{EKK0161B004CBSMsg.KEY_PCRS_CD, ""},
				// "" => ＫＥＹ＿サービス期間中フラグ
				{EKK0161B004CBSMsg.KEY_SVC_PRD_FLG, ""}
			};
		
		// ◇ 上りマッピング処理
		// 上りマッピングの処理結果を取得するためのマップを生成
		paramMap = editInMsg(param, TEMPLATE_ID_EKK0161B004, "1", ekk0161b004IN);
		templateList.add(((CAANMsg[])paramMap.get(JCMConstants.TEMPLATE_LIST_KEY))[0]);
		
		// リストに追加されたCAANMsgを配列に変換し、TEMPLATE_LIST_KEYに入れなおす
		CAANMsg[] templates = templateList.toArray(new CAANMsg[0]);
		paramMap.put(JCMConstants.TEMPLATE_LIST_KEY, templates);
		
		return paramMap;
	}
	
	/**
	 * SC呼出
	 * @param handle
	 * @param param
	 * @param scCall
	 * @param svcIfId
	 * @param funcCode
	 * @param svcIfMapData
	 * @return CAANMsg
	 * @throws Exception
	 */
	@SuppressWarnings("unchecked")
	private CAANMsg callSC(SessionHandle handle, 
							IRequestParameterReadWrite param, 
							ServiceComponentRequestInvoker scCall,
							String svcIfId,
							String funcCode,
							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, (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
	 * @param funcCode
	 * @param mapData
	 * @return HashMap<String, Object>
	 * @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 ("".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;
	}
	
	/**
	 * サービスインターフェイスのエラー情報をマッピングする。
	 * <br>
	 * @param param
	 * @param templates
	 * @param returnCode
	 * @return リクエストパラメータ
	 * @throws RequestParameterException
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite editErrorInfo(IRequestParameterReadWrite param, 
													CAANMsg[] templates,
													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 inMap = (HashMap)param.getData(SERVICE_ID);
			
			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;
	}
}
