/*********************************************************************
* All Rights reserved,Copyright (c) K-Opticom
**********************************************************************
*＜プログラム内容＞
*	システム名		：eo顧客基幹システム
*	モジュール名	：JKKTsushinJokyoInfoCC
*	ソースファイル名：JKKTsushinJokyoInfoCC.java
*	作成者			：富士通
*	日付			：2011年12月13日
*＜機能概要＞
*	通信状況情報照会CC
*＜修正履歴＞
*	バージョン	修正日		修正者		修正内容
*	v1.00.00	2011/12/13	FJ)古内		新規作成
*	v3.00.00	2012/08/20	FJ)団		【ANK-0546-02-00】イオスマートリンク対応
*	v4.00.00	2013/02/03	FJ)強矢		【ST1-2013-0000217】対応
*	v4.00.01	2013/05/02	FJ)強矢		【IT1-2013-0001118】性能障害対応
*
**********************************************************************/
package com.fujitsu.futurity.bp.custom.common;

import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

import com.fujitsu.futurity.bp.x21.bpm.ServiceComponentRequestInvoker;
import com.fujitsu.futurity.bp.x21.bpm.db.SessionHandle;
import com.fujitsu.futurity.bp.x21.bpm.exception.RequestParameterException;
import com.fujitsu.futurity.bp.x21.bpm.parameter.IRequestParameterReadWrite;
import com.fujitsu.futurity.bp.x21.cc.AbstractCommonComponent;
import com.fujitsu.futurity.bp.x21.cc.exception.SCCallException;
import com.fujitsu.futurity.common.JCMConstants;
import com.fujitsu.futurity.common.x01.sc.SCControlMapKeys;
import com.fujitsu.futurity.mapping.bp.common.TemplateErrorUtil;
import com.fujitsu.futurity.model.base.CAANMsg;
import com.fujitsu.futurity.model.common.JCMAPLConstMgr;

import eo.ejb.cbs.cbsmsg.EAC0581B501CBSMsg;
import eo.ejb.cbs.cbsmsg.EAC0581B501CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EAC0591B501CBSMsg;
import eo.ejb.cbs.cbsmsg.EAC0591B501CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0161B004CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0161B004CBSMsg1List;
import eo.ejb.cbs.cbsmsg.EKK0171A010CBSMsg;
import eo.ejb.cbs.cbsmsg.EKK0171A010CBSMsg1List;


/**
 * 通信状況情報の照会を行う部品。
 * 
 * @author 909141
 *
 */
public class JKKTsushinJokyoInfoCC 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(EKK0161B004) */
	private static final String TEMPLATE_ID_EKK0161B004 = "EKK0161B004";
	
	/** テンプレートID(EKK0171A010) */
	private static final String TEMPLATE_ID_EKK0171A010 = "EKK0171A010";
	
	/** テンプレートID(EAC0581B501) */
	private static final String TEMPLATE_ID_EAC0581B501 = "EAC0581B501";
	
	/** テンプレートID(EAC0591B501) */
	private static final String TEMPLATE_ID_EAC0591B501 = "EAC0591B501";
	
	/** サービスID */
	private static final String SERVICE_ID = "KKSV033402CC";
	
	/** 単位変換[MB]→[GB] */
	private static final String MB_TO_CONV_GB = "1000";
	
	/** 単位変換[KB]→[GB] */
	private static final String KB_TO_CONV_GB = "1000000";
	
	/**
	 * 通信状況情報の照会を行う。
	 * 
	 * @param handle
	 * @param param
	 * @param fixedText
	 * @return IRequestParameterReadWrite
	 * @throws Exception
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite searchTsushinJokyoInfo(SessionHandle handle, 
																IRequestParameterReadWrite param, 
																String fixedText) throws Exception
	{
		HashMap<String, Object> work02Map = new HashMap<String, Object>();
		param.setData("KKSV0334WORK02", work02Map);
		
		// ◇ SC呼出実行
		// SC呼び出し部品のインスタンス生成（引数にはログに出力するクラス名を渡す。空文字を設定した場合はログに出力されない）
		ServiceComponentRequestInvoker scCall = new ServiceComponentRequestInvoker();
		
		// ◇ 上りマッピング処理
		HashMap<String, Object> ccMsg = (HashMap<String, Object>)param.getData(SERVICE_ID);
		
		// *** EKK0161B004(サービス契約内訳一覧照会) の実行 ***
		Object[][] ekk0161b004IN = {
			// CC.ＫＥＹ＿サービス契約番号 => ＫＥＹ＿サービス契約番号
			{EKK0161B004CBSMsg.KEY_SVC_KEI_NO, (String)ccMsg.get("key_svc_kei_no")},
			// "" => ＫＥＹ＿料金コースコード
			{EKK0161B004CBSMsg.KEY_PCRS_CD, ""},
			// "" => ＫＥＹ＿サービス期間中フラグ
			{EKK0161B004CBSMsg.KEY_SVC_PRD_FLG, ""}
		};
		
		// ◇ SC呼出実行
		CAANMsg ekk0161b004Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0161B004, "1", ekk0161b004IN);
		
		// 処理結果判定
		CAANMsg[] ekk0161b004MsgList = ekk0161b004Msg.getCAANMsgList(EKK0161B004CBSMsg.EKK0161B004CBSMSG1LIST);
		
		// 認証ＩＤリスト
		ArrayList<String> ispNinshoIdList = new ArrayList<String>();
		
		for (int i = 0; i < ekk0161b004MsgList.length; i++)
		{
			String svcKeiUcwkStat = ekk0161b004MsgList[i].getString(EKK0161B004CBSMsg1List.SVC_KEI_UCWK_STAT);
			// "020"(照査済)"030"(締結済)"100"(サービス提供中)"210"(休止・中断中)"220"(停止中)以外は対象データとしない。
			if (!("020".equals(svcKeiUcwkStat) 
					|| "030".equals(svcKeiUcwkStat) 
					|| "100".equals(svcKeiUcwkStat) 
					|| "210".equals(svcKeiUcwkStat)
					|| "220".equals(svcKeiUcwkStat)))
			{
				continue;
			}
			
			// *** EKK0171A010(サービス契約内訳＜eo光ネット＞一意照会) の実行 ***
			Object[][] ekk0171a010IN = {
				// EKK0161B004.サービス契約内訳番号 => ＫＥＹ＿サービス契約内訳番号
				{EKK0171A010CBSMsg.KEY_SVC_KEI_UCWK_NO, ekk0161b004MsgList[i].getString(EKK0161B004CBSMsg1List.SVC_KEI_UCWK_NO)},
				// EKK0161B004.世代登録年月日時分秒 => ＫＥＹ＿世代登録年月日時分秒
				{EKK0171A010CBSMsg.KEY_GENE_ADD_DTM, ekk0161b004MsgList[i].getString(EKK0161B004CBSMsg1List.GENE_ADD_DTM)}
			};
			
			// ◇ SC呼出実行
			CAANMsg ekk0171a010Msg = callSC(handle, param, scCall, TEMPLATE_ID_EKK0171A010, "1", ekk0171a010IN);
			
			// 処理結果判定
			CAANMsg[] ekk0171a010MsgList = ekk0171a010Msg.getCAANMsgList(EKK0171A010CBSMsg.EKK0171A010CBSMSG1LIST);
			
			// ※ 一覧照会に対する一意照会であるため、必ずデータが存在する
			
			// ISP認証IDを取得できない場合
			if (ekk0171a010MsgList[0].isNull(EKK0171A010CBSMsg1List.ISP_NINSHO_ID))
			{
				continue;
			}
			
			String ispNinshoId = ekk0171a010MsgList[0].getString(EKK0171A010CBSMsg1List.ISP_NINSHO_ID);
			
			// 認証ＩＤが取得できた場合、認証ＩＤリストに追加する。
			ispNinshoIdList.add(ispNinshoId);
		}
		
		if (0 < ispNinshoIdList.size())
		{
			HashMap<String, Object> work03Map = (HashMap<String, Object>)param.getData("KKSV0334WORK03");
			
			// *** EAC0581B501(通信料超過通知設定一覧照会(上限超過通知種別検索)) の実行 ***
			Object[][] ekk0581b501IN = {
				// work03.料金コースコード => ＫＥＹ＿料金コースコード
				{EAC0581B501CBSMsg.KEY_PCRS_CD, work03Map.get("pcrs_cd")},
				// work03.料金プランコード => ＫＥＹ＿料金プランコード
				{EAC0581B501CBSMsg.KEY_PPLAN_CD, work03Map.get("pplan_cd")},
				// "2"(帯域制御超過通知) => ＫＥＹ＿上限超過通知種別コード
				{EAC0581B501CBSMsg.KEY_UPPL_CHOK_TCH_SBT_CD, "2"}
			};
			
			// ◇ SC呼出実行
			CAANMsg ekk0581b501Msg = callSC(handle, param, scCall, TEMPLATE_ID_EAC0581B501, "1", ekk0581b501IN);
			
			// 処理結果判定
			CAANMsg[] ekk0581b501MsgList = ekk0581b501Msg.getCAANMsgList(EAC0581B501CBSMsg.EAC0581B501CBSMSG1LIST);
			
			String max_data_dnsu_ryo = "";
			
			if (0 < ekk0581b501MsgList.length)
			{
				// 通知対象通信量[MB]
				max_data_dnsu_ryo = (String)ekk0581b501MsgList[0].getString(EAC0581B501CBSMsg1List.TCHI_TG_TSRYO);
				if (max_data_dnsu_ryo != null && !"".equals(max_data_dnsu_ryo))
				{
					BigDecimal max_tsryo = new BigDecimal(max_data_dnsu_ryo);
					
					// [MB]→[GB]変換(小数第１位を切り上げ)
					BigDecimal calc = max_tsryo.divide(new BigDecimal(MB_TO_CONV_GB), 0, BigDecimal.ROUND_UP);
					max_data_dnsu_ryo = String.valueOf(calc.longValue());
				}
			}
			
			// オンライン処理日(当月,前月,前々月)
			String[] onlineProcessDate = {
				JPCBPCommon.getOpeDate(null).substring(0, 6),
				JPCBPCommon.addMonth(JPCBPCommon.getOpeDate(null), -1).substring(0, 6),
				JPCBPCommon.addMonth(JPCBPCommon.getOpeDate(null), -2).substring(0, 6)
			};
			
			for (int i = 0; i < ispNinshoIdList.size(); i++)
			{
				// ＩＳＰ認証ＩＤ
				String ispNinshoId = ispNinshoIdList.get(i);
				
				HashMap<String, Object> transportData = new HashMap<String, Object>();
				
				// オンライン処理日の件数分繰り返す。
				for (int j = 0; j < onlineProcessDate.length; j++)
				{
					// *** EAC0591B501(通信量明細合計一覧照会(複合検索)) の実行 ***
					Object[][] ekk0591b501IN = {
						// CC.ＫＥＹ＿サービス契約番号 => ＫＥＹ＿サービス契約番号
						{EAC0591B501CBSMsg.KEY_SVC_KEI_NO, (String)ccMsg.get("key_svc_kei_no")},
						// work03.料金コースコード => ＫＥＹ＿料金コースコード
						{EAC0591B501CBSMsg.KEY_PCRS_CD, work03Map.get("pcrs_cd")},
						// "" => ＫＥＹ＿料金サービスコード
						{EAC0591B501CBSMsg.KEY_PRC_SVC_CD, ""},
						// EKK0171A010.ＩＳＰ認証ＩＤ => ＫＥＹ＿認証ＩＤ
						{EAC0591B501CBSMsg.KEY_NINSHO_ID, ispNinshoId},
						// オンライン処理日(当月,前月,前々月) => ＫＥＹ＿利用年月
						{EAC0591B501CBSMsg.KEY_USE_YM, onlineProcessDate[j]},
						// "" => ＫＥＹ＿データ取得年月日
						{EAC0591B501CBSMsg.KEY_DATA_STKU_YMD, ""}
					};
					
					// ◇ SC呼出実行
					CAANMsg ekk0591b501Msg = callSC(handle, param, scCall, TEMPLATE_ID_EAC0591B501, "1", ekk0591b501IN);
					
					// 処理結果判定
					CAANMsg[] ekk0591b501MsgList = ekk0591b501Msg.getCAANMsgList(EAC0591B501CBSMsg.EAC0591B501CBSMSG1LIST);
					
					if (ekk0591b501MsgList.length > 0)
					{
						// 利用通信量合計[KB]
						String use_tsryo_gk = (String)ekk0591b501MsgList[0].getString(EAC0591B501CBSMsg1List.USE_TSRYO_GK);
						// 補正通信量合計[KB](マイナス有)
						String hosei_tsryo_gk = (String)ekk0591b501MsgList[0].getString(EAC0591B501CBSMsg1List.HOSEI_TSRYO_GK);
						
						// 利用通信量合計[KB]と補正通信量合計[KB]の合計[GB]
						String calcDnsuRyo = calculateDnsuRyo(use_tsryo_gk, hosei_tsryo_gk);
						
						switch (j)
						{
							case 0:
								// データ伝送量（当月）
								transportData.put("data_dnsu_ryo", calcDnsuRyo);
								break;
								
							case 1:
								// データ伝送量（前月）
								transportData.put("data_dnsu_ryo_1", calcDnsuRyo);
								break;
								
							case 2:
								// データ伝送量（前々月）
								transportData.put("data_dnsu_ryo_2", calcDnsuRyo);
								break;
								
							default:
								break;
						}
					}
				}
				
				// ISP認証ID
				transportData.put("isp_ninsho_id", ispNinshoId);
				
				// 最大データ伝送量
				transportData.put("max_data_dnsu_ryo", max_data_dnsu_ryo);
				
				// *** 下りデータの作成 ***
				if (!ccMsg.containsKey("transportDataList"))
				{
					ccMsg.put("transportDataList", new ArrayList<HashMap<String, Object>>());
				}
				((ArrayList<HashMap<String, Object>>)ccMsg.get("transportDataList")).add(transportData);
			}
		}
		
		// *** 下りデータの作成 ***
		// 通信状況出力フラグ
		ccMsg.put("communication_flg", ((0 < ispNinshoIdList.size()) ? "1" : "0"));
		
		work02Map.put("return_result_cd", "000000");
		
		return param;
	}
	
	/**
	 * 利用通信量合計[KB]と補正通信量合計[KB]を加算した値[GB]を返却する。
	 * 
	 * @param useTsryoGk 利用通信量合計[KB]
	 * @param hoseiTsryoGk 補正通信量合計[KB]
	 * @return 合計値[GB]
	 */
	private String calculateDnsuRyo(String useTsryoGk, String hoseiTsryoGk)
	{
		BigDecimal use = new BigDecimal("0.000");
		BigDecimal hosei = new BigDecimal("0.000");
		BigDecimal gk = new BigDecimal("0.000");
		
		if (useTsryoGk != null && !"".equals(useTsryoGk))
		{
			use = new BigDecimal(useTsryoGk);
			
			if (hoseiTsryoGk != null && !"".equals(hoseiTsryoGk))
			{
				// 利用通信量合計[KB] + 補正通信量合計[KB](マイナス有)
				hosei = new BigDecimal(hoseiTsryoGk);
				gk = use.add(hosei);
			}
			else
			{
				gk = use;
			}
			
			// [KB]→[GB]変換(小数点以下３桁まで(小数第４位を切り上げ))
			BigDecimal calc = gk.divide(new BigDecimal(KB_TO_CONV_GB), 3, BigDecimal.ROUND_UP);
			return String.valueOf(calc.doubleValue());
		}
		else
		{
			return "";
		}
	}
	
	/**
	 * チェック処理用のサービスインターフェイスを作成し、返却する。
	 * 
	 * @param handle セッションハンドル
	 * @param param リクエストパラメータ
	 * @param fixedText サービスメッセージ
	 * @return HashMap<String, Object> サービスインターフェイスマップ
	 * @throws RequestParameterException
	 */
	@SuppressWarnings("unchecked")
	public HashMap<String, Object> getInvokeCBS(SessionHandle handle, 
												IRequestParameterReadWrite param, 
												String fixedText) throws RequestParameterException
	{
		
		HashMap<String, Object> ccMsg = (HashMap<String, Object>)param.getData(SERVICE_ID);
		
		// *** EKK0161B004(サービス契約内訳一覧照会) の実行 ***
		Object[][] ekk0161b004IN = {
			// CC.ＫＥＹ＿サービス契約番号 => ＫＥＹ＿サービス契約番号
			{EKK0161B004CBSMsg.KEY_SVC_KEI_NO, (String)ccMsg.get("key_svc_kei_no")},
			// "" => ＫＥＹ＿料金コースコード
			{EKK0161B004CBSMsg.KEY_PCRS_CD, ""},
			// "" => ＫＥＹ＿サービス期間中フラグ
			{EKK0161B004CBSMsg.KEY_SVC_PRD_FLG, ""}
		};
		
		// 上りマッピングの処理結果を取得するためのマップを生成
		HashMap<String, Object> paramMap = editInMsg(param, TEMPLATE_ID_EKK0161B004, "1", ekk0161b004IN);
		
		return paramMap;
	}
	
	/**
	 * SC呼出
	 * @param handle
	 * @param param
	 * @param scCall
	 * @param svcIfId
	 * @param funcCode
	 * @param svcIfMapData
	 * @return CAANMsg
	 * @throws Exception
	 */
	@SuppressWarnings("unchecked")
	private CAANMsg callSC(SessionHandle handle, 
							IRequestParameterReadWrite param, 
							ServiceComponentRequestInvoker scCall,
							String svcIfId,
							String funcCode,
							Object[][] svcIfMapData) throws Exception
	{
		
		// ◇ 上りマッピング処理
		// 上りマッピングの処理結果を取得するためのマップを生成
		HashMap<String, Object> paramMap = editInMsg(param, svcIfId, funcCode, svcIfMapData);
		
		// SC呼出実行
		Map<?, ?> result = scCall.run(paramMap, handle);
		
		// 処理結果の判定
		CAANMsg[] templates = (CAANMsg[])result.get(JCMConstants.TEMPLATE_LIST_KEY);
		
		// 取得したリターンコード、ステータスの内容を見て異常かどうかの判断をする。
		String rtnCode = result.get(JCMConstants.RET_CD_INT_KEY).toString();
		
		// エラー情報の転記
		editErrorInfo(param, templates, (Integer)result.get(JCMConstants.RET_CD_INT_KEY));
		
		//エラー情報のマップを取得
		ArrayList<Object> errList = (ArrayList<Object>)param.getControlMapData(SCControlMapKeys.ERROR_INFO);
		if (errList == null)
		{
			errList = new ArrayList<Object>();
		}
		// コントロールマップに設定
		param.setControlMapData(SCControlMapKeys.ERROR_INFO, TemplateErrorUtil.getErrorInfo(result, errList));
		
		int status = templates[0].getInt(STATUS);
		// 異常の場合、SCCallExceptionを生成してスローする
		if(!("0".equals(rtnCode) && 0 == status))
		{
			throw new SCCallException("戻り値不正", rtnCode, status);
		}
		
		return ((CAANMsg[])result.get(JCMConstants.TEMPLATE_LIST_KEY))[0];
	}
	
	/**
	 * 上りマッピング処理
	 * 
	 * @param param
	 * @param svcIfId
	 * @param funcCode
	 * @param mapData
	 * @return HashMap<String, Object>
	 * @throws RequestParameterException
	 */
	private HashMap<String, Object> editInMsg(IRequestParameterReadWrite param, 
												String svcIfId, 
												String funcCode,
												Object[][] mapData) throws RequestParameterException
	{
		HashMap<String, Object> paramMap = new HashMap<String, Object>();
		
		// 【取得元：電文ヘッダ(ヘッダ)】
		// 電文ID
		paramMap.put(JCMConstants.TRANZACTION_ID_KEY, param.getTelegramID());
		// ユースケースID
		paramMap.put(JCMConstants.USECASE_ID_KEY, param.getUsecaseID());
		// オペレーションID
		paramMap.put(JCMConstants.OPERATION_ID_KEY, param.getOperationID());
		// サービス呼び出し区分
		paramMap.put(JCMConstants.CALL_TYPE_KEY, param.getCallType());

		// 【取得元：ユーザエリア(コントロールマップ)】
		// 依頼先ホスト名
		paramMap.put(JCMConstants.CLIENT_HOST_NAME_KEY, param.getControlMapData(SCControlMapKeys.REQ_HOSTNAME));
		// 依頼元IPアドレス
		paramMap.put(JCMConstants.CLIENT_IP_ADDRESS_KEY, param.getControlMapData(SCControlMapKeys.REQ_HOSTIP));
		// 依頼元画面ID
		paramMap.put(JCMConstants.INVOKE_GAMEN_ID_KEY, param.getControlMapData(SCControlMapKeys.REQ_VIEWID));
		// オペレータID
		paramMap.put(JCMConstants.OPERATOR_ID_KEY, param.getControlMapData(SCControlMapKeys.OPERATOR_ID));

		CAANMsg template = new CAANMsg("eo.ejb.cbs.cbsmsg." + svcIfId + "CBSMsg");

		// テンプレートID
		template.set(TEMPLATEID, svcIfId);

		template.set(FUNC_CODE, funcCode);

		// オペレータID
		Object operatorId = param.getControlMapData(SCControlMapKeys.OPERATOR_ID);
		template.set(JCMConstants.OPERATOR_ID_KEY, operatorId);

		// 運用日付
		Object operateDate = param.getControlMapData(SCControlMapKeys.OPE_DATE);
		template.set(JCMConstants.OPERATE_DATE_KEY, operateDate);

		// 運用日時
		Object operateDateTime = param.getControlMapData(SCControlMapKeys.OPE_TIME);
		template.set(JCMConstants.OPERATE_DATETIME_KEY, operateDateTime);
		
		for (int i = 0; i < mapData.length; i++)
		{
			Object[] mapVal = mapData[i];
			if ("".equals(mapVal[1]))
			{
				template.setNull((String)mapVal[0]);
			}
			else
			{
				template.set((String)mapVal[0], (String)mapVal[1]);
			}
		}
		
		CAANMsg[] templates = new CAANMsg[1];
		templates[0] = template;
		paramMap.put(JCMConstants.TEMPLATE_LIST_KEY, templates);
		
		return paramMap;
	}
	
	/**
	 * サービスインターフェイスのエラー情報をマッピングする。
	 * <br>
	 * @param param
	 * @param templates
	 * @param returnCode
	 * @return リクエストパラメータ
	 * @throws RequestParameterException
	 */
	@SuppressWarnings("unchecked")
	public IRequestParameterReadWrite editErrorInfo(IRequestParameterReadWrite param, 
													CAANMsg[] templates,
													int returnCode) throws RequestParameterException
	{
		CAANMsg template = templates[0];
		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<String, Object> inMap = (HashMap<String, Object>)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;
	}
}
