/*********************************************************************
* All Rights reserved,Copyright (c) K-Opticom
**********************************************************************
*＜プログラム内容＞
*	システム名		：eo顧客基幹システム
*	モジュール名	：JKKCsCrsOpchAddUpdShokaiCC
*	ソースファイル名：JKKCsCrsOpchAddUpdShokaiCC.java
*	作成者			：FJ
*	日付			：2024/09/27
*＜機能概要＞
*	CSコース用オプションチャンネル登録・更新・照会の共通コンポーネント
*＜修正履歴＞
*	バージョン	修正日		修正者		修正内容
*   v74.00.00   2025/02/05  FJ）謝      【ANK-4592-00-00】テレビ新コース（スカパー用）導入対応
**********************************************************************/
package com.fujitsu.futurity.bp.custom.common;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import com.fujitsu.futurity.bp.custom.constant.JKKSvcConst;
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.ErrorInfoMapKeys;
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.constant.JPCModelConstant;
import eo.common.util.JKKCommonUtil;
import eo.common.util.JKKStringUtil;
import eo.common.util.JPCDateUtil;
import eo.common.util.JPCFomatString;
import eo.ejb.cbs.cbsmsg.EKK0011D020CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0011D020CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0021C060CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0081A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0081A010CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0321B002CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0321B002CBSMsg1List;

/**
 * CSコース用オプションチャンネル登録・更新・照会の共通コンポーネントクラスです。
 * <br>
 * @author 富士通
 *
 */
public class JKKCsCrsOpchAddUpdShokaiCC extends AbstractCommonComponent
{
	/** サービス契約一意照会 */
	private static final String TEMPLATE_ID_EKK0081A010 = "EKK0081A010";
	/** 課金先一覧照会（請求契約番号/サービス契約番号） */
	private static final String TEMPLATE_ID_EKK0321B002 = "EKK0321B002";
	/** 申込内容承認登録 */
	private static final String TEMPLATE_ID_EKK0011D020 = "EKK0011D020";
	/** 申込明細照査・後続業務依頼 */
	private static final String TEMPLATE_ID_EKK0021C060 = "EKK0021C060";
	
	/** 変更区分 登録 */
	private static final String CHG_DIV_ADD = "1";
	/** 変更区分 削除 */
	private static final String CHG_DIV_DEL = "2";
	/** 変更区分 取消 */
	private static final String CHG_DIV_CSL = "3";
	
	/** スカパー適用基準年月日区分 即時 */
	private static final String SPTV_APLY_STDARDYMD_DIV_SOKUJI = "1";
	/** スカパー適用基準年月日区分 次月 */
	private static final String SPTV_APLY_STDARDYMD_DIV_JIGETSU = "2";
	/** スカパー適用基準年月日区分 次々月 */
	private static final String SPTV_APLY_STDARDYMD_DIV_JIJIGETSU = "3";
	
	/** オプションチャンネルオペレーション方法コード(1：自動) */
	private static final String OP_CHANNEL_OPRTN_WAY_CD_1 = "1";
	
	/**
	 * CSコース用オプションチャンネル登録・更新・照会を行う。
	 * <br>
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param fixedText ユーザ定義文字列
	 * @return リクエストパラメータ
	 * @throws Throwable 例外が発生した場合
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite execute(SessionHandle handle, IRequestParameterReadWrite param, String fixedText)
	throws Throwable
	{
		String dataMapKey = fixedText;
		
		// CSコース用オプションチャンネル登録・更新・照会CCマップ
		HashMap<String, Object> ccMsg = (HashMap<String, Object>)param.getData(dataMapKey);
		if (ccMsg == null)
		{
			return param;
		}
		
		// 機能コードが未設定の場合、処理を終了する。
		if (JKKCommonUtil.isNull((String)ccMsg.get("func_cd")))
		{
			return param;
		}
		
		// SC呼び出し部品のインスタンス生成（引数にはログに出力するクラス名を渡す。空文字を設定した場合はログに出力されない）
		ServiceComponentRequestInvoker scCall = new ServiceComponentRequestInvoker();
		
		// オプション登録対象リスト
		ArrayList<HashMap<String, Object>> opAddTrgtList = new ArrayList<HashMap<String, Object>>();
		// オプション削除対象リスト
		ArrayList<HashMap<String, Object>> opDelTrgtList = new ArrayList<HashMap<String, Object>>();
		// オプション登録取消対象リスト
		ArrayList<HashMap<String, Object>> opAddCslTrgtList = new ArrayList<HashMap<String, Object>>();
		// オプション解約取消対象リスト
		ArrayList<HashMap<String, Object>> opDelCslTrgtList = new ArrayList<HashMap<String, Object>>();
		
		// オプション更新リスト
		ArrayList<HashMap<String, Object>> opUpdList = (ArrayList<HashMap<String, Object>>)ccMsg.get("op_upd_list");
		if (opUpdList != null && opUpdList.size() > 0)
		{
			for (int i = 0; i < opUpdList.size(); i++)
			{
				HashMap<String, Object> childMap = opUpdList.get(i);
				// 変更区分が登録の場合
				if (CHG_DIV_ADD.equals((String)childMap.get("chg_div")))
				{
					opAddTrgtList.add(childMap);
				}
				// 変更区分が削除の場合
				else if (CHG_DIV_DEL.equals((String)childMap.get("chg_div")))
				{
					opDelTrgtList.add(childMap);
				}
				// 変更区分が取消の場合
				else if (CHG_DIV_CSL.equals((String)childMap.get("chg_div")))
				{
					// オプションサービス契約ステータスがサービス提供中以前の場合
					if (!JKKStringUtil.isNullBlank((String)childMap.get("op_svc_kei_stat"))
							&& JKKStrConst.OP_SVC_STAT_SVCTK_CHU.compareTo((String)childMap.get("op_svc_kei_stat")) > 0)
					{
						opAddCslTrgtList.add(childMap);
					}
					else
					{
						opDelCslTrgtList.add(childMap);
					}
				}
			}
		}
		
		// オプションチャンネル登録処理を行う
		addOpch(handle, scCall, param, dataMapKey, ccMsg, opAddTrgtList);
		
		try
		{
			// スカパー鍵情報操作処理を行う
			callSptvKeyInfOperate(handle, scCall, param, dataMapKey, ccMsg);
		}
		catch(Throwable ex)
		{
			editErrInfo(param);
			throw ex;
		}
		
		// スカパー鍵情報操作処理エラーの場合
		if (!JKKStringUtil.isNullBlank((String)ccMsg.get("err_cd")))
		{
			return param;
		}
		
		// オプションチャンネル解約処理を行う
		dslOpch(handle, scCall, param, dataMapKey, ccMsg, opDelTrgtList);
		
		// オプションチャンネル登録取消処理を行う
		cslOpch(handle, scCall, param, dataMapKey, ccMsg, opAddCslTrgtList, true);
		
		// オプションチャンネル解約取消処理を行う
		cslOpch(handle, scCall, param, dataMapKey, ccMsg, opDelCslTrgtList, false);
		
		return param;
	}
	
	/**
	 * スカパー鍵情報操作処理を行う。
	 * @param handle セッションハンドル
	 * @param scCall SC呼出
	 * @param param パラメーター
	 * @param dataMapKey データマップキー
	 * @param ccMsg データ情報
	 * @throws Throwable
	 */
	@SuppressWarnings("unchecked")
	private void callSptvKeyInfOperate(SessionHandle handle,
									ServiceComponentRequestInvoker scCall,
									IRequestParameterReadWrite param,
									String dataMapKey,
									HashMap<String, Object> ccMsg) throws Throwable
	{
		// スカパー鍵情報操作CCマップ
		Map<String, Object> sptvKeyInfOperateMap = new HashMap<String, Object>();
		
		// 機能コード
		sptvKeyInfOperateMap.put("func_cd", (String)ccMsg.get("func_cd"));
		// 処理コード
		sptvKeyInfOperateMap.put("shori_cd", (String)ccMsg.get("shori_cd"));
		// サービス契約番号
		sptvKeyInfOperateMap.put("svc_kei_no", (String)ccMsg.get("svc_kei_no"));
		// 相手システムID
		sptvKeyInfOperateMap.put("aite_sysid", (String)ccMsg.get("aite_sysid"));
		// CASカード番号
		sptvKeyInfOperateMap.put("cascd_no", (String)ccMsg.get("cascd_no"));
		// オプション更新リスト
		sptvKeyInfOperateMap.put("op_upd_list", (List<Map<String, String>>)ccMsg.get("op_upd_list"));
		
		param.setData("JKKSptvKeyInfOperateCC", sptvKeyInfOperateMap);
		
		try
		{
			new JKKSptvKeyInfOperateCC().execute(handle, param, "JKKSptvKeyInfOperateCC");
		}
		finally
		{
			// エラーコード
			ccMsg.put("err_cd", sptvKeyInfOperateMap.get("err_cd"));
			// CATV顧客番号
			ccMsg.put("catv_cust_no", sptvKeyInfOperateMap.get("catv_cust_no"));
			// 料金コース
			ccMsg.put("pcrs", sptvKeyInfOperateMap.get("pcrs"));
			// 料金コース名称
			ccMsg.put("pcrs_nm", sptvKeyInfOperateMap.get("pcrs_nm"));
			// 料金プラン
			ccMsg.put("pplan", sptvKeyInfOperateMap.get("pplan"));
			// 料金プラン名称
			ccMsg.put("pplan_nm", sptvKeyInfOperateMap.get("pplan_nm"));
			// エラーオプションサービスコード
			ccMsg.put("err_op_svc_cd", sptvKeyInfOperateMap.get("err_op_svc_cd"));
			// エラーオプションサービスコード名称
			ccMsg.put("err_op_svc_cd_nm", sptvKeyInfOperateMap.get("err_op_svc_cd_nm"));
			// オプションリスト
			ccMsg.put("op_list", sptvKeyInfOperateMap.get("op_list"));
		}
		
		param.removeData("JKKSptvKeyInfOperateCC");
	}
	
	/**
	 * 登録処理を行う。
	 * @param handle セッションハンドル
	 * @param scCall SC呼出
	 * @param param パラメーター
	 * @param dataMapKey データマップキー
	 * @param ccMsg データ情報
	 * @param opTrgtList オプション対象リスト
	 * @throws Throwable
	 */
	@SuppressWarnings("unchecked")
	private void addOpch(SessionHandle handle,
									ServiceComponentRequestInvoker scCall,
									IRequestParameterReadWrite param,
									String dataMapKey,
									HashMap<String, Object> ccMsg,
									ArrayList<HashMap<String, Object>> opTrgtList) throws Throwable
	{
		if (opTrgtList == null || opTrgtList.size() == 0)
		{
			// 処理なし
			return;
		}
		
		// サービス契約一意照会
		CAANMsg ekk0081a010Msg = execEKK0081A010(handle, scCall, param, dataMapKey, (String)ccMsg.get("svc_kei_no"));
		if (ekk0081a010Msg == null)
		{
			// 処理なし
			return;
		}
		
		// 運用日付
		String opeDate = JPCBPCommon.getOpeDate(null);
		// 運用日の次月1日
		String jigetu = JPCDateUtil.addMonth(opeDate, 1);
		jigetu = jigetu.substring(0,6) + "01";// 運用日付から1ヶ月加算した日付の年月（6桁）＋ "01"を設定
		// 運用日の次々月1日
		String jijigetu = JPCDateUtil.addMonth(opeDate, 2);
		jijigetu = jijigetu.substring(0,6) + "01";// 運用日付から2ヶ月加算した日付の年月（6桁）＋ "01"を設定
		
		// オプション更新対象リスト
		ArrayList<HashMap<String, Object>> eoOpList = new ArrayList<HashMap<String, Object>>();
		// 進捗特記事項
		StringBuilder prgTkjk = new StringBuilder();
		
		// 進捗特記事項のチャンネル情報以外の文言を設定
		prgTkjk.append("オプションチャンネル情報登録　CAS番号：");
		prgTkjk.append((String)ccMsg.get("cascd_no"));
		
		for (int i = 0; i < opTrgtList.size(); i++)
		{
			HashMap<String, Object> childMap = new HashMap<String, Object>();
			
			// オプションサービスコード
			childMap.put("op_svc_cd", (String)opTrgtList.get(i).get("op_svc_cd"));
			// オプションチャンネルオペレーション方法コード
			childMap.put("op_channel_oprtn_way_cd", OP_CHANNEL_OPRTN_WAY_CD_1);
			
			// 進捗特記事項のチャンネル情報の文言を設定（登録されたチャンネル分）
			prgTkjk.append("　チャンネル名：");
			prgTkjk.append((String)opTrgtList.get(i).get("op_svc_cd_nm"));
			prgTkjk.append("　利用開始日：");
			if (SPTV_APLY_STDARDYMD_DIV_SOKUJI.equals((String)opTrgtList.get(i).get("sptv_aply_stdardymd_div")))
			{
				// サービス開始年月日
				childMap.put("svc_staymd", opeDate);
				// 利用開始日指定
				childMap.put("staymd", "0");
				
				prgTkjk.append("即時");
			}
			else if (SPTV_APLY_STDARDYMD_DIV_JIGETSU.equals((String)opTrgtList.get(i).get("sptv_aply_stdardymd_div")))
			{
				// サービス開始年月日
				childMap.put("svc_staymd", jigetu);
				// 利用開始日指定
				childMap.put("staymd", "1");
				
				prgTkjk.append("次月より");
			}
			else if (SPTV_APLY_STDARDYMD_DIV_JIJIGETSU.equals((String)opTrgtList.get(i).get("sptv_aply_stdardymd_div")))
			{
				// サービス開始年月日
				childMap.put("svc_staymd", jijigetu);
				// 利用開始日指定
				childMap.put("staymd", "2");
				
				prgTkjk.append("次々月より");
			}
			
			eoOpList.add(childMap);
		}
		
		// 課金先一覧照会（請求契約番号／サービス契約番号）
		CAANMsg ekk0321b002Msg = execEKK0321B002(handle, scCall, param, dataMapKey, (String)ccMsg.get("svc_kei_no"));
		
		// 課金先一覧照会（請求契約番号／サービス契約番号）マップ
		Map<String, Object> kksv0192scMap = new HashMap<String, Object>();
		ArrayList<HashMap<String, Object>> dataList = new ArrayList<HashMap<String,Object>>();
		if (ekk0321b002Msg != null)
		{
			HashMap<String, Object> dataMap = new HashMap<String, Object>();
			dataMap.put("seiky_kei_no", ekk0321b002Msg.getString(EKK0321B002CBSMsg1List.SEIKY_KEI_NO));
			dataList.add(dataMap);
		}
		kksv0192scMap.put("EKK0321B002CBSMsg1List", dataList);
		param.setData("KKSV0192SC", kksv0192scMap);
		
		// eo光テレビオプションチャンネル登録CCマップ
		Map<String, Object> createEohTvOpChMap = new HashMap<String, Object>();
		
		// 機能コード
		createEohTvOpChMap.put("func_code", (String)ccMsg.get("func_cd"));
		// SYSID
		createEohTvOpChMap.put("sysid", ekk0081a010Msg.getString(EKK0081A010CBSMsg1List.SYSID));
		// 異動区分
		createEohTvOpChMap.put("ido_div", (String)ccMsg.get("ido_div"));
		// 異動理由明細
		createEohTvOpChMap.put("EKK1091D010CBSMsg1List", (ArrayList<HashMap<String, Object>>)ccMsg.get("EKK1091D010CBSMsg1List"));
		// サービス契約番号
		createEohTvOpChMap.put("svc_kei_no", (String)ccMsg.get("svc_kei_no"));
		// 親契約識別コード(01：サービス契約)
		createEohTvOpChMap.put("oya_kei_skbt_cd", JKKStrConst.CD_DIV_OYAKEISKBTCD_SVCKEI);
		// 申込受付年月日時分秒
		createEohTvOpChMap.put("mskm_uk_dtm", JPCBPCommon.getOpeDateTimeStamp(null));
		// 申込年月日
		createEohTvOpChMap.put("mskm_ymd", opeDate);
		// 後続業務依頼年月日
		createEohTvOpChMap.put("kzkwrk_reqymd", opeDate);
		// オプション更新対象リスト
		createEohTvOpChMap.put("eo_op_list", eoOpList);
		// 進捗特記事項1,進捗特記事項2
		if(prgTkjk.length() < 129)
		{
			createEohTvOpChMap.put("prg_tkjk_1", prgTkjk.toString());
			createEohTvOpChMap.put("prg_tkjk_2", "");
		}
		else if(prgTkjk.length() < 257)
		{
			createEohTvOpChMap.put("prg_tkjk_1", prgTkjk.substring(0, 128));
			createEohTvOpChMap.put("prg_tkjk_2", prgTkjk.substring(128));
		}
		else
		{
			createEohTvOpChMap.put("prg_tkjk_1", prgTkjk.substring(0, 128));
			createEohTvOpChMap.put("prg_tkjk_2", prgTkjk.substring(128, 256));
		}
		// 更新年月日時分秒（更新前）
		createEohTvOpChMap.put("upd_dtm_bf", ekk0081a010Msg.getString(EKK0081A010CBSMsg1List.LAST_UPD_DTM));
		// ＫＥＹ＿料金コースコード
		createEohTvOpChMap.put("key_pcrs_cd", ekk0081a010Msg.getString(EKK0081A010CBSMsg1List.PCRS_CD));
		
		param.setData("KKSV0192CC", createEohTvOpChMap);
		
		new JKKCreateEohTvOpChCC().createEohTvOpCh(handle, param, "KKSV0192CC");
		
		// オプション更新リスト
		ArrayList<HashMap<String, Object>> opUpdList = (ArrayList<HashMap<String, Object>>)ccMsg.get("op_upd_list");
		// 登録情報リスト
		ArrayList<HashMap<String, Object>> opInfoList = (ArrayList<HashMap<String, Object>>)createEohTvOpChMap.get("op_info_list");
		
		if (opUpdList != null && opUpdList.size() > 0
				&& opInfoList != null && opInfoList.size() > 0)
		{
			for (int i = 0; i < opUpdList.size(); i++)
			{
				// 変更区分が登録以外の場合
				if (!CHG_DIV_ADD.equals((String)opUpdList.get(i).get("chg_div")))
				{
					continue;
				}
				
				for (int j = 0; j < opInfoList.size(); j++)
				{
					String opSvcCd = (String)opInfoList.get(j).get("op_svc_cd");
					if (!JKKStringUtil.isNullBlank(opSvcCd) && opSvcCd.equals((String)opUpdList.get(i).get("op_svc_cd")))
					{
						// オプションサービス契約番号
						opUpdList.get(i).put("op_svc_kei_no", (String)opInfoList.get(j).get("op_svc_kei_no"));
						break;
					}
				}
			}
		}
		
		param.removeData("KKSV0192CC");
		param.removeData("KKSV0192SC");
	}
	
	/**
	 * 解約処理を行う。
	 * @param handle セッションハンドル
	 * @param scCall SC呼出
	 * @param param パラメーター
	 * @param dataMapKey データマップキー
	 * @param ccMsg データ情報
	 * @param opTrgtList オプション対象リスト
	 * @throws Throwable
	 */
	@SuppressWarnings("unchecked")
	private void dslOpch(SessionHandle handle,
									ServiceComponentRequestInvoker scCall,
									IRequestParameterReadWrite param,
									String dataMapKey,
									HashMap<String, Object> ccMsg,
									ArrayList<HashMap<String, Object>> opTrgtList) throws Throwable
	{
		if (opTrgtList == null || opTrgtList.size() == 0)
		{
			// 処理なし
			return;
		}
		
		// サービス契約一意照会
		CAANMsg ekk0081a010Msg = execEKK0081A010(handle, scCall, param, dataMapKey, (String)ccMsg.get("svc_kei_no"));
		if (ekk0081a010Msg == null)
		{
			// 処理なし
			return;
		}
		
		// 機能コード
		String funcCd = (String)ccMsg.get("func_cd");
		// SYSID
		String sysid = ekk0081a010Msg.getString(EKK0081A010CBSMsg1List.SYSID);
		// 申込種別コード 解約申込
		String mskmSbtCd = JKKSvcConst.MSKM_SBT_CD_00005;
		
		// 申込内容承認登録
		CAANMsg eKK0011D020Msg = execEKK0011D020(handle, scCall, param, dataMapKey, funcCd, sysid, mskmSbtCd);
		
		// 機能コードが"1"の場合のみ実行
		if (JPCModelConstant.FUNC_CD_1.equals(funcCd))
		{
			CAANMsg[] eKK0011D020Msg1List = eKK0011D020Msg.getCAANMsgList(EKK0011D020CBSMsg.EKK0011D020CBSMSG1LIST);
			String mskmDtlNo =eKK0011D020Msg1List[0].getString(EKK0011D020CBSMsg1List.MSKM_DTL_NO);
			String lastUpdDtmMskm = eKK0011D020Msg.getString(EKK0011D020CBSMsg.UPD_DTM);
			
			// 申込内容照査・後続業務依頼
			execEKK0021C060(handle, scCall, param, dataMapKey, funcCd, mskmDtlNo, lastUpdDtmMskm);
			
			// 作業領域の取得
			Map workMap = (Map)param.getMappingWorkArea();
			if (workMap == null) {
				workMap = new HashMap();
				param.setMappingWorkArea(workMap);
			}
			
			// 結果を詰めるマップを取得。
			HashMap dataMap = null;
			
			/* 作業領域データ設定 */
			dataMap = (HashMap)workMap.get("WORK");
			if (dataMap == null) {
				dataMap = new HashMap();
				workMap.put("WORK", dataMap);
			}
			dataMap.put("kk0011_upd_dtm", lastUpdDtmMskm);
			dataMap.put("kk0011_mskm_dtl_no", mskmDtlNo);
		}
		
		// オプション更新対象リスト
		ArrayList<HashMap<String, Object>> eoOpList = new ArrayList<HashMap<String, Object>>();
		// 進捗特記事項
		StringBuilder prgTkjk = new StringBuilder();
		
		for (int i = 0; i < opTrgtList.size(); i++)
		{
			HashMap<String, Object> childMap = new HashMap<String, Object>();
			
			// 共通情報のマッピング
			childMap.put("func_code", (String)ccMsg.get("func_cd"));
			// サービス契約番号
			childMap.put("svc_kei_no", (String)ccMsg.get("svc_kei_no"));
			// オプションサービス契約番号
			childMap.put("op_svc_kei_no", (String)opTrgtList.get(i).get("op_svc_kei_no"));
			// 利用終了日
			childMap.put("svc_endymd", (String)opTrgtList.get(i).get("use_endymd"));
			// 課金終了日
			childMap.put("svc_chrg_endymd", (String)opTrgtList.get(i).get("svc_chrg_endymd"));
			// サービス解約理由コード
			childMap.put("svc_dlre_cd", JKKStrConst.CD00879_TUJYO_DSL);
			// 異動区分
			childMap.put("ido_div", (String)ccMsg.get("ido_div"));
			// 更新年月日時分秒（更新前）（サービス契約）
			childMap.put("upd_dtm_bf", ekk0081a010Msg.getString(EKK0081A010CBSMsg1List.LAST_UPD_DTM));
			// 進捗特記事項1,進捗特記事項2（解約するオプションチャンネル分　進捗は最後のレコードの情報で登録を行う）
			if(prgTkjk.length() == 0)
			{
				prgTkjk.append("オプションチャンネル情報解約");
			}
			prgTkjk.append("　チャンネル名：");
			prgTkjk.append((String)opTrgtList.get(i).get("op_svc_cd_nm"));
			prgTkjk.append("　利用終了日：");
			prgTkjk.append(JPCFomatString.formatDate((String)opTrgtList.get(i).get("use_endymd")));
			prgTkjk.append("　課金終了日：");
			prgTkjk.append(JPCFomatString.formatDate((String)opTrgtList.get(i).get("svc_chrg_endymd")));
			
			if(prgTkjk.length() < 129)
			{
				childMap.put("prg_tkjk_1", prgTkjk.toString());
				childMap.put("prg_tkjk_2", "");
			}
			else if(prgTkjk.length() < 257)
			{
				childMap.put("prg_tkjk_1", prgTkjk.substring(0, 128));
				childMap.put("prg_tkjk_2", prgTkjk.substring(128));
			}
			else
			{
				childMap.put("prg_tkjk_1", prgTkjk.substring(0, 128));
				childMap.put("prg_tkjk_2", prgTkjk.substring(128, 256));
			}
			
			// 異動理由明細
			childMap.put("EKK1091D010CBSMsg1List", (ArrayList<HashMap<String, Object>>)ccMsg.get("EKK1091D010CBSMsg1List"));
			
			eoOpList.add(childMap);
		}
		
		// eo光テレビオプションチャンネル解約CCマップ
		Map<String, Object> dslEohTvOpChMap = new HashMap<String, Object>();
		
		dslEohTvOpChMap.put("option_ch_select", eoOpList);
		
		param.setData("KKSV016503CC", dslEohTvOpChMap);
		
		new JKKEohTvOpChDslCC().execDissolution(handle, param, "KKSV016503CC");
		
		param.removeData("KKSV016503CC");
	}
	
	/**
	 * 取消処理を行う。
	 * @param handle セッションハンドル
	 * @param scCall SC呼出
	 * @param param パラメーター
	 * @param dataMapKey データマップキー
	 * @param ccMsg データ情報
	 * @param opTrgtList オプション対象リスト
	 * @param addCslFlg 登録取消フラグ
	 * @throws Throwable
	 */
	@SuppressWarnings("unchecked")
	private void cslOpch(SessionHandle handle,
									ServiceComponentRequestInvoker scCall,
									IRequestParameterReadWrite param,
									String dataMapKey,
									HashMap<String, Object> ccMsg,
									ArrayList<HashMap<String, Object>> opTrgtList,
									boolean addCslFlg) throws Throwable
	{
		if (opTrgtList == null || opTrgtList.size() == 0)
		{
			// 処理なし
			return;
		}
		
		// サービス契約一意照会
		CAANMsg ekk0081a010Msg = execEKK0081A010(handle, scCall, param, dataMapKey, (String)ccMsg.get("svc_kei_no"));
		if (ekk0081a010Msg == null)
		{
			// 処理なし
			return;
		}
		
		// 機能コード
		String funcCd = (String)ccMsg.get("func_cd");
		// SYSID
		String sysid = ekk0081a010Msg.getString(EKK0081A010CBSMsg1List.SYSID);
		// 申込種別コード キャンセル申込
		String mskmSbtCd = JKKSvcConst.MSKM_SBT_CD_00006;
		
		// 申込内容承認登録
		CAANMsg eKK0011D020Msg = execEKK0011D020(handle, scCall, param, dataMapKey, funcCd, sysid, mskmSbtCd);
		
		// 機能コードが"1"の場合のみ実行
		if (JPCModelConstant.FUNC_CD_1.equals(funcCd))
		{
			CAANMsg[] eKK0011D020Msg1List = eKK0011D020Msg.getCAANMsgList(EKK0011D020CBSMsg.EKK0011D020CBSMSG1LIST);
			String mskmDtlNo =eKK0011D020Msg1List[0].getString(EKK0011D020CBSMsg1List.MSKM_DTL_NO);
			String lastUpdDtmMskm = eKK0011D020Msg.getString(EKK0011D020CBSMsg.UPD_DTM);
			
			// 申込内容照査・後続業務依頼
			execEKK0021C060(handle, scCall, param, dataMapKey, funcCd, mskmDtlNo, lastUpdDtmMskm);
			
			// 作業領域の取得
			Map workMap = (Map)param.getMappingWorkArea();
			if (workMap == null) {
				workMap = new HashMap();
				param.setMappingWorkArea(workMap);
			}
			
			// 結果を詰めるマップを取得。
			HashMap dataMap = null;
			
			/* 作業領域データ設定 */
			dataMap = (HashMap)workMap.get("WORK");
			if (dataMap == null) {
				dataMap = new HashMap();
				workMap.put("WORK", dataMap);
			}
			dataMap.put("kk0011_upd_dtm", lastUpdDtmMskm);
			dataMap.put("kk0011_mskm_dtl_no", mskmDtlNo);
		}
		
		// 運用日付
		String opeDate = JPCBPCommon.getOpeDate(null);
		
		// オプション更新対象リスト
		ArrayList<HashMap<String, Object>> eoOpList = new ArrayList<HashMap<String, Object>>();
		// 進捗特記事項
		StringBuilder prgTkjk = new StringBuilder();
		
		for (int i = 0; i < opTrgtList.size(); i++)
		{
			HashMap<String, Object> childMap = new HashMap<String, Object>();
			
			// 共通情報のマッピング
			childMap.put("func_code", (String)ccMsg.get("func_cd"));
			// サービス契約番号
			childMap.put("svc_kei_no", (String)ccMsg.get("svc_kei_no"));
			// オプションサービス契約番号
			childMap.put("op_svc_kei_no", (String)opTrgtList.get(i).get("op_svc_kei_no"));
			// オプションサービス契約ステータス
			childMap.put("op_svc_kei_stat", (String)opTrgtList.get(i).get("op_svc_kei_stat"));
			// 利用終了日
			childMap.put("svc_endymd", opeDate);
			// サービス解約理由コード
			childMap.put("svc_dlre_cd", JKKStrConst.CD00879_TUJYO_DSL);
			// 異動区分
			childMap.put("ido_div", (String)ccMsg.get("ido_div"));
			// 更新年月日時分秒（更新前）（サービス契約）
			childMap.put("upd_dtm_bf", ekk0081a010Msg.getString(EKK0081A010CBSMsg1List.LAST_UPD_DTM));
			// 進捗特記事項1,進捗特記事項2
			if(prgTkjk.length() == 0)
			{
				// 登録取消の場合
				if (addCslFlg)
				{
					prgTkjk.append("オプションチャンネル情報登録予約取消");
				}
				else
				{
					prgTkjk.append("オプションチャンネル情報解約予約取消");
				}
			}
			prgTkjk.append("　チャンネル名：");
			prgTkjk.append((String)opTrgtList.get(i).get("op_svc_cd_nm"));
			
			if(prgTkjk.length() < 129)
			{
				childMap.put("prg_tkjk_1", prgTkjk.toString());
				childMap.put("prg_tkjk_2", "");
			}
			else if(prgTkjk.length() < 257)
			{
				childMap.put("prg_tkjk_1", prgTkjk.substring(0, 128));
				childMap.put("prg_tkjk_2", prgTkjk.substring(128));
			}
			else
			{
				childMap.put("prg_tkjk_1", prgTkjk.substring(0, 128));
				childMap.put("prg_tkjk_2", prgTkjk.substring(128, 256));
			}
			
			// 異動理由明細
			childMap.put("EKK1091D010CBSMsg1List", (ArrayList<HashMap<String, Object>>)ccMsg.get("EKK1091D010CBSMsg1List"));
			
			eoOpList.add(childMap);
		}
		
		// eo光テレビオプションチャンネル解約CCマップ
		Map<String, Object> dslEohTvOpChMap = new HashMap<String, Object>();
		
		dslEohTvOpChMap.put("option_ch_select", eoOpList);
		
		param.setData("KKSV016504CC", dslEohTvOpChMap);
		
		new JKKEohTvOpChClCC().execCancel(handle, param, "KKSV016504CC");
		
		param.removeData("KKSV016504CC");
	}
	
	/**
	 * @param param
	 * @param mappingData
	 * @return HashMap<String, Object>
	 * @throws RequestParameterException
	 */
	private HashMap<String, Object> editInMsg(IRequestParameterReadWrite param,
												Object[][] mappingData) 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));
		
		String svcIf = (String)mappingData[0][1];
		
		CAANMsg template = new CAANMsg(String.format("eo.ejb.cbs.cbsmsg.%sCBSMsg", svcIf));

		// オペレータID
		template.set(JCMConstants.OPERATOR_ID_KEY, param.getControlMapData(SCControlMapKeys.OPERATOR_ID));
		// 運用日付
		template.set(JCMConstants.OPERATE_DATE_KEY, param.getControlMapData(SCControlMapKeys.OPE_DATE));
		// 運用日時
		template.set(JCMConstants.OPERATE_DATETIME_KEY, param.getControlMapData(SCControlMapKeys.OPE_TIME));
		
		for (int i = 0; i < mappingData.length; i++)
		{
			/* CAANMsg[]の場合 */
			if (mappingData[i][1] instanceof CAANMsg[])
			{
				 template.set((String)mappingData[i][0], (CAANMsg[])mappingData[i][1]);
			}
			/* CAANMsg[]の以外 */
			else
			{
				/* nullの場合*/
				if ("".equals(mappingData[i][1]))
				{
					template.setNull((String)mappingData[i][0]);
				}
				/* 他の場合*/
				else
				{
					template.set((String)mappingData[i][0], mappingData[i][1]);
				}
			}
		}
		
		CAANMsg[] templates = new CAANMsg[1];
		templates[0] = template;
		paramMap.put(JCMConstants.TEMPLATE_LIST_KEY, templates);

		return paramMap;
	}
	
	/**
	 * 
	 * @param param
	 * @param templates
	 * @param returnCode
	 * @param dataMapKey
	 * @param mappingData
	 * @return IRequestParameterReadWrite
	 * @throws RequestParameterException
	 */
	@SuppressWarnings("unchecked")
	private IRequestParameterReadWrite editErrorInfoCom(IRequestParameterReadWrite param,
														CAANMsg[] templates, 
														int returnCode,
														String dataMapKey,
														Object[][] mappingData) throws RequestParameterException
	{
		// 本来はサービスインターフェイス分の処理が必要
		CAANMsg template = templates[0];
		int templateStatus = template.getInt(JCMConstants.STATUS_INT_KEY);
		
		/* 0以外のとき */
		if (returnCode != 0)
		{
			templateStatus = 9000;
		}
		
		/* nullのとき */
		if (JCMAPLConstMgr.getString("RETURN_MESSAGE_" + String.format("%1$04d", templateStatus)) == null)
		{
			templateStatus = 0;
		}

		int bpStatus = 0;
		Object obj = param.getControlMapData(SCControlMapKeys.RETURN_CODE);
		/* nullの場合 */
		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, String> inMap = null;

		// ユーザデータ情報
		inMap = (HashMap<String, String>)param.getData(dataMapKey);
		
		for (int i = 0; i < mappingData.length; i++)
		{
			/* nullチェック　*/
			if (!template.isNull(mappingData[i][0] + "_err"))
			{
				/* ユーザデータ情報にエラーを確認する　*/
				if (!inMap.containsKey(mappingData[i][0] + "_err"))
				{
					inMap.put(mappingData[i][0] + "_err", template.getString(mappingData[i][0] + "_err"));
				}
			}
		}
		return param;
	}
	
	/**
	 * SC(サービスインターフェイス）を呼び出す。
	 * 
	 * @param handle
	 * @param scCall
	 * @param param
	 * @param dataMapKey
	 * @param mappingData
	 * @return CAANMsg
	 * @throws Exception
	 */
	@SuppressWarnings("unchecked")
	private CAANMsg callSC(SessionHandle handle, 
							ServiceComponentRequestInvoker scCall, 
							IRequestParameterReadWrite param, 
							String dataMapKey,
							Object[][] mappingData) throws Exception
	{
		HashMap<String, Object> paramMap = editInMsg(param, mappingData);
		
		Map<?, ?> result = scCall.run(paramMap, handle);
		
		CAANMsg[] templates = (CAANMsg[])result.get(JCMConstants.TEMPLATE_LIST_KEY);
		CAANMsg msg = templates[0];
		
		// リターンコード取得
		Object return_code = result.get(JCMConstants.RET_CD_INT_KEY);
		int status = msg.getInt(JCMConstants.STATUS_INT_KEY);
		
		editErrorInfoCom(param, templates, (Integer)return_code, dataMapKey, mappingData);
		
		//エラー情報のマップを取得
		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));
		
		/* 異常の場合、SCCallExceptionを生成してスローする */
		if (!("0".equals(return_code.toString()) && 0 == status))
		{
			throw new SCCallException("戻り値不正", return_code.toString(), status);
		}
		return msg;
	}
	
	/**
	 * 申込内容承認登録を行う。
	 * @param handle セッションハンドル
	 * @param scCall SC呼び出し部品
	 * @param param リクエストパラメータ
	 * @param dataMapKey データマップキー
	 * @param funcCd 機能コード
	 * @param sysid SYSID
	 * @param mskmSbtCd 申込種別コード
	 * @return CAANMsg
	 * @throws Throwable
	 */
	private CAANMsg execEKK0011D020(SessionHandle handle,
									ServiceComponentRequestInvoker scCall,
									IRequestParameterReadWrite param,
									String dataMapKey,
									String funcCd,
									String sysid,
									String mskmSbtCd) throws Throwable
	{
		
		Object[][] ekk0011d020In =
		{
				{EKK0011D020CBSMsg.TEMPLATEID, TEMPLATE_ID_EKK0011D020},
				{EKK0011D020CBSMsg.FUNC_CODE, funcCd},
				{EKK0011D020CBSMsg.SYSID, sysid},
				{EKK0011D020CBSMsg.MSKM_SBT_CD, mskmSbtCd},
				{EKK0011D020CBSMsg.MSKM_UK_DTM, JPCBPCommon.getOpeDateTimeStamp(null)},
				{EKK0011D020CBSMsg.MSKM_YMD, JCCBPCommon.getOpeDate(null)},
				{EKK0011D020CBSMsg.CONSMBSN_MSKM_STAT_SKBT_CD, JKKStrConst.CD00760_04},
				{EKK0011D020CBSMsg.EKK0011D020CBSMSG1LIST, new CAANMsg[]{new CAANMsg(EKK0011D020CBSMsg1List.class.getName())}},
		};
		
		return callSC(handle, scCall, param, dataMapKey, ekk0011d020In);
	}
	
	/**
	 * 申込明細照査・後続業務依頼を行う。
	 * @param handle セッションハンドル
	 * @param scCall SC呼び出し部品
	 * @param param リクエストパラメータ
	 * @param dataMapKey データマップキー
	 * @param funcCd 機能コード
	 * @param mskmDtlNo 申込明細番号
	 * @param updDtmBf 更新年月日時分秒(更新前)
	 * @throws Throwable
	 */
	private void execEKK0021C060(SessionHandle handle,
									ServiceComponentRequestInvoker scCall,
									IRequestParameterReadWrite param,
									String dataMapKey,
									String funcCd,
									String mskmDtlNo,
									String updDtmBf) throws Throwable
	{
		
		Object[][] ekk0021c060In =
		{
				{EKK0021C060CBSMsg.TEMPLATEID, TEMPLATE_ID_EKK0021C060},
				{EKK0021C060CBSMsg.FUNC_CODE, funcCd},
				{EKK0021C060CBSMsg.MSKM_DTL_NO, mskmDtlNo},
				{EKK0021C060CBSMsg.KZKWRK_REQYMD, JCCBPCommon.getOpeDate(null)},
				{EKK0021C060CBSMsg.UPD_DTM_BF, updDtmBf},
		};
		
		callSC(handle, scCall, param, dataMapKey, ekk0021c060In);
	}
	
	/**
	 * 課金先一覧照会（請求契約番号/サービス契約番号）を行う。
	 * @param handle セッションハンドル
	 * @param scCall SC呼出
	 * @param param パラメーター
	 * @param dataMapKey データマップキー
	 * @param svcKeiNo サービス契約番号
	 * @return CAANMsg
	 * @throws Throwable
	 */
	private CAANMsg execEKK0321B002(SessionHandle handle,
									ServiceComponentRequestInvoker scCall,
									IRequestParameterReadWrite param,
									String dataMapKey,
									String svcKeiNo) throws Throwable
	{
		CAANMsg result = null;
		
		Object[][] ekk0321b002In =
		{
				{EKK0321B002CBSMsg.TEMPLATEID, TEMPLATE_ID_EKK0321B002},
				{EKK0321B002CBSMsg.FUNC_CODE, JPCModelConstant.FUNC_CD_1},
				{EKK0321B002CBSMsg.KEY_SVC_KEI_NO, svcKeiNo},
		};
		
		CAANMsg ekk0321b002Msg = callSC(handle, scCall, param, dataMapKey, ekk0321b002In);
		
		CAANMsg[] ekk0321b002MsgList = ekk0321b002Msg.getCAANMsgList(EKK0321B002CBSMsg.EKK0321B002CBSMSG1LIST);
		
		if (ekk0321b002MsgList != null && ekk0321b002MsgList.length > 0)
		{
			result = ekk0321b002MsgList[0];
		}
		
		return result;
	}
	
	/**
	 * サービス契約一意照会を行う。
	 * @param handle セッションハンドル
	 * @param scCall SC呼出
	 * @param param パラメーター
	 * @param dataMapKey データマップキー
	 * @param svcKeiNo サービス契約番号
	 * @return CAANMsg
	 * @throws Throwable
	 */
	private CAANMsg execEKK0081A010(SessionHandle handle,
									ServiceComponentRequestInvoker scCall,
									IRequestParameterReadWrite param,
									String dataMapKey,
									String svcKeiNo) throws Throwable
	{
		CAANMsg result = null;
		
		Object[][] ekk0081a010In =
		{
				{EKK0081A010CBSMsg.TEMPLATEID, TEMPLATE_ID_EKK0081A010},
				{EKK0081A010CBSMsg.FUNC_CODE, JPCModelConstant.FUNC_CD_2},
				{EKK0081A010CBSMsg.KEY_SVC_KEI_NO, svcKeiNo},
				{EKK0081A010CBSMsg.KEY_GENE_ADD_DTM, ""},
				{EKK0081A010CBSMsg.KEY_RSV_APLY_YMD, JCCBPCommon.getOpeDate(null)},
		};
		
		CAANMsg ekk0081a010Msg = callSC(handle, scCall, param, dataMapKey, ekk0081a010In);
		
		CAANMsg[] ekk0081a010MsgList = ekk0081a010Msg.getCAANMsgList(EKK0081A010CBSMsg.EKK0081A010CBSMSG1LIST);
		
		if (ekk0081a010MsgList != null && ekk0081a010MsgList.length > 0)
		{
			result = ekk0081a010MsgList[0];
		}
		
		return result;
	}
	
	/**
	 * コントロールマップエラー情報格納処理を行います。
	 * @param param リクエストパラメータ
	 * @param itemplateStatus サービスインターフェイスステータス
	 * @param returnCode リターンコード
	 * @return リクエストパラメータ
	 * @throws RequestParameterException
	 */
	private IRequestParameterReadWrite setControlMapErrInfo(IRequestParameterReadWrite param, int itemplateStatus, int returnCode)
	throws RequestParameterException
	{
		int templateStatus = itemplateStatus;
		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);
		}
		
		return param;
	}
	
	/**
	 * エラー情報を設定します。
	 * @param iparam IRequestParameterReadWrite
	 * @return IRequestParameterReadWrite
	 * @throws RequestParameterException
	 */
	private IRequestParameterReadWrite editErrInfo(IRequestParameterReadWrite iparam) throws RequestParameterException
	{
		IRequestParameterReadWrite param = setControlMapErrInfo(iparam, JPCModelConstant.RELATION_ERR, JPCModelConstant.NORMAL_END);
		
		// エラー情報のリストを取得
		ArrayList<Object> errList = (ArrayList<Object>)param.getControlMapData(SCControlMapKeys.ERROR_INFO);
		if (errList == null)
		{
			errList = new ArrayList<Object>();
		}
		
		// コントロールマップに設定
		HashMap<String, Object> errorMap = new HashMap<String, Object>();
		HashMap<String, String> errorMapChild = new HashMap<String, String>();
		
		errorMap.put(ErrorInfoMapKeys.RETURN_CODE, JPCModelConstant.NORMAL_END);
		errorMap.put(ErrorInfoMapKeys.STATUS, JPCModelConstant.RELATION_ERR);
		
		errorMap.put(ErrorInfoMapKeys.ITEM_CHECK_ERRORS, errorMapChild);
		
		errList.add(errorMap);
		
		param.setControlMapData(SCControlMapKeys.ERROR_INFO, errList);
		
		return param;
	}
}