/*********************************************************************
*	All Rights reserved,Copyright (c) K-Opticom
**********************************************************************
*＜プログラム内容＞
*	システム名		：eo顧客基幹システム
*	モジュール名	：JCRToiawaseRirekiInfoShokaiOpsCC
*	ソースファイル名：JCRToiawaseRirekiInfoShokaiOpsCC.java
*	作成者			：FJ)吉田
*	日付			：2019年04月01日
*＜機能概要＞
*	問合わせ履歴情報照会（オペレータ支援）CCの共通コンポーネントクラスです。
*＜修正履歴＞
*	バージョン	修正日		修正者		修正内容
*	v42.00.00	2019/04/01	FJ)吉田		【ANK-3593-00-00】オペレータ支援対応に伴うeo顧客機能改修（お客さまカルテ）
*	v49.00.00	2020/04/21	FJ)坂本		【ANK-3831-00-00】オペレータ支援向けAPI・画面機能変更(2020年5月下旬リリース)
**********************************************************************/
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.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.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.constant.JPCModelConstant;
import eo.common.util.JKKStringUtil;
import eo.ejb.cbs.cbsmsg.ECR0011B170CBSMsg;
import eo.ejb.cbs.cbsmsg.ECR0011B170CBSMsg1List;

public class JCRToiawaseRirekiInfoShokaiOpsCC extends AbstractCommonComponent{
	
	// ----------
	// TemplateID
	// ----------
	/** TamplateID：ECR0011B170_対応記録一覧照会(問合せ履歴) */
	private final String TEMPLATE_ID_ECR0011B170 = "ECR0011B170";
	
	// ------------------------
	// CCパラメータ
	// ------------------------
	/** CCパラメータ：ボディ情報 */
	private final String CC_PARAM_BODY_INFO = "BODY_INFO";
	/** CCパラメータ：エラー情報 */
	private final String CC_PARAM_ERROR_INFO = "ERROR_INFO";
	
	// --------------
	// パラメータ定数
	// --------------
	/** パラメータ：エラーコード */
	private final String PARAM_ERRCODE = "errCode";
	/** パラメータ：エラーメッセージ */
	private final String PARAM_ERRMESSAGE = "errMessage";
	/** パラメータ：SYSID */
	private final String PARAM_SYSID = "sysid";
	/** パラメータ：問合せ履歴情報リスト */
	private final String PARAM_TOIAWASE_RIREKI_IF_LIST = "toiawase_rireki_if_list";
	/** パラメータ：問合せID */
	private final String PARAM_TOIAWASE_ID = "toiawase_id";
	/** パラメータ：問合せ受付種別コード */
	private final String PARAM_TOIAWASE_UK_SBT_CD = "toiawase_uk_sbt_cd";
	/** パラメータ：問合せ受付種別名称*/
	private final String PARAM_TOIAWASE_UK_SBT_NM = "toiawase_uk_sbt_nm";
	/** パラメータ：対応開始年月日時分秒 */
	private final String PARAM_TAIO_STA_DTM = "taio_sta_dtm";
	/** パラメータ：コンタクト先コード */
	private final String PARAM_CONTACT_SAKI_CD = "contact_saki_cd";
	/** パラメータ：コンタクト先名称 */
	private final String PARAM_CONTACT_SAKI_NM = "contact_saki_nm";
	/** パラメータ：対応部署名称*/
	private final String PARAM_TAIO_BUSHO_NM = "taio_busho_nm";
// ANK-3831-00-00 ADD START
	/** パラメータ：対応最終更新年月日時分秒*/
	private final String PARAM_TAIO_LAST_UPD_DTM = "taio_last_upd_dtm";
// ANK-3831-00-00 ADD END
	
	// ----------
	// エラー定数
	// ----------
	/** エラーコード */
	public static final String ERROR_CODE = "errCode";
	/** エラーメッセージ */
	public static final String ERROR_MESSAGE = "errMessage";
	/** リターンメッセージ取得キー */
	private static final String RETURN_MESSAGE_STRING = "RETURN_MESSAGE_";
	/** リターンメッセージの書式指定文字列 */
	private static final String RETURN_MESSAGE_FORMAT = "%1$04d";
	/** エラー項目接尾辞 */
	private static final String ERRITEM_SUFFIX = "_err";
	
	/**
	 * 問合わせ履歴情報照会（オペレータ支援）を行う。
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param fixedText ユーザ定義文字列
	 * @return IRequestParameterReadWrite
	 * @throws Throwable 例外が発生した場合
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite searchToiawaseRirekiOps(SessionHandle handle,
			IRequestParameterReadWrite param, String fixedText) throws Throwable
	{
		// ------------
		// CCマップ取得
		// ------------
		// 問合わせ履歴情報照会（オペレータ支援）CCマップの取得
		Map<String, Object> ccMsg = (Map<String, Object>)param.getData(fixedText);
		
		// ------------------
		// 各種エラーチェック
		// ------------------
		// 単項目チェック
		if (!unitErrorChk(ccMsg))
		{
			param.setControlMapData(SCControlMapKeys.RETURN_CODE, JKKStrConst.RETURN_CD_5000);
			return param;
		}
		
		// 問合せ履歴情報照会メイン処理
		getToiawaseRireki(handle, param, fixedText, ccMsg);
		
		return param;
	}
	
	/**
	 * 単項目エラーチェックを行います。
	 * 
	 * @param ccMsg 
	 * @return
	 */
	private boolean unitErrorChk(Map<String, Object> ccMsg)
	{
		// エラーリスト
		List<Map<String, String>> errList = new ArrayList<Map<String, String>>();
		String itemValue = null;
		
		// SYSID
		itemValue = (String)ccMsg.get(PARAM_SYSID);
		// 必須チェック
		if (JKKStringUtil.isNullBlank(itemValue))
		{
			// 必須チェックエラー
			errList.add(getReqErrInfMap());
		}
		else if (!HalfCharCheck.isEnNumber1Check(itemValue))
		{
			// 属性チェックエラー
			errList.add(getFormErrInfMap());
		}
		else if (!LengthCheck.isLength1Check(itemValue, 10))
		{
			// 桁数チェックエラー
			errList.add(getLenErrInfMap());
		}
		
		// エラーが存在する場合
		if (errList.size() > 0)
		{
			// body用マップ
			Map<String, Object> bodyMap = new HashMap<String, Object>();
			
			ccMsg.put(CC_PARAM_BODY_INFO, bodyMap);
			ccMsg.put(CC_PARAM_ERROR_INFO, errList);
			return false;
		}
		
		return true;
	}
	
	/**
	 * 問合わせ履歴情報照会（オペレータ支援）メイン処理
	 * 
	 * @param handle
	 * @param param
	 * @param fixedText
	 * @param ccMsg 
	 * @throws Throwable 
	 */
	private void getToiawaseRireki(SessionHandle handle,IRequestParameterReadWrite param, String fixedText, Map<String, Object> ccMsg) throws Throwable
	{
		// SYSID
		String sysid = (String)ccMsg.get(PARAM_SYSID);
		
		// ECR0011B170_対応記録一覧照会(問合せ履歴)の実行
		CAANMsg[] resultECR0011B170 = execECR0011B170(handle, param, fixedText, sysid);
		
		// 取得データがなかった場合
		if (resultECR0011B170 == null || resultECR0011B170.length == 0)
		{
			// 件数結果チェックエラー
			setResponse(param, ccMsg, JKKStrConst.RETURN_CD_2001, "");
			param.setControlMapData(SCControlMapKeys.RETURN_CODE, JKKStrConst.RETURN_CD_5000);
			return;
		}
		
		// レスポンス用：問合せ履歴情報リスト
		List<Map<String, String>> toiawaseRirekiIfList = new ArrayList<Map<String, String>>();
		
		// データが取得できている場合は、返却データを対応する項目にセットする。
		// データ件数分ループ
		for (int listcnt = 0; listcnt < resultECR0011B170.length; listcnt++)
		{
			// レスポンス用：問合せ履歴情報マップ
			Map<String, String> toiawaseRirekiIfMap = new HashMap<String, String>();
			// レコードを取得
			CAANMsg dataMap = resultECR0011B170[listcnt];
			
			// 対応する項目に取得データをセット
			// 問合せID
			toiawaseRirekiIfMap.put(PARAM_TOIAWASE_ID, dataMap.getString(ECR0011B170CBSMsg1List.TAIO_KIROK_NO));
			// 問合せ受付種別コード
			toiawaseRirekiIfMap.put(PARAM_TOIAWASE_UK_SBT_CD, JKKStringUtil.nullToBlank(dataMap.getString(ECR0011B170CBSMsg1List.TOIAWASE_UK_SBT_CD)));
			// 問合せ受付種別名称
			toiawaseRirekiIfMap.put(PARAM_TOIAWASE_UK_SBT_NM, JKKStringUtil.nullToBlank(dataMap.getString(ECR0011B170CBSMsg1List.TOIAWASE_UK_SBT_NM)));
			// 対応開始年月日時分秒
			toiawaseRirekiIfMap.put(PARAM_TAIO_STA_DTM, dataMap.getString(ECR0011B170CBSMsg1List.TAIO_STA_DTM));
			// コンタクト先コード
			toiawaseRirekiIfMap.put(PARAM_CONTACT_SAKI_CD, JKKStringUtil.nullToBlank(dataMap.getString(ECR0011B170CBSMsg1List.CONTACT_SAKI_CD)));
			// コンタクト先名称
			toiawaseRirekiIfMap.put(PARAM_CONTACT_SAKI_NM, JKKStringUtil.nullToBlank(dataMap.getString(ECR0011B170CBSMsg1List.CONTACT_SAKI_NM)));
			// 対応部署名称
			toiawaseRirekiIfMap.put(PARAM_TAIO_BUSHO_NM, JKKStringUtil.nullToBlank(dataMap.getString(ECR0011B170CBSMsg1List.TAIO_BUSHO_NM)));
// ANK-3831-00-00 ADD START
			// 対応最終更新年月日時分秒
			toiawaseRirekiIfMap.put(PARAM_TAIO_LAST_UPD_DTM, JKKStringUtil.nullToBlank(dataMap.getString(ECR0011B170CBSMsg1List.TAIO_LAST_UPD_DTM)));
// ANK-3831-00-00 ADD END
			
			// 問合せ履歴情報マップを問合せ履歴情報リストに追加
			toiawaseRirekiIfList.add(toiawaseRirekiIfMap);
		}
		// ボディ情報マップ
		Map<String, Object> bodyMap = new HashMap<String, Object>();
		// 作成した問合せ履歴情報リストをボディ情報にセット
		bodyMap.put(PARAM_TOIAWASE_RIREKI_IF_LIST, toiawaseRirekiIfList);
		bodyMap.put(PARAM_SYSID, sysid);
		
		// 作成したボディ情報をセット
		ccMsg.put(CC_PARAM_BODY_INFO, bodyMap);
		// 空のエラー情報をセット
		ccMsg.put(CC_PARAM_ERROR_INFO, new ArrayList<Map<String, String>>());
	}
	
	/**
	 * 単項目チェックの必須エラーマップ作成メソッドです。
	 * 
	 * @return
	 */
	public Map<String, String> getReqErrInfMap()
	{
		String errorCode = JKKStrConst.RETURN_CD_1001;
		String errorMessage = "";
		
		return getCommonErrInfMap(errorCode, errorMessage);
	}
	
	/**
	 * 単項目チェックの属性エラーマップ作成メソッドです。
	 * 
	 * @return
	 */
	public Map<String, String> getFormErrInfMap()
	{
		String errorCode = JKKStrConst.RETURN_CD_1002;
		String errorMessage = "";
		
		return getCommonErrInfMap(errorCode, errorMessage);
	}
	
	/**
	 * 単項目チェックの桁数エラーマップ作成メソッドです。
	 * 
	 * @return
	 */
	public Map<String, String> getLenErrInfMap()
	{
		String errorCode = JKKStrConst.RETURN_CD_1003;
		String errorMessage = "";
		
		return getCommonErrInfMap(errorCode, errorMessage);
	}
	
	/**
	 * 各業務エラーチェックのエラーマップ作成メソッド
	 * 
	 * @param return_cd
	 * @param errorMessage
	 * @return
	 */
	private Map<String, String> getCommonErrInfMap(String return_cd, String errorMessage)
	{
		Map<String, String> returnMap = new HashMap<String, String>();
		
		returnMap.put(PARAM_ERRCODE, return_cd);
		returnMap.put(PARAM_ERRMESSAGE, errorMessage);
		
		return returnMap;
	}
	
	/**
	 * レスポンス項目を設定します。
	 * <br>
	 * @param ccMsg お客様住所情報更新受付(CMP)CCマップ
	 * @param returnCode リターンコード
	 * @param returnMessage リターンメッセージ
	 */
	private void setResponse(IRequestParameterReadWrite param,  Map<String, Object> ccMsg, String errCode, String errMessage)
	throws Exception
	{
		// エラーリスト
		List<Map<String, String>> errList = new ArrayList<Map<String, String>>();
		// body用マップ
		Map<String, Object> bodyMap = new HashMap<String, Object>();
		
		// 引数のエラー情報をエラーリストにセット
		errList.add(getCommonErrInfMap(errCode, errMessage));
		
		ccMsg.put(CC_PARAM_BODY_INFO, bodyMap);
		ccMsg.put(CC_PARAM_ERROR_INFO, errList);
	}
	
	/**
	 * ECR0011B170_対応記録一覧照会(問合せ履歴)を呼び出す。
	 * 
	 * @param handle
	 * @param param
	 * @param fixedText
	 * @param sysid
	 * @return 
	 * @throws Throwable
	 */
	private CAANMsg[] execECR0011B170(SessionHandle handle,IRequestParameterReadWrite param, String fixedText, String sysid) throws Throwable
	{
		// 上りマッピング
		Object[][] inmap = {
				  {ECR0011B170CBSMsg.TEMPLATEID, TEMPLATE_ID_ECR0011B170}
				, {ECR0011B170CBSMsg.FUNC_CODE, JPCModelConstant.FUNC_CD_1}
				, {ECR0011B170CBSMsg.KEY_SYSID, sysid}
		};
		
		// SC実行
		CAANMsg[] msgList = callSC(handle,  param, inmap,
				fixedText, new ECR0011B170CBSMsg().getContents()).getCAANMsgList(ECR0011B170CBSMsg.ECR0011B170CBSMSG1LIST);
		
		return msgList;
	}
	
	/**
	 * SC(サービスインターフェイス）を呼び出す。
	 * @param handle セッションハンドル
	 * @param scCall SC呼び出し部品
	 * @param param リクエストパラメータ
	 * @param mappingData マッピングデータ
	 * @param fixedText ユーザ定義文字列
	 * @param contents CAANMsgのコンテンツ
	 * @return CAANMsg
	 * @throws Throwable 例外が発生した場合
	 */
	@SuppressWarnings("unchecked")
	private CAANMsg callSC(SessionHandle handle,
			IRequestParameterReadWrite param, Object[][] mappingData,
			String fixedText, Object[][] contents) throws Throwable
	{
		// 上りマッピング処理
		HashMap<String, Object> paramMap = editInMsg(param, mappingData);
		
		// SC呼び出し部品のインスタンス生成（引数にはログに出力するクラス名を渡す。空文字を設定した場合はログに出力されない）
		ServiceComponentRequestInvoker scCall = new ServiceComponentRequestInvoker();
		
		// サービスI/F呼出し実行
		Map<?, ?> result = scCall.run(paramMap, handle);
		
		// SCからの戻り値からCAANMsgを取得
		CAANMsg[] templates = (CAANMsg[])result.get(JCMConstants.TEMPLATE_LIST_KEY);
		CAANMsg msg = templates[0];
		
		// リターンコード取得
		Object returnCode = result.get(JCMConstants.RET_CD_INT_KEY);
		// ステータス取得
		int status = msg.getInt(JCMConstants.STATUS_INT_KEY);
		
		// エラー情報の設定
		editErrorInfo(param, templates, (Integer)returnCode, fixedText, contents);
		
		// エラー情報のマップを取得
		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));
		
		// 異常の場合、CCExceptionを生成してスローする
		if(!("0".equals(returnCode.toString()) && 0 == status))
		{
			throw new CCException("サービスコンポーネントエラー", new Exception());
		}
		
		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));
		
		// サービスインターフェイス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 (mappingData[i][1] instanceof CAANMsg[])
			{
				template.set((String)mappingData[i][0], (CAANMsg[])mappingData[i][1]);
			}
			else
			{
				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 CAANMsgクラス
	 * @param returnCode リターンコード
	 * @param fixedText ユーザ定義文字列
	 * @param contents CAANMsgのコンテンツ
	 * @return IRequestParameterReadWrite
	 * @throws RequestParameterException リクエストパラメータの操作でエラーが発生した場合
	 */
	@SuppressWarnings("unchecked")
	private IRequestParameterReadWrite editErrorInfo(IRequestParameterReadWrite param,
			CAANMsg[] templates, int returnCode, String fixedText, Object[][] contents) throws RequestParameterException
	{
		CAANMsg template = templates[0];
		int templateStatus = template.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 (bpStatus < templateStatus)
		{
			// 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);
		}
		
		// ユーザデータ情報
		HashMap<String, String> inMap = (HashMap<String, String>)param.getData(fixedText);
		
		for (int i = 0; null != contents && i < contents.length ; i++)
		{
			String itemNm = (String)contents[i][0];
			if (itemNm.endsWith(ERRITEM_SUFFIX))
			{
				String errCd = (String)template.getString(itemNm);
				if (!JKKStringUtil.isNullBlank(errCd))
				{
					inMap.put(itemNm, errCd);
				}
			}
		}
		
		return param;
	}
}
