/*********************************************************************
* All Rights reserved,Copyright (c) K-Opticom
**********************************************************************
*＜プログラム内容＞
*   システム名      ：eo顧客基幹システム
*   モジュール名    ：JKKCrdtKssaiSnnReqCC
*   ソースファイル名：JKKCrdtKssaiSnnReqCC.java
*   作成者          ：富士通
*   日付            ：2022年04月05日
*＜機能概要＞
*   クレジット決済承認依頼CC部品
*＜修正履歴＞
*   バージョン  修正日		修正者		修正内容
*   v58.00.00	2022/04/05	FJ) 藤本涼	【ANK-3846-00-00】【クレカ】クリアパス→ペイジェント移行
*
**********************************************************************/
package com.fujitsu.futurity.bp.custom.common;


import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.concurrent.TimeUnit;

import com.fujitsu.futurity.bp.custom.constant.JCNCreditAuthConstCC;
import com.fujitsu.futurity.bp.custom.constant.JCNYoshinJdgConstCC;
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.model.common.check.HalfCharCheck;
import com.fujitsu.futurity.model.common.check.LengthCheck;

import eo.common.constant.JKKStrConst;
import eo.common.util.JKKStringUtil;
import eo.ejb.cbs.cbsmsg.EKKA0010004CBSMsg;

/**
 * クレジット決済承認依頼CC部品です。
 * 
 * <br>
 * @author 富士通
 */
public class JKKCrdtKssaiSnnReqCC extends AbstractCommonComponent
{

	/** リクエストキー：システムID */
	private static final String IN_SYSTEM_ID = "system_id";
	/** リクエストキー：サイトID */
	private static final String IN_SITE_ID = "site_id";
	/** リクエストキー：請求契約番号 */
	private static final String IN_SEIKY_KEI_NO = "seiky_kei_no";
	/** リクエストキー：SYSID */
	private static final String IN_SYSID = "sysid";
	/** リクエストキー：連携支払方法ID */
	private static final String IN_REQ_PAYWAYID = "req_payWayId";
	/** リクエストキー：決済金額 */
	private static final String IN_CR_AMOUNT = "cr_amount";
	/** リクエストキー：顧客ID */
	private static final String IN_CUSTOMER_ID = "customer_id";
	
	/** レスポンスキー：処理結果コード */
	private static final String OUT_CR_RESULTCODE = "cr_resultcode";
	/** レスポンスキー：レスポンスコード */
	private static final String OUT_RESPONSE_CODE = "response_code";
	/** レスポンスキー：レスポンス詳細 */
	private static final String OUT_RESPONSE_DETAIL = "response_detail";
	/** レスポンスキー：決済ID */
	private static final String OUT_PAYMENT_ID = "payment_id";
	/** レスポンスキー：マーチャント取引ID */
	private static final String OUT_TRADING_ID = "trading_id";
	/** レスポンスキー：イシュア区分 */
	private static final String OUT_ISSUER_CLASS = "issuer_class";
	/** レスポンスキー：仕向先カード会社コード */
	private static final String OUT_CR_BRANDCODE = "cr_brandcode";
	/** レスポンスキー：ハッシュタグ */
	private static final String OUT_HASHTAG = "hashtag";
	/** レスポンスキー：カード番号 */
	private static final String OUT_CR_CARDNUMBER = "cr_cardnumber";
	/** レスポンスキー：カード有効期限 */
	private static final String OUT_CARD_VALID_TERM = "card_valid_term";
	/** レスポンスキー：オーソリ承認番号 */
	private static final String OUT_CR_AUTHNUMBER = "cr_authnumber";
	/** レスポンスキー：カード種別 */
	private static final String OUT_CARD_TYPE = "card_type";
	/** レスポンスキー：カードブランド */
	private static final String OUT_CARD_BRAND_OPTCD = "card_brand_optcd";
	/** レスポンスキー：支払方法ID */
	private static final String OUT_PAY_WAY_ID = "pay_way_id";

	/** CCパラメータ：エラー情報 */
	private static final String ERROR_INFO = "ERROR_INFO";
	/** CCパラメータ：エラーコード */
	private static final String ERROR_CODE = "errCode";
	/** CCパラメータ：エラーメッセージ */
	private static final String ERROR_MESSAGE = "errMessage";

	/** APIエラーコード_E1001：必須チェックエラー */
	private static final String RES_KEY_ERROR_CODE_E1001 = "E1001";
	/** APIエラーコード_E2001：形式チェックエラー */
	private static final String RES_KEY_ERROR_CODE_E2001 = "E2001";
	/** APIエラーコード_E3001：桁数チェックエラー */
	private static final String RES_KEY_ERROR_CODE_E3001 = "E3001";
	/** APIエラーコード_E5001：サイトIDチェックエラー */
	private static final String RES_KEY_ERROR_CODE_E5002 = "E5001";
	
	/** リターンメッセージの書式指定文字列 */
	private static final String RETURN_MESSAGE_FORMAT = "%1$04d";
	/** エラー項目接尾辞 */
	private static final String ERRITEM_SUFFIX = "_err";
	/** リターンメッセージ取得キー */
	private static final String RETURN_MESSAGE_STRING = "RETURN_MESSAGE_";
	
	/** サイトID組み合わせ一覧 */
	private static final String SITE_ID_PAIR = "SITE_ID_";

	/** APIリターンコード_0000 正常終了 */
	private static final String RETURN_CD_0000 = "0000";
	/** APIリターンコード_5000_業務エラー */
	private static final String RETURN_CD_5000 = "5000";
	
	/** 取引年月日 */
	private static final String CR_ORDERDATE = "cr_orderdate";
	/** 取引時間 */
	private static final String CR_ORDERTIME = "cr_ordertime";
	
	/** 支払方法：クレジット */
	private static final String PAYWAY_CREDIT = "1";
	
	/** オーソリ実行フラグ：決済オーソリ */
	private static final String AUTHORI_RUN_FLG = "2";
	
	/** サイトID：コンシューマ */
	private static final String SITE_ID_CONSUMER = "4211";
	/** サイトID：コンテンツ */
	private static final String SITE_ID_CONTENT = "4212";
	/** サイトID：法人 */
	private static final String SITE_ID_HOJIN = "4411";
	
	/** サービスコンポーネント名：決済オーソリ依頼 */
	private static final String SC_NAME_EKKA0010004 = "EKKA0010004";	
	/** サービスコンポーネント名：与信判定 */
	private static final String CC_NAME_YOSHIN_JDG = "JCNYoshinJdgCC";	
	/** サービスコンポーネント名：クレジットオーソリ承認 */
	private static final String CC_NAME_CREDIT_AUTH = "JCNCreditAuthCC";	
	
	/** クレジットオーソリ承認実行フラグ：実行する */
	private static final String CREDIT_AUTH_FLG_ON = "1";	
	
	/** 実行処理判定フラグ：SC */
	private static final String JDG_FLG_0 = "0";	
	/** 実行処理判定フラグ：CC */
	private static final String JDG_FLG_1 = "1";	
	
	/** マーチャントＩＤ */
	private static final String PAYGENT_MERCHANT_ID = "PAYGENT_MERCHANT_ID";	
	/** 接続ID */
	private static final String PAYGENT_CONNECT_ID = "PAYGENT_CONNECT_ID";	
	/** 接続パスワード */
	private static final String PAYGENT_CONNECT_PASSWORD = "PAYGENT_CONNECT_PASSWORD";	
	/** 電文バージョン番号 */
	private static final String PAYGENT_TELEGRAM_VERSION = "PAYGENT_TELEGRAM_VERSION";	

	/**
	 * クレジット決済承認依頼処理です。
	 * 
	 * <br>
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param fixedText サービスメッセージ
	 * @return リクエストパラメータ管理クラス
	 * @throws Throwable 
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite getCrdtKssaiSnnReq(SessionHandle handle, IRequestParameterReadWrite param, String fixedText) throws Throwable
	{
		
		// ユーザデータ取得
		HashMap userData = (HashMap)param.getData(fixedText);
		
		try
		{
			
			// 取引年月日:当該オーソリを依頼した時点のeo顧客基幹システムのシステム日時の年月日(yyyyMMdd)を設定
			String orderdate = com.fujitsu.futurity.bp.custom.common.JFUBPCommon.getSysDateTime().substring(0, 8);
			// 取引時間:当該オーソリを依頼した時点のeo顧客基幹システムにおけるシステム日時の時分秒(HHmmss)
			String ordertime = com.fujitsu.futurity.bp.custom.common.JFUBPCommon.getSysDateTime().substring(8, 14);
	
			// 単項目・関連チェック
			if (!checkUnitParam(userData))
			{
				param.setControlMapData(SCControlMapKeys.RETURN_CODE, RETURN_CD_5000);
				param.setControlMapData(SCControlMapKeys.RETURN_MESSAGE, null);
				return param;
			}
			
			// 処理呼出
			String siteId = (String)userData.get(IN_SITE_ID);
			
			if (SITE_ID_CONSUMER.equals(siteId) || SITE_ID_HOJIN.equals(siteId))
			{
				// 決済オーソリ依頼を実行
				CAANMsg ekka0010004Rslt = callEKKA0010004(handle, param, fixedText, userData);
				
				// 呼出処理結果の取得
				setResultData(ekka0010004Rslt, null,  userData, JDG_FLG_0, orderdate, ordertime);
				
			}
			else if (SITE_ID_CONTENT.equals(siteId))
			{
				// マーチャント取引ID
				String tradingId = createTradingId(userData, orderdate, ordertime);
				
				// 呼出処理結果取得用マップ
				HashMap rsltMap = new HashMap();
				// 与信判定サービスへのパラメータマップ作成
				createYoshinJdgMap(param, userData, tradingId);
				// クレジットオーソリ承認サービスへのパラメータマップ作成
				createCreditAuthMap(param, userData, tradingId);
				
				// 与信判定CC実行
				JCNYoshinJdgCC yoshinJdg = new JCNYoshinJdgCC();
				yoshinJdg.executeYoshinJdg(handle, param, CC_NAME_YOSHIN_JDG);
				
				// 処理結果を格納
				rsltMap = (HashMap)param.getData(CC_NAME_YOSHIN_JDG);
				// クレジットオーソリ承認実行フラグの取得
				String creditAuthFlg = (String) rsltMap.get(JCNYoshinJdgConstCC.OUTPUT_CREDIT_AUTH_FLG);
				
				// クレジットオーソリ承認実行フラグ判定
				if (CREDIT_AUTH_FLG_ON.equals(creditAuthFlg))
				{
					// 実行フラグが「1」の場合、下記より実行
					// クレジットオーソリ承認CC実行
					JCNCreditAuthCC creditAuth = new JCNCreditAuthCC();
					creditAuth.executeAuthMain(handle, param, CC_NAME_CREDIT_AUTH);
					
					// 処理結果を格納
					rsltMap = (HashMap)param.getData(CC_NAME_CREDIT_AUTH);
					
				}
				
				//エラーコードを取得
				String errCode = (String)rsltMap.get(JCNYoshinJdgConstCC.OUTPUT_ERR_CODE);
				
				//エラーコードが存在する場合、エラーリストを返却
				if (!(JKKStringUtil.isNullBlank(errCode)))
				{
					// エラーリスト
					List<Map<String, String>> errList = new ArrayList<Map<String, String>>();
					Map<String, String> returnMap = new HashMap<String, String>();
					
					returnMap.put(ERROR_CODE, errCode);
					returnMap.put(ERROR_MESSAGE, (String)rsltMap.get(JCNYoshinJdgConstCC.OUTPUT_ERR_MESSAGE));
					
					errList.add(returnMap);
					userData.put(ERROR_INFO, errList);
					
					param.setControlMapData(SCControlMapKeys.RETURN_CODE, RETURN_CD_5000);
					param.setControlMapData(SCControlMapKeys.RETURN_MESSAGE, null);
					return param;
				}
				
				// 呼出処理結果の取得
				setResultData(null, rsltMap, userData, JDG_FLG_1, orderdate, ordertime);
				
			}
	
			// 正常終了
			param.setControlMapData(SCControlMapKeys.RETURN_CODE, RETURN_CD_0000);
			param.setControlMapData(SCControlMapKeys.RETURN_MESSAGE, null);
	
			return param;
		}
		catch (Throwable e)
		{
			// 更新に失敗した場合は
			param.setControlMapData(SCControlMapKeys.RETURN_CODE, JKKStrConst.RETURN_CD_9000);
			throw new CCException("", e);
		}
		
	}

	/**
	 * 単項目・関連チェック
	 * 
	 * @param userData
	 * @return チェックの判定
	 * @throws Exception
	 */
	private boolean checkUnitParam(Map<String, Object> userData) throws Exception
	{
		// エラーリスト
		List<Map<String, String>> errList = new ArrayList<Map<String, String>>();
		
		String itemName = null;
		String itemValue = null;
		String systemId = null;
		String systemIdValue = null;
		String siteId = (String)userData.get(IN_SITE_ID);
		
		// システムID
		itemName = IN_SYSTEM_ID;
		itemValue = (String)userData.get(itemName);
		// 必須チェック
		if ( itemValue == null || "".equals(itemValue) )
		{
			// 必須チェックエラー
			errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E1001));
		}
		else if (!HalfCharCheck.isEnNumber1Check(itemValue))
		{
			// 形式チェックエラー
			errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E2001));
		}
		else if (!LengthCheck.isLength1Check(itemValue, 4))
		{
			// 桁数チェックエラー
			errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E3001));
		}
		
		// サイトID
		itemName = IN_SITE_ID;
		itemValue = (String)userData.get(itemName);
		// 必須チェック
		if ( itemValue == null || "".equals(itemValue) )
		{
			// 必須チェックエラー
			errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E1001));
		}
		else if (!HalfCharCheck.isEnNumber1Check(itemValue))
		{
			// 形式チェックエラー
			errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E2001));
		}
		else if (!LengthCheck.isLength1Check(itemValue, 4))
		{
			// 桁数チェックエラー
			errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E3001));
		}
		
		// 請求契約番号
		// サイトID = 4411(法人)の場合、必須チェック実行
		itemName = IN_SEIKY_KEI_NO;
		itemValue = (String)userData.get(itemName);
		if (SITE_ID_HOJIN.equals(siteId))
		{
			// 必須チェック
			if ( itemValue == null || "".equals(itemValue) )
			{
				// 必須チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E1001));
			}
		}
		if (!(itemValue == null || "".equals(itemValue)))
		{
			if (!HalfCharCheck.isEnNumber1Check(itemValue))
			{
				// 形式チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E2001));
			}
			else if (!LengthCheck.isLength1Check(itemValue, 10))
			{
				// 桁数チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E3001));
			}
		}
		
		// SYSID
		// サイトID = 4212(コンテンツ)の場合、実行
		if (SITE_ID_CONTENT.equals(siteId))
		{
			itemName = IN_SYSID;
			itemValue = (String)userData.get(itemName);
			// 必須チェック
			if ( itemValue == null || "".equals(itemValue) )
			{
				// 必須チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E1001));
			}
			else if (!HalfCharCheck.isEnNumber1Check(itemValue))
			{
				// 形式チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E2001));
			}
			else if (!LengthCheck.isLength1Check(itemValue, 10))
			{
				// 桁数チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E3001));
			}
		}
		
		// 連携支払方法ID
		// サイトID = 4212(コンテンツ)の場合、実行
		if (SITE_ID_CONTENT.equals(siteId))
		{
			itemName = IN_REQ_PAYWAYID;
			itemValue = (String)userData.get(itemName);
			// 必須チェック
			if ( itemValue == null || "".equals(itemValue) )
			{
				// 必須チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E1001));
			}
			else if (!HalfCharCheck.isEnNumber1Check(itemValue))
			{
				// 形式チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E2001));
			}
			else if (!LengthCheck.isLength2Check(itemValue, 9, 10))
			{
				// 桁数チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E3001));
			}
		}
		
		// 決済金額
		itemName = IN_CR_AMOUNT;
		itemValue = (String)userData.get(itemName);
		// 必須チェック
		if ( itemValue == null || "".equals(itemValue) )
		{
			// 必須チェックエラー
			errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E1001));
		}
		else if (!HalfCharCheck.isLong1Check(itemValue))
		{
			// 形式チェックエラー
			errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E2001));
		}
		else if (!LengthCheck.isLength2Check(itemValue, 1, 8))
		{
			// 桁数チェックエラー
			errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E3001));
		}
		
		// 顧客ID
		// サイトID = 4211(コンシューマ)、4411(法人)の場合、実行
		if (SITE_ID_CONSUMER.equals(siteId) || SITE_ID_HOJIN.equals(siteId))
		{
			itemName = IN_CUSTOMER_ID;
			itemValue = (String)userData.get(itemName);
			// 必須チェック
			if ( itemValue == null || "".equals(itemValue) )
			{
				// 必須チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E1001));
			}
			else if (!HalfCharCheck.isEnNumber1Check(itemValue))
			{
				// 形式チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E2001));
			}
			else if (!LengthCheck.isLength1Check(itemValue, 16))
			{
				// 桁数チェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E3001));
			}
		}
		
		//単項目チェックエラーが発生していない場合、実行
		if (errList.size() == 0)
		{
			// サイトID組み合わせチェック
			itemName = IN_SITE_ID;
			itemValue = (String)userData.get(itemName);
			systemId = (String)userData.get(IN_SYSTEM_ID);
			systemIdValue = systemId.substring(0, 3);
			
			// サイトID組み合わせ取得
			String siteIdPair = JCMAPLConstMgr.getString(SITE_ID_PAIR + itemValue);
			
			if (siteIdPair == null || "".equals(siteIdPair))
			{
				// サイトIDチェックエラー
				errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E5002));
				userData.put(ERROR_INFO, errList);
				return false;
			}
			else
			{
				List<String> siteIdPairList = Arrays.asList(siteIdPair.split(","));
				
				if (!siteIdPairList.contains(systemIdValue))
				{
					// サイトIDチェックエラー
					errList.add(createErrorMap(itemName, itemValue, RES_KEY_ERROR_CODE_E5002));
					userData.put(ERROR_INFO, errList);
					return false;
				}
			}
		}
		
		// エラーが存在する場合
		if (errList.size() > 0)
		{
			userData.put(ERROR_INFO, errList);
			return false;
		}
		
		return true;
	}
	
	/**
	 * EKKA0010004_決済オーソリ依頼を実行する。
	 * <br>
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param fixedText ユーザ定義文字列
	 * @param userData ユーザーデータ
	 * @return 結果
	 * @throws Throwable 例外が発生した場合
	 */
	private CAANMsg callEKKA0010004(SessionHandle handle, IRequestParameterReadWrite param, String fixedText, HashMap<String, Object> userData) throws Throwable
	{
		// SC呼び出し部品のインスタンス生成（引数にはログに出力するクラス名を渡す。空文字を設定した場合はログに出力されない）
		ServiceComponentRequestInvoker scCall = new ServiceComponentRequestInvoker();
		
		// 決済オーソリ依頼の上りマッピング
		Object[][] ekka0010004In =
		{
			{EKKA0010004CBSMsg.TEMPLATEID, SC_NAME_EKKA0010004}
			// 機能コード："1"
			, {EKKA0010004CBSMsg.FUNC_CODE, "1"}
			// マーチャントＩＤ(後続処理でパラメータ取得用のキー情報)
			, {EKKA0010004CBSMsg.PY_MERCHANT_ID, PAYGENT_MERCHANT_ID}
			// 接続ID(後続処理でパラメータ取得用のキー情報)
			, {EKKA0010004CBSMsg.PY_CONNECT_ID, PAYGENT_CONNECT_ID}
			// 接続パスワード(後続処理でパラメータ取得用のキー情報)
			, {EKKA0010004CBSMsg.PY_CONNECT_PASSWORD, PAYGENT_CONNECT_PASSWORD}
			// 電文バージョン番号(後続処理でパラメータ取得用のキー情報)
			, {EKKA0010004CBSMsg.PY_TELEGRAM_VERSION, PAYGENT_TELEGRAM_VERSION}
			// 請求契約番号
			, {EKKA0010004CBSMsg.PY_SEIKY_KEI_NO, (String)userData.get(IN_SEIKY_KEI_NO)}
			// 決済金額
			, {EKKA0010004CBSMsg.PY_PAYMENT_AMOUNT, (String)userData.get(IN_CR_AMOUNT)}
			// 顧客ID
			, {EKKA0010004CBSMsg.PY_CUSTOMER_ID, (String)userData.get(IN_CUSTOMER_ID)}
			// サイトID
			, {EKKA0010004CBSMsg.PY_SITE_ID, (String)userData.get(IN_SITE_ID)}
		};
		
		// サービスIF実行
		CAANMsg ekka0010004MsgList = callSC(handle, scCall, param, fixedText, ekka0010004In);
		
		return ekka0010004MsgList;
	}
	
	/**
	 * SC(サービスインターフェイス）を呼び出す。
	 * 
	 * @param handle
	 * @param scCall
	 * @param param
	 * @param dataMapKey
	 * @param mappingData
	 * @return CAANMsg
	 * @throws Exception
	 */
	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];
		
		// リターンコード取得
		int returnCode = (Integer)result.get(JCMConstants.RET_CD_INT_KEY);
		
		int templateStatus = msg.getInt(JCMConstants.STATUS_INT_KEY);
		
		if (returnCode != 0)
		{
			templateStatus = 9000;
		}
		
		if (JCMAPLConstMgr.getString(RETURN_MESSAGE_STRING + String.format(RETURN_MESSAGE_FORMAT, 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(RETURN_MESSAGE_FORMAT, templateStatus);
			String message = JCMAPLConstMgr.getString(RETURN_MESSAGE_STRING + formatStatus);
			param.setControlMapData(SCControlMapKeys.RETURN_CODE, formatStatus);
			param.setControlMapData(SCControlMapKeys.RETURN_MESSAGE, message);
		}
		
		setErrorInf(msg, (HashMap<String, Object>)param.getData(dataMapKey));
		
		//エラー情報のマップを取得
		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));
		
		/* ◇処理結果の判定*/
		// 取得したリターンコード、ステータスの内容を見て異常かどうかの判断をする。
		String rtnCode = result.get(JCMConstants.RET_CD_INT_KEY).toString();
		Integer status = msg.getInt(JCMConstants.STATUS_INT_KEY);
		
		// 異常の場合、SCCallExceptionを生成してスローする
		if(!("0".equals(rtnCode) && 0 == status.intValue()))
		{
			SCCallException scCallEx = new SCCallException("戻り値不正", rtnCode, status);
			throw scCallEx;
		}
		return msg;
	}
	
	/**
	 * 共通領域のメッセージを作成します。
	 * 
	 * @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++)
		{
			if ("".equals(mappingData[i][1]))
			{
				template.setNull((String)mappingData[i][0]);
			}
			else
			{
				template.set((String)mappingData[i][0], mappingData[i][1]);
			}
		}
		
		setNullToMsg(template);
		
		CAANMsg[] templates = new CAANMsg[1];
		templates[0] = template;
		paramMap.put(JCMConstants.TEMPLATE_LIST_KEY, templates);
		return paramMap;
	}
	
	/**
	 * 未入力項目にnullを設定します。
	 * @param msg
	 */
	private void setNullToMsg(CAANMsg msg)
	{
		Iterator<String> caanMsgKeys = msg.getSchema().getSchemaKeySet().iterator();
		while (caanMsgKeys.hasNext())
		{
			String key = caanMsgKeys.next();
			if (key.endsWith(ERRITEM_SUFFIX))
			{
				String tmpKey = key.substring(0, key.length() - 4);
				
				if (!msg.containsKeyOfMsgData(tmpKey))
				{
					msg.setNull(tmpKey);
				}
			}
		}
	}
	
	/**
	 * エラー情報を設定します。
	 * 
	 * @param msg
	 * @param map
	 * 
	 */
	private void setErrorInf(CAANMsg msg, HashMap<String, Object> map)
	{
		Iterator<String> msgKey = msg.getSchema().getSchemaKeySet().iterator();
		
		while (msgKey.hasNext())
		{
			String key = msgKey.next();
			
			if (key.endsWith(ERRITEM_SUFFIX))
			{
				if (!msg.isNull(key))
				{
					if (!map.containsKey(key))
					{
						map.put(key, msg.getString(key));
					}
				}
			}
		}
	}
	
	/**
	 * エラーマップ作成メソッドです。
	 * 
	 * @param itemName 項目名
	 * @return
	 */
	private Map<String, String> createErrorMap(String itemName, String itemValue, String errCd)
	{
		Map<String, String> returnMap = new HashMap<String, String>();
		
		// エラーメッセージ設定
		String errMessage = itemName + JKKStrConst.COLON + itemValue;
		
		returnMap.put(ERROR_CODE, errCd);
		returnMap.put(ERROR_MESSAGE, errMessage);
		
		return returnMap;
	}
	
	/**
	 * 与信判定CCへのパラメータを作成する
	 * @param userData
	 * @throws Exception
	 */
	private void createYoshinJdgMap(IRequestParameterReadWrite param, Map<String, String> userData, String tradingId) throws Exception
	{
		
		HashMap<String, Object> targetMap = null;
		
		// ユーザデータ情報
		targetMap = (HashMap)param.getData(CC_NAME_YOSHIN_JDG);
		if (null == targetMap)
		{
			targetMap = new HashMap<String, Object>();
			param.setData(CC_NAME_YOSHIN_JDG, targetMap);
		}
		else
		{
			targetMap.clear();
		}
		targetMap.put(JCNYoshinJdgConstCC.INPUT_PAYWAY, PAYWAY_CREDIT); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_AUTHORI_RUN_FLG, AUTHORI_RUN_FLG); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_SYSTEM_ID, getParameterToString(userData, IN_SYSTEM_ID)); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_TRADING_ID, tradingId);
		targetMap.put(JCNYoshinJdgConstCC.INPUT_SITE_ID, getParameterToString(userData, IN_SITE_ID)); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_SYSID, getParameterToString(userData, IN_SYSID)); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_PAYWAYID, getParameterToString(userData, IN_REQ_PAYWAYID)); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_CR_AMOUNT, getParameterToString(userData, IN_CR_AMOUNT)); 
	}
	
	/**
	 * クレジットオーソリ承認CCへのパラメータを作成する
	 * @param userData
	 * @throws Exception
	 */
	private void createCreditAuthMap(IRequestParameterReadWrite param, Map<String, String> userData, String tradingId) throws Exception
	{
		
		HashMap<String, Object> targetMap = null;
		
		int amount = Integer.parseInt(userData.get(IN_CR_AMOUNT));
		
		// ユーザデータ情報
		targetMap = (HashMap)param.getData(CC_NAME_CREDIT_AUTH);
		if (null == targetMap)
		{
			targetMap = new HashMap<String, Object>();
			param.setData(CC_NAME_CREDIT_AUTH, targetMap);
		}
		else
		{
			targetMap.clear();
		}
		targetMap.put(JCNYoshinJdgConstCC.INPUT_PAYWAY, PAYWAY_CREDIT); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_AUTHORI_RUN_FLG, AUTHORI_RUN_FLG); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_SYSTEM_ID, getParameterToString(userData, IN_SYSTEM_ID)); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_TRADING_ID, tradingId);
		targetMap.put(JCNYoshinJdgConstCC.INPUT_SITE_ID, getParameterToString(userData, IN_SITE_ID)); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_SYSID, getParameterToString(userData, IN_SYSID)); 
		targetMap.put(JCNYoshinJdgConstCC.INPUT_PAYWAYID, getParameterToString(userData, IN_REQ_PAYWAYID)); 
		
		// 決済金額が0円、または、30000円以下の場合は、決済金額は1円で行う。
		if (amount == 0 || amount <= 30000) 
		{
			amount = 1;
		}
		targetMap.put(JCNYoshinJdgConstCC.INPUT_CR_AMOUNT, String.valueOf(amount));
	}
	
	/**
	 * パラメータマップを解析し、キーに一致する値を返却する。
	 * @param requestParamMap
	 * @param keyName
	 * @return String
	 * @throws 処理中に発生した例外をスロー
	 */
	private static String getParameterToString(Map<String, String> requestParamMap, String keyName) throws Exception
	{
		String value = "";
		
		Iterator<?> keyIterator = requestParamMap.keySet().iterator();
		
		while(keyIterator.hasNext())
		{
			String key = (String)keyIterator.next();
			
			if(key.equals(keyName) ||  key.matches(keyName + "\\[[0-9]*\\]"))
			{
				value = requestParamMap.get(key);
			}
			
		}
		return new String(value.getBytes(), "UTF-8");
	}
	
	/**
	 * 下りユーザデータマップのセット。
	 * @param ekka0010004Rslt SCの呼出処理の結果リスト
	 * @param rsltMap CCの呼出処理の結果マップ
	 * @param userData ユーザデータ
	 * @param jdgFlg 判定フラグ
	 * @throws Throwable
	 */
	@SuppressWarnings("unchecked")
	private static void setResultData(CAANMsg ekka0010004Rslt, HashMap rsltMap, HashMap userData, String jdgFlg, String orderdate, String ordertime) throws Throwable
	{
		// 実行された処理がSCの場合
		if (JDG_FLG_0.equals(jdgFlg))
		{
			// 処理結果コード
			userData.put(OUT_CR_RESULTCODE, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_RESULTCODE)));
			// レスポンスコード
			userData.put(OUT_RESPONSE_CODE, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_RESPONSE_CODE)));
			// レスポンス詳細
			userData.put(OUT_RESPONSE_DETAIL, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_RESPONSE_DETAIL)));
			// 決済ID
			userData.put(OUT_PAYMENT_ID, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_PAYMENT_ID)));
			// マーチャント取引ID
			userData.put(OUT_TRADING_ID, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_TRADING_ID)));
			// イシュア区分
			userData.put(OUT_ISSUER_CLASS, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_ISSUR_CLASS)));
			// 仕向先カード会社コード
			userData.put(OUT_CR_BRANDCODE, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_ACQ_ID)));
			// ハッシュタグ
			userData.put(OUT_HASHTAG, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_FINGERPRINT)));
			// カード番号
			userData.put(OUT_CR_CARDNUMBER, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_MASKED_CARD_NUMBER)));
			// カード有効期限
			userData.put(OUT_CARD_VALID_TERM, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_CARD_VALID_TERM)));
			// オーソリ承認番号
			userData.put(OUT_CR_AUTHNUMBER, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_SHONIN_NO)));
			// カード種別
			userData.put(OUT_CARD_TYPE, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_DEBIT_PREPAID_TYPE)));
			// カードブランド
			userData.put(OUT_CARD_BRAND_OPTCD, JKKStringUtil.nullToBlank(ekka0010004Rslt.getString(EKKA0010004CBSMsg.PY_CARD_BRAND)));
			// 取引年月日
			userData.put(CR_ORDERDATE, JKKStringUtil.nullToBlank(orderdate));
			// 取引時間
			userData.put(CR_ORDERTIME, JKKStringUtil.nullToBlank(ordertime));
			// 支払方法ID
			userData.put(OUT_PAY_WAY_ID, "");
		}
		// 実行された処理がCCの場合
		else if (JDG_FLG_1.equals(jdgFlg))
		{
			// 処理結果コード
			userData.put(OUT_CR_RESULTCODE, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_CR_RESULTCODE)));
			// レスポンスコード
			userData.put(OUT_RESPONSE_CODE, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_RESPONSE_CODE)));
			// レスポンス詳細
			userData.put(OUT_RESPONSE_DETAIL, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_RESPONSE_DETAIL)));
			// 決済ID
			userData.put(OUT_PAYMENT_ID, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_PAYMENT_ID)));
			// マーチャント取引ID
			userData.put(OUT_TRADING_ID, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_TRADING_ID)));
			// イシュア区分
			userData.put(OUT_ISSUER_CLASS, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_ISSUER_CLASS)));
			// 仕向先カード会社コード
			userData.put(OUT_CR_BRANDCODE, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_CR_BRANDCODE)));
			// ハッシュタグ
			userData.put(OUT_HASHTAG, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_HASHTAG)));
			// カード番号
			userData.put(OUT_CR_CARDNUMBER, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_CR_CARDNUMBER)));
			// カード有効期限
			userData.put(OUT_CARD_VALID_TERM, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_CARD_VALID_TERM)));
			// オーソリ承認番号
			userData.put(OUT_CR_AUTHNUMBER, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_CR_AUTHNUMBER)));
			// カード種別
			userData.put(OUT_CARD_TYPE, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_CARD_TYPE)));
			// カードブランド
			userData.put(OUT_CARD_BRAND_OPTCD, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_CARD_BRAND_OPTCD)));
			// 取引年月日
			userData.put(CR_ORDERDATE, JKKStringUtil.nullToBlank(orderdate));
			// 取引時間
			userData.put(CR_ORDERTIME, JKKStringUtil.nullToBlank(ordertime));
			// 支払方法ID
			userData.put(OUT_PAY_WAY_ID, JKKStringUtil.nullToBlank((String)rsltMap.get(JCNCreditAuthConstCC.OUTPUT_PAYWAYID)));
		}
		
	}
	
	/**
	 * マーチャント取引IDを作成する
	 * @param userData
	 */
	private String createTradingId(HashMap userData, String orderdate, String ordertime)
	{
		String tradingId = null;
		
		//サイトID
		String siteId = (String)userData.get(IN_SITE_ID);
		// 連携支払方法ID 
		String reqPayWayId = (String)userData.get(IN_REQ_PAYWAYID);
		// オンライン運用日(YYMMDD)
		String date = orderdate.substring(2, 8);
		// オンライン時刻(NNNNN)
		String time = changeTime(ordertime);
		
		// マーチャント取引ID = サイトID＋連携支払方法ID＋0＋YYMMDD＋NNNNN（NNNNN=1日24時間としたときの秒数）
		tradingId = siteId + reqPayWayId + "0" + date + time;
		
		return tradingId;
	}
	
	/**
	 * 時刻を秒に変換する
	 * @param time オンライン時刻
	 */
	private String changeTime(String time)
	{
		long sumTime = 0;
		String rsltTime = null;
		int digitno = 5;
		
		// 時
		long changeHours = TimeUnit.HOURS.toSeconds(Integer.parseInt(time.substring(0, 2)));
		// 分
		long changeMinutes = TimeUnit.MINUTES.toSeconds(Integer.parseInt(time.substring(2, 4)));
		// 秒
		long seconds = Integer.parseInt(time.substring(4, 6));
		
		// 時、分、秒の合計値
		sumTime = changeHours + changeMinutes + seconds;
		
		rsltTime = String.valueOf(sumTime);
		
		if (digitno > rsltTime.length())
		{
			// 合計時間が5桁以下の場合、不足している桁数分「0」を先頭に結合
			for (int i = rsltTime.length(); i < digitno; i++)
			{
				rsltTime = "0" + rsltTime;
			}
		}
		
		return rsltTime;
		
	}
	
}