/*********************************************************************
* All Rights reserved,Copyright (c) K-Opticom
**********************************************************************
*＜プログラム内容＞
*	システム名		：eo顧客基幹システム
*	モジュール名	：JKKKojihiKapKeiUpdCC
*	ソースファイル名：JKKKojihiKapKeiUpdCC.java
*	作成者			：FJ
*	日付			：2018年01月25日
*＜機能概要＞
*	工事費割賦契約更新コモンコンポーネント部品
*＜修正履歴＞
*   バージョン  修正日       修正者      修正内容
*	v36.00.00	2018/01/25	 FJ)澤田	 ANK-3296-00-00
*	v36.00.01	2018/02/08	 FJ)麻生	 IT1-2018-0000020
*	v37.00.00	2018/06/01	 FJ)麻生	 【IT2-2018-0000042】課金開始されていない契約が処理対象とならない（水平展開）
*	v37.00.01	2018/06/06	 FJ)麻生	 【IT1-2018-0000080】割賦契約課金開始年月日が変更されない
*   v45.00.00	2019/07/19	 FJ)吉田	 【ANK-3636-00-00】消費増税対応（8%⇒10%）
*   v53.00.00	2021/04/05	 FJ)謝		 【ANK-3951-00-00】「競争ルールの検証に関するWG」に関するシステム対応
**********************************************************************/
package com.fujitsu.futurity.bp.custom.common;

import java.text.ParseException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
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.CCException;
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 com.fujitsu.futurity.rule.common.JRuleXMLHandler;
import com.fujitsu.futurity.rule.xml.compare.JRuleXMLPattern;

import eo.common.constant.JKKStrConst;
import eo.common.constant.JPCModelConstant;
import eo.common.util.JKKCommonUtil;
import eo.common.util.JKKStringUtil;
import eo.common.util.JPCDateUtil;
import eo.ejb.cbs.cbsmsg.EKK0081A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0081A010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0241B002CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0241B002CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK1091D010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK1091D010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK3131C010CBSMsg;

/**
 * 工事費割賦契約更新コモンコンポーネント部品を行います。<p>
 * <BR>
 * @author FJ)澤田
 */
public class JKKKojihiKapKeiUpdCC 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 */
	private static final String SERVICE_ID = "KKSV088601CC";
	
	/** 工事費割賦契約更新 */
	private static final String TEMPLATE_ID_EKK3131C010 = "EKK3131C010";
	
	/** テンプレートID：進捗登録 */
	private static final String TEMPLATE_ID_EKK1091D010 = "EKK1091D010";
	
	/** 回線対象サービス契約一覧照会 */
	private static final String TEMPLATE_ID_EKK0241B002 = "EKK0241B002";
	
	// IT2-2018-0000042 ADD START
	/** サービス契約一意照会 */
	private static final String TEMPLATE_ID_EKK0081A010 = "EKK0081A010";

	/** ルール定義（XMLファイル）格納パス取得用キー */
	private static final String KEY_SOKAN_CHECK_XML_PATH = "SOKAN_CHECK_XML_PATH";

	/** 課金開始月設定(工事費割賦契約)ルール */
	private static final String RULE0093001 = "RULE0093001";

	/** 判定結果コード(正常) */
	private static final String JDG_RSLT_CD_NORMAL = "1";

	/** 基準日(サービス開始年月日) */
	private static final String STD_DT_SVC_STAYMD = "1";

	/** 基準日(サービス課金開始年月日) */
	private static final String STD_DT_SVC_CHRG_STAYMD = "2";

	/** 当月/翌月(翌月1日) */
	private static final String TOGTU_YKGTU_YKGTU = "1";

	/** 当月/翌月(当月) */
	private static final String TOGTU_YKGTU_TOGTU = "2";
	// IT2-2018-0000042 ADD END
	
	/** 入力項目：異動理由明細 */
	private static final String IDO_RSN_CD_LIST = "ido_rsn_cd_list";
	/** 入力項目：異動理由コード */
	private static final String IDO_RSN_CD = "ido_rsn_cd";
	/** 入力項目：異動理由メモ */
	private static final String IDO_RSN_MEMO = "ido_rsn_memo";

	/**
	 * 工事費割賦契約更新コモンコンポーネント部品
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param fixedText サービスメッセージ
	 * @return リクエストパラメータ
	 * @throws Throwable スロー可能オブジェクト
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite updKojihiKapKei(SessionHandle handle, IRequestParameterReadWrite param, String fixedText) throws Throwable
	{
		HashMap<String, Object> ccMsg = (HashMap<String, Object>)param.getData(fixedText);
		HashMap<String, Object> paramMap = null;
		Map<?, ?> result = null;
		//パラメータがNULLの場合、処理を終了する。
		if (ccMsg == null)
		{
			return param;
		}
		
		// SC呼び出し部品
		ServiceComponentRequestInvoker scCall = new ServiceComponentRequestInvoker();
		//回線対象サービス契約一覧照会
		List<Map<String, Object>> eKK0241B002MsgList = new ArrayList<Map<String, Object>>();
		
		// IT2-2018-0000042 ADD START
		// 初回請求サービス契約番号
		String firstSeikySvcKeiNo = getMapValue(ccMsg, "first_seiky_svc_kei_no");
		
		if (!JKKStringUtil.isNullBlank(firstSeikySvcKeiNo))
		{
			// サービス契約一意照会
			HashMap<String, Object> svcKeiInf = callGetSvcKei(handle, param, scCall, firstSeikySvcKeiNo);
			
			// 割賦契約課金開始年月日の設定
			setKapKeiChrgStaymd(svcKeiInf, ccMsg);
		}
		// IT2-2018-0000042 ADD END
		
		//工事費割賦契約更新
		paramMap = editInMsgUpd(param, ccMsg);
		
		result = scCall.run(paramMap, handle);
		
		// エラー情報の設定
		// IT2-2018-0000042 MOD START
//		IRequestParameterReadWrite wParam = editResultRP(result, param, TEMPLATE_ID_EKK3131C010);
		editResultRP(result, param, TEMPLATE_ID_EKK3131C010);
		// IT2-2018-0000042 MOD END
		
		checkExecutionResult(result);
		
		//サービス契約番号
		String svc_kei_kaisen_ucwk_no = getMapValue(ccMsg, "svc_kei_kaisen_ucwk_no");
		
		//回線対象サービス契約一覧照会SC(EKK0241B002)を取得
		// IT2-2018-0000042 MOD START
//		eKK0241B002MsgList = callGetKaisenTgSvkei(handle, wParam, scCall, svc_kei_kaisen_ucwk_no);
		eKK0241B002MsgList = callGetKaisenTgSvkei(handle, param, scCall, svc_kei_kaisen_ucwk_no);
		// IT2-2018-0000042 MOD END
		
		//ネット・電話のサービスに進捗を登録する。
		for (int i = 0; i < eKK0241B002MsgList.size(); i++)
		{
			// IT1-2018-0000020 MOD START
//			HashMap<String,Object> eKK0241B002Msg = (HashMap<String,Object>)eKK0241B002MsgList.get(0);
			HashMap<String, Object> eKK0241B002Msg = (HashMap<String, Object>)eKK0241B002MsgList.get(i);
			// IT1-2018-0000020 MOD END

			String svcCd = JKKStringUtil.nullToBlank((String)eKK0241B002Msg.get(EKK0241B002CBSMsg1List.SVC_CD));
			String svcKeiNo = JKKStringUtil.nullToBlank((String)eKK0241B002Msg.get(EKK0241B002CBSMsg1List.SVC_KEI_NO));	
			//サービスコードが"01"ネットの場合　または　サービスコードが"02"電話
			if ("01".equals(svcCd) || "02".equals(svcCd))
			{
				// IT2-2018-0000042 MOD START
//				addPrg(handle, wParam, fixedText, ccMsg, svcKeiNo);
				addPrg(handle, param, fixedText, ccMsg, svcKeiNo);
				// IT2-2018-0000042 MOD END
			}
		}
		
		// IT2-2018-0000042 MOD START
//		return wParam;
		return param;
		// IT2-2018-0000042 MOD END
	}
	
	/**
	 * 工事費割賦契約更新のサービスI/Fのメッセージを作成します。
	 * 
	 * @param param リクエストパラメータ
	 * @param inMap 工事費割賦契約更新CCマップ
	 * @return HashMap<String, Object>
	 * @throws RequestParameterException リクエストパラメータ操作時の例外
	 */
	private HashMap<String, Object> editInMsgUpd(IRequestParameterReadWrite param,
			HashMap<String, Object> inMap) 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(EKK3131C010CBSMsg.class.getName());
		
		// テンプレートID
		template.set(EKK3131C010CBSMsg.TEMPLATEID, TEMPLATE_ID_EKK3131C010);
		
		// 機能コード
		String funcCode = (String)inMap.get("func_code");
		
		// 機能コードが設定されている場合
		if (!JKKCommonUtil.isNull(funcCode))
		{
			template.set(EKK3131C010CBSMsg.FUNC_CODE, funcCode);
		}
		// 機能コードが設定されていない場合
		else
		{
			template.set(EKK3131C010CBSMsg.FUNC_CODE, JPCModelConstant.FUNC_CD_1);
		}
		
		// オペレータ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);
		
		// ◇ INデータの設定
		//FROM:工事費割賦契約更新:工事費割賦契約番号 TO:工事費割賦契約:工事費割賦契約番号
		template.set(EKK3131C010CBSMsg.KOJIHI_KAP_KEI_NO, inMap.get(EKK3131C010CBSMsg.KOJIHI_KAP_KEI_NO));
		//FROM:工事費割賦契約更新:割賦プランコード TO:工事費割賦契約:割賦プランコード
		template.set(EKK3131C010CBSMsg.KAP_PLAN_CD, inMap.get(EKK3131C010CBSMsg.KAP_PLAN_CD));
		//FROM:工事費割賦契約更新:割賦一括請求切替年月 TO:工事費割賦契約:割賦一括請求切替年月
		template.set(EKK3131C010CBSMsg.KAP_IKT_SIKY_SWCH_YM, inMap.get(EKK3131C010CBSMsg.KAP_IKT_SIKY_SWCH_YM));
		//FROM:工事費割賦契約更新:割賦請求開始年月 TO:工事費割賦契約:割賦請求開始年月
		template.set(EKK3131C010CBSMsg.KAP_SEIKY_STA_YM, inMap.get(EKK3131C010CBSMsg.KAP_SEIKY_STA_YM));
		//FROM:工事費割賦契約更新:割賦支払い残回数 TO:工事費割賦契約:割賦支払い残回数
		template.set(EKK3131C010CBSMsg.KAP_PAY_ZAN_CNT, inMap.get(EKK3131C010CBSMsg.KAP_PAY_ZAN_CNT));
		// IT2-2018-0000042 ADD START
		//FROM:工事費割賦契約更新:割賦契約課金開始年月日 TO:工事費割賦契約:割賦契約課金開始年月日
		template.set(EKK3131C010CBSMsg.KAP_KEI_CHRG_STAYMD, inMap.get(EKK3131C010CBSMsg.KAP_KEI_CHRG_STAYMD));
		// IT2-2018-0000042 ADD END
		// FROM:工事費割賦契約更新:更新年月日時分秒(更新前) 工事費割賦契約更新:更新年月日時分秒(更新前)
		template.set(EKK3131C010CBSMsg.UPD_DTM_BF, inMap.get(EKK3131C010CBSMsg.UPD_DTM_BF));
		
		CAANMsg[] templates = new CAANMsg[1];
		templates[0] = template;
		paramMap.put(JCMConstants.TEMPLATE_LIST_KEY, templates);
		
		return paramMap;
	}
	
	/**
	 * 進捗を登録します。
	 * <br>
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param fixedText ユーザ定義文字列
	 * @param ccMsg 工事費割賦契約更新CCマップ
	 * @param svcKeiNo サービス契約番号
	 * @throws Throwable 例外が発生した場合
	 */
	@SuppressWarnings("unchecked")
	private void addPrg(SessionHandle handle, IRequestParameterReadWrite param, String fixedText, Map<String, Object> ccMsg, String svcKeiNo)
	throws Throwable
	{
		// SC呼び出し部品のインスタンス生成（引数にはログに出力するクラス名を渡す。空文字を設定した場合はログに出力されない）
		ServiceComponentRequestInvoker scCall = new ServiceComponentRequestInvoker();

		// 異動理由明細の作成
		List<Map<String, String>> idoRsnCdList = (List<Map<String, String>>)ccMsg.get(IDO_RSN_CD_LIST);
		CAANMsg[] ekk1091d010Msg1List = new CAANMsg[idoRsnCdList.size()];
		for (int i = 0; i < idoRsnCdList.size(); i++)
		{
			CAANMsg childTemplate = new CAANMsg(EKK1091D010CBSMsg1List.class.getName());
			// 異動理由コード
			childTemplate.set(EKK1091D010CBSMsg1List.IDO_RSN_CD, (String)idoRsnCdList.get(i).get(IDO_RSN_CD));
			// 異動理由メモ
			childTemplate.set(EKK1091D010CBSMsg1List.IDO_RSN_MEMO, (String)idoRsnCdList.get(i).get(IDO_RSN_MEMO));
			ekk1091d010Msg1List[i] = childTemplate;
		}

		// 進捗登録の上りマッピング
		Object[][] ekk1091d010In =
		{
			// サービス契約番号
			{EKK1091D010CBSMsg.SVC_KEI_NO, svcKeiNo}
			// 異動区分
			, {EKK1091D010CBSMsg.IDO_DIV, (String)ccMsg.get("ido_div")}
			// 異動年月日時分秒
			, {EKK1091D010CBSMsg.IDO_DTM, JCCBPCommon.getSysDateTimeStamp()}
			// 進捗ステータス
			, {EKK1091D010CBSMsg.PRG_STAT, "H032"}
			// 進捗特記事項1
			, {EKK1091D010CBSMsg.PRG_TKJK_1, (String)ccMsg.get("prg_tkjk_1")}
			// 異動理由明細
			, {EKK1091D010CBSMsg.EKK1091D010CBSMSG1LIST, ekk1091d010Msg1List}
		};

		// 機能コード
		String funcCode = (String)ccMsg.get("func_code");

		// 機能コードが設定されていない場合
		if (JKKCommonUtil.isNull(funcCode))
		{
			funcCode = JPCModelConstant.FUNC_CD_1;
		}

		// サービスIF実行
		callSC(handle, param, scCall, TEMPLATE_ID_EKK1091D010, funcCode, ekk1091d010In);
	}
	
	/**
	 * 回線対象サービス契約一覧照会SC(EKK0241B002)を呼び出す。
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param scCall SC呼出部品のインスタンス
	 * @param svcKeiKaisenUcwkNo サービス契約回線内訳番号
	 * @return 回線対象サービス契約一覧照会の結果
	 * @throws Exception SC実行時の例外
	 */
	private List<Map<String, Object>> callGetKaisenTgSvkei(SessionHandle handle, 
			IRequestParameterReadWrite param, 
			ServiceComponentRequestInvoker scCall,
			String svcKeiKaisenUcwkNo) throws Exception
	{
		// 回線対象サービス契約一覧照会の結果
		List<Map<String, Object>> list = new ArrayList<Map<String, Object>>();
		
		// 回線対象サービス契約一覧照会の検索パラメータを設定
		Object[][] eKK0241B002IN = {
				{EKK0241B002CBSMsg.KEY_SVC_KEI_KAISEN_UCWK_NO, svcKeiKaisenUcwkNo}	// ＫＥＹ＿サービス契約回線内訳番号

		};

		// ◇ SC呼出実行
		CAANMsg eKK0241B002Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0241B002, "1", eKK0241B002IN);

		// 処理結果
		CAANMsg[] eKK0241B002MsgList = eKK0241B002Msg.getCAANMsgList(EKK0241B002CBSMsg.EKK0241B002CBSMSG1LIST);
		if (eKK0241B002MsgList != null && 0 < eKK0241B002MsgList.length)
		{
			for (CAANMsg cAANMsg : eKK0241B002MsgList)
			{
				Map<String, Object> map = new HashMap<String, Object>();
				map.put(EKK0241B002CBSMsg1List.SVC_KEI_KAISEN_UCWK_NO, cAANMsg.getString(EKK0241B002CBSMsg1List.SVC_KEI_KAISEN_UCWK_NO));
				map.put(EKK0241B002CBSMsg1List.SVC_KEI_NO, cAANMsg.getString(EKK0241B002CBSMsg1List.SVC_KEI_NO));
				map.put(EKK0241B002CBSMsg1List.SVC_CD, cAANMsg.getString(EKK0241B002CBSMsg1List.SVC_CD));
				map.put(EKK0241B002CBSMsg1List.SVC_KEI_STAT, cAANMsg.getString(EKK0241B002CBSMsg1List.SVC_KEI_STAT));
				map.put(EKK0241B002CBSMsg1List.KAISEN_UCWK_USE_ENDYMD, cAANMsg.getString(EKK0241B002CBSMsg1List.KAISEN_UCWK_USE_ENDYMD));
				map.put(EKK0241B002CBSMsg1List.PRC_GRP_NM, cAANMsg.getString(EKK0241B002CBSMsg1List.PRC_GRP_NM));
				list.add(map);
			}
		}
		// 処理結果
		return list;
	}
		
	// IT2-2018-0000042 ADD START
	/**
	 * サービス契約一覧照会SC(EKK0081A010)を呼び出す。
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param scCall SC呼出部品のインスタンス
	 * @param svcKeiNo サービス契約番号
	 * @return サービス契約一意照会の結果
	 * @throws Exception SC実行時の例外
	 */
	private HashMap<String, Object> callGetSvcKei(SessionHandle handle, 
			IRequestParameterReadWrite param, 
			ServiceComponentRequestInvoker scCall,
			String svcKeiNo) throws Exception
	{
		// サービス契約一覧照会の結果
		HashMap<String, Object> map = new HashMap<String, Object>();
		
		// サービス契約一覧照会の検索パラメータを設定
		Object[][] eKK0081A010IN = {
				{EKK0081A010CBSMsg.KEY_SVC_KEI_NO, svcKeiNo},	// ＫＥＹ＿サービス契約番号
				// IT1-2018-0000080 MOD START
//				{EKK0081A010CBSMsg.KEY_RSV_APLY_YMD, param.getControlMapData(SCControlMapKeys.OPE_DATE)}	// ＫＥＹ＿予約適用年月日
				{EKK0081A010CBSMsg.KEY_RSV_APLY_YMD, JPCBPCommon.getOpeDate(null)}	// ＫＥＹ＿予約適用年月日
				// IT1-2018-0000080 MOD END
		};

		// ◇ SC呼出実行
		CAANMsg eKK0081A010Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0081A010, "2", eKK0081A010IN);

		// 処理結果
		CAANMsg[] eKK0081A010MsgList = eKK0081A010Msg.getCAANMsgList(EKK0081A010CBSMsg.EKK0081A010CBSMSG1LIST);
		if (eKK0081A010MsgList != null && 0 < eKK0081A010MsgList.length)
		{
			CAANMsg cAANMsg = eKK0081A010MsgList[0];
			// 料金グループコード
			map.put(EKK0081A010CBSMsg1List.PRC_GRP_CD, cAANMsg.getString(EKK0081A010CBSMsg1List.PRC_GRP_CD));
			// サービス開始年月日
			map.put(EKK0081A010CBSMsg1List.SVC_STA_YMD, cAANMsg.getString(EKK0081A010CBSMsg1List.SVC_STA_YMD));
			// サービス課金開始年月日
			map.put(EKK0081A010CBSMsg1List.SVC_CHRG_STAYMD, cAANMsg.getString(EKK0081A010CBSMsg1List.SVC_CHRG_STAYMD));
		}
		// 処理結果
		return map;
	}
	// IT2-2018-0000042 ADD END
	
	/**
	 * 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,
							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];
	}
	

	/**
	 * エラー情報マッピング処理
	 * 
	 * <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;
	}

	/**
	 * 上りマッピング処理
	 * 
	 * @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 (mapVal[1] instanceof CAANMsg[])
			{
				template.set((String)mapVal[0], (CAANMsg[])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;
	}
	
	/**
	 * <dl>
	 * <dt>処理概要：
	 * <dd>サービスコンポーネント実行後に、IRequestParameterReadWriteに必要なデータをマッピングする<br>
	 * <dt>処理補足：
	 * <dd>
	 * </dl>
	 * 
	 * @param msgList SCの実行結果
	 * @param param リクエストパラメータ
	 * @param mapName 結果を入出力する際のキー
	 * @return IRequestParameterReadWrite
	 * @throws Throwable リクエストパラメータ入出力時の例外
	 */
	@SuppressWarnings({ "unchecked", "cast", "boxing" })
	public IRequestParameterReadWrite editResultRP(Map<?, ?> msgList,
			IRequestParameterReadWrite param,
			String mapName) throws Throwable
	{
		// SCからの戻り値からCAANMsgを取得する。
		CAANMsg[] templates = (CAANMsg[])msgList.get(JCMConstants.TEMPLATE_LIST_KEY);
		
		// リターンコード取得
		Object return_code = msgList.get(JCMConstants.RET_CD_INT_KEY);
		
		// 作業領域の取得
		Map workMap = (Map)param.getMappingWorkArea();
		if (workMap == null)
		{
			workMap = new HashMap();
			param.setMappingWorkArea(workMap);
		}
		
		// 結果を詰めるマップを取得。
		HashMap dataMap = null;
		
		// 業務データ設定
		dataMap = (HashMap)param.getData(mapName);
		if (dataMap == null)
		{
			dataMap = new HashMap();
			param.setData(mapName, dataMap);
		}
		
		// エラー情報の設定
		// IT2-2018-0000042 MOD START
		// 可読性の改善
//		IRequestParameterReadWrite wParam = editErrorInfo(param, templates, (Integer)return_code);
		editErrorInfo(param, templates, (Integer)return_code);
		// IT2-2018-0000042 MOD END
		
		// エラー情報のマップを取得
		// IT2-2018-0000042 MOD START
		// 可読性の改善
//		ArrayList<Object> errList = (ArrayList<Object>)wParam.getControlMapData(SCControlMapKeys.ERROR_INFO);
		ArrayList<Object> errList = (ArrayList<Object>)param.getControlMapData(SCControlMapKeys.ERROR_INFO);
		// IT2-2018-0000042 MOD END
		if (errList == null)
		{
			errList = new ArrayList<Object>();
		}
		
		// コントロールマップに設定
		// IT2-2018-0000042 MOD START
		// 可読性の改善
//		wParam.setControlMapData(SCControlMapKeys.ERROR_INFO, TemplateErrorUtil.getErrorInfo(msgList, errList));
		param.setControlMapData(SCControlMapKeys.ERROR_INFO, TemplateErrorUtil.getErrorInfo(msgList, errList));
		// IT2-2018-0000042 MOD END
		
		
		// IT2-2018-0000042 MOD START
		// 可読性の改善
//		return wParam;
		return param;
		// IT2-2018-0000042 MOD END
	}
	
	/**
	 * <dl>
	 * <dt>処理概要：
	 * <dd>呼び出したサービスIFの処理結果を判定する.<br>
	 * <dt>処理補足：
	 * <dd>
	 * </dl>
	 * 
	 * @param result SCの実行結果
	 * @throws Throwable SCの呼び出し例外
	 */
	@SuppressWarnings("boxing")
	private void checkExecutionResult(Map<?, ?> result) throws Throwable
	{
		// 取得したリターンコード、ステータスの内容を見て異常かどうかの判断をする。
		String rtnCode = result.get(JCMConstants.RET_CD_INT_KEY).toString();
		CAANMsg[] templates = (CAANMsg[])result.get(JCMConstants.TEMPLATE_LIST_KEY);
		Integer status = templates[0].getInt(JCMConstants.STATUS_INT_KEY);
		
		// 異常の場合、SCCallExceptionを生成してスローする
		if (!("0".equals(rtnCode) && 0 == status.intValue()))
		{
			SCCallException scCallEx = new SCCallException("戻り値不正", rtnCode, status);
			throw scCallEx;
		}
	}
	
	/**
	 * マップから値を取得
	 * 
	 * @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;
	}
	
	// IT2-2018-0000042 ADD START
	/**
	 * 相関ルールの実行
	 * 
	 * @param svcKeiInf サービス契約
	 * @param prcGrpCd 料金グループコード
	 * @param kapPlanCd 割賦プランコード
	 * @return 実行結果
	 */
	private HashMap<String, Object> runRULE0093001(HashMap<String, Object> svcKeiInf, String prcGrpCd, String kapPlanCd)
	{
		HashMap<String, Object> resultMap = new HashMap<String, Object>();

		HashMap<String, Object> paramMap = new HashMap<String, Object>();
		// 異動区分
		paramMap.put("IDO_DIV", JKKStrConst.CD00576_00001);
		// 料金グループコード
		paramMap.put("PRC_GRP_CD", prcGrpCd);
		// 住居形態
		paramMap.put("JK_FORM", JKKStrConst.CD00762_HOME);
		// 割賦プランコード
		paramMap.put("KAP_PLAN_CD", kapPlanCd);

		JRuleXMLHandler xmlHandler = new JRuleXMLHandler(JCCBPCommon.getApplicationConst(KEY_SOKAN_CHECK_XML_PATH), RULE0093001,
				JRuleXMLPattern.SINGLE_MODE_TYPE);

		ArrayList<ArrayList<HashMap<String, Object>>> checkResultList = xmlHandler.getCheckResultArray(paramMap);

		if (checkResultList != null && checkResultList.size() > 0)
		{
			if (checkResultList.get(0) != null && checkResultList.get(0).size() > 0)
			{
				resultMap = checkResultList.get(0).get(0);
			}
		}

		return resultMap;
	}

	/**
	 * ルール基準課金開始年月日の取得
	 * 
	 * @param svcKeiInf サービス契約
	 * @param prcGrpCd 料金グループコード
	 * @param kapPlanCd 割賦プランコード
	 * @return ルール基準課金開始年月日
	 * @throws CCException 
	 */
	private String getRuleBaseChrgStaymd(HashMap<String, Object> svcKeiInf, String prcGrpCd, String kapPlanCd) throws CCException
	{
		String ruleBaseChrgStaymd = "";

		if (svcKeiInf != null)
		{
			HashMap<String, Object> resultMap = runRULE0093001(svcKeiInf, prcGrpCd, kapPlanCd);

			// ルールの判定結果を取得
			String rsltCd = (String) resultMap.get("JDG_RSLT_CD");
			if (JDG_RSLT_CD_NORMAL.equals(rsltCd))
			{
				// 基準日
				String stdDt = "";
				// 判定結果基準日
				String jdgRsltStdDt = getMapValue(resultMap, "STD_DT");
				if (STD_DT_SVC_STAYMD.equals(jdgRsltStdDt))
				{
					// サービス開始年月日
					stdDt = getMapValue(svcKeiInf, EKK0081A010CBSMsg1List.SVC_STA_YMD);
				}
				else if (STD_DT_SVC_CHRG_STAYMD.equals(jdgRsltStdDt))
				{
					// サービス課金開始年月日
					stdDt = getMapValue(svcKeiInf, EKK0081A010CBSMsg1List.SVC_CHRG_STAYMD);
				}
				else
				{
					return "";
				}
				if (!JKKStringUtil.isNullBlank(stdDt))
				{
					// 判定結果当月/翌月
					String jdgRsltTogtuYkgtu = getMapValue(resultMap, "TOGTU_YKGTU");
					if (TOGTU_YKGTU_YKGTU.equals(jdgRsltTogtuYkgtu))
					{
						// 課金開始年月日に基準日の翌月1日を設定
						try
						{
							ruleBaseChrgStaymd = JPCDateUtil.addMonth(stdDt.substring(0, 6) + "01", 1);
						}
						catch (ParseException e)
						{
							throw new CCException("", e);
						}
					}
					else if (TOGTU_YKGTU_TOGTU.equals(jdgRsltTogtuYkgtu))
					{
						// 課金開始年月日に基準日を設定
						ruleBaseChrgStaymd = stdDt;
					}
					else
					{
						return "";
					}
				}
			}
		}

		return ruleBaseChrgStaymd;
	}

	/**
	 * 割賦契約課金開始年月日の設定
	 * 
	 * @param svcKeiInf サービス契約
	 * @param ccMsg CCの入力情報
	 * @throws CCException 
	 */
	private void setKapKeiChrgStaymd(HashMap<String, Object> svcKeiInf, HashMap<String, Object> ccMsg) throws CCException
	{
		if (svcKeiInf != null)
		{
			// 料金グループコード
			String prcGrpCd = getMapValue(svcKeiInf, "prc_grp_cd");
			// ANK-3636-00-00 MOD START
			// 一括の場合の課金開始年月日
//			String iktChrgStaymd = getRuleBaseChrgStaymd(svcKeiInf, prcGrpCd, JKKStrConst.KAP_PLAN_CD_BUNKATSU);
			String iktChrgStaymd = "";
			// ANK-3636-00-00 MOD END
			// 変更前割賦一括請求切替年月
			String bfKapIktSikySwchYm = getMapValue(ccMsg, "bf_kap_ikt_siky_swch_ym");
			// 割賦一括請求切替年月
			String kapIktSikySwchYm = getMapValue(ccMsg, "kap_ikt_siky_swch_ym");
			// 変更前割賦プランコード
			String bfKapPlanCd = getMapValue(ccMsg, "bf_kap_plan_cd");
			// 割賦プランコード
			String kapPlanCd = getMapValue(ccMsg, "kap_plan_cd");
			// 一括の場合の課金開始年月
			String iktChrgStaymdYm = "";
			
			// ANK-3636-00-00 ADD START
			if (JKKStrConst.KAP_PLAN_CD_IKKATSU.equals(kapPlanCd)
					|| JKKStrConst.KAP_PLAN_CD_BUNKATSU.equals(kapPlanCd)
				)
			{
				iktChrgStaymd = getRuleBaseChrgStaymd(svcKeiInf, prcGrpCd, JKKStrConst.KAP_PLAN_CD_BUNKATSU);
			}
			else if (JKKStrConst.KAP_PLAN_CD_IKKATSU_10_PERCENT.equals(kapPlanCd)
					|| JKKStrConst.KAP_PLAN_CD_BUNKATSU_10_PERCENT.equals(kapPlanCd)
				)
			{
				iktChrgStaymd = getRuleBaseChrgStaymd(svcKeiInf, prcGrpCd, JKKStrConst.KAP_PLAN_CD_BUNKATSU_10_PERCENT);
			}
			// ANK-3636-00-00 ADD END
// ANK-3951-00-00 ADD START
			else if (JKKStrConst.KAP_PLAN_CD_BUNKATSU_24_10PER.equals(kapPlanCd))
			{
				iktChrgStaymd = getRuleBaseChrgStaymd(svcKeiInf, prcGrpCd, JKKStrConst.KAP_PLAN_CD_BUNKATSU_24_10PER);
			}
// ANK-3951-00-00 ADD END
			
			if (iktChrgStaymd.length() == 8)
			{
				iktChrgStaymdYm = iktChrgStaymd.substring(0, 6);
			}

			if (!bfKapIktSikySwchYm.equals(kapIktSikySwchYm)
					&& !JKKStringUtil.isNullBlank(kapIktSikySwchYm)
					&& kapIktSikySwchYm.compareTo(iktChrgStaymdYm) < 0)
			{
				// サービス開始年月日
				String svcStaYmd = getMapValue(svcKeiInf, EKK0081A010CBSMsg1List.SVC_STA_YMD);
				// 割賦契約課金開始年月日にサービス開始年月日を設定
				ccMsg.put("kap_kei_chrg_staymd", svcStaYmd);
			}
			else if ((!bfKapIktSikySwchYm.equals(kapIktSikySwchYm)
					&& !JKKStringUtil.isNullBlank(bfKapIktSikySwchYm)
					&& bfKapIktSikySwchYm.compareTo(iktChrgStaymdYm) < 0)
					|| !bfKapPlanCd.equals(kapPlanCd))
			{
				String ruleBaseChrgStaymd = getRuleBaseChrgStaymd(svcKeiInf, prcGrpCd, kapPlanCd);
				ccMsg.put("kap_kei_chrg_staymd", ruleBaseChrgStaymd);
			}
		}
	}
	// IT2-2018-0000042 ADD END
}