View Javadoc
1   /**
2    * Copyright 2004-2014 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  package org.kuali.kpme.tklm.leave.payout.validation;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.joda.time.LocalDate;
20  import org.kuali.kpme.core.api.accrualcategory.AccrualCategory;
21  import org.kuali.kpme.core.api.accrualcategory.rule.AccrualCategoryRule;
22  import org.kuali.kpme.core.api.earncode.EarnCode;
23  import org.kuali.kpme.core.api.principal.PrincipalHRAttributes;
24  import org.kuali.kpme.core.service.HrServiceLocator;
25  import org.kuali.kpme.core.util.HrConstants;
26  import org.kuali.kpme.core.util.ValidationUtils;
27  import org.kuali.kpme.tklm.api.leave.override.EmployeeOverrideContract;
28  import org.kuali.kpme.tklm.api.leave.summary.LeaveSummaryContract;
29  import org.kuali.kpme.tklm.api.leave.summary.LeaveSummaryRowContract;
30  import org.kuali.kpme.tklm.leave.payout.LeavePayout;
31  import org.kuali.kpme.tklm.leave.service.LmServiceLocator;
32  import org.kuali.rice.krad.util.GlobalVariables;
33  import org.kuali.rice.krad.util.ObjectUtils;
34  
35  import java.math.BigDecimal;
36  
37  public class LeavePayoutValidationUtils {
38  
39  	public static boolean validatePayout(LeavePayout leavePayout) {
40  		boolean isValid = true;
41  		String principalId = leavePayout.getPrincipalId();
42  		LocalDate effectiveDate = leavePayout.getEffectiveLocalDate();
43  		String fromAccrualCategory = leavePayout.getFromAccrualCategory();
44  		String payoutEarnCode = leavePayout.getEarnCode();
45  		
46  		if(!ValidationUtils.validateAccrualCategory(fromAccrualCategory, effectiveDate)) {
47  			GlobalVariables.getMessageMap().putError("leavePayout.fromAccrualCategory", "leavePayout.fromAccrualCategory.exists");
48  			isValid &= false;
49  		}
50  		
51  		if(!ValidationUtils.validateEarnCode(payoutEarnCode, effectiveDate)) {
52  			GlobalVariables.getMessageMap().putError("leavePayout.earnCode", "leavePayout.earncode.exists");
53  			isValid &= false;
54  		}
55  		
56  		if(!ValidationUtils.validatePrincipalId(principalId)) {
57  			GlobalVariables.getMessageMap().putError("leavePayout.principalId", "leavePayout.principal.exists");
58  			isValid &= false;
59  		}
60  		
61  		if(effectiveDate.isAfter(LocalDate.now().plusYears(1))) {
62  			GlobalVariables.getMessageMap().putError("leavePayout.effectiveDate", "leavePayout.effectiveDate.overOneYear");
63  			isValid &= false;
64  		}
65  		
66  		
67  		
68  		if(isValid) {
69  			
70  			AccrualCategory fromCat = HrServiceLocator.getAccrualCategoryService().getAccrualCategory(fromAccrualCategory, effectiveDate);
71  			EarnCode earnCode = HrServiceLocator.getEarnCodeService().getEarnCode(payoutEarnCode, effectiveDate);
72  			PrincipalHRAttributes pha = HrServiceLocator.getPrincipalHRAttributeService().getPrincipalCalendar(principalId,effectiveDate);
73  			
74  			if(ObjectUtils.isNotNull(pha)) {
75  				if(ObjectUtils.isNotNull(pha.getLeavePlan())) {
76  					AccrualCategoryRule acr = HrServiceLocator.getAccrualCategoryRuleService().getAccrualCategoryRuleForDate(fromCat, effectiveDate, pha.getServiceLocalDate());
77  
78  					if(ObjectUtils.isNotNull(acr)) {
79  						if(ObjectUtils.isNotNull(acr.getMaxBalFlag())
80  								&& StringUtils.isNotBlank(acr.getMaxBalFlag())
81  								&& StringUtils.isNotEmpty(acr.getMaxBalFlag())
82  								&& StringUtils.equals(acr.getMaxBalFlag(), "Y")) {
83  							if(ObjectUtils.isNotNull(acr.getMaxPayoutEarnCode()) || StringUtils.equals(HrConstants.ACTION_AT_MAX_BALANCE.LOSE, acr.getActionAtMaxBalance())) {
84  								isValid = validateForfeitedAmount(leavePayout.getForfeitedAmount());
85  								isValid &= validatePayoutAmount(leavePayout.getPayoutAmount(),fromCat, earnCode, principalId, effectiveDate, acr);
86  							}
87  							else {
88  								//should never be the case if accrual category rules are validated correctly.
89  								GlobalVariables.getMessageMap().putError("leavePayout.fromAccrualCategory",
90  										"leavePayout.fromAccrualCategory.rules.payoutToEarnCode",
91  										fromAccrualCategory);
92  								isValid &= false;
93  							}
94  						}
95  						else {
96  							//max bal flag null, blank, empty, or "N"
97  							GlobalVariables.getMessageMap().putError("leavePayout.fromAccrualCategory",
98  									"leavePayout.fromAccrualCategory.rules.maxBalFlag", fromAccrualCategory);
99  							isValid &= false;
100 						}
101 					}
102 					else {
103 						//department admins must validate amount to transfer does not exceed current balance.
104 						GlobalVariables.getMessageMap().putError("leavePayout.fromAccrualCategory",
105 								"leavePayout.fromAccrualCategory.rules.exist",fromCat.getAccrualCategory());
106 						isValid &= false;
107 					}
108 				}
109 				else {
110 					//if the principal doesn't have a leave plan, there aren't any accrual categories that can be debited/credited.
111 					GlobalVariables.getMessageMap().putError("leavePayout.principalId","leavePayout.principal.noLeavePlan");
112 					isValid &=false;
113 				}
114 			}
115 			else  {
116 				//if the principal has no principal hr attributes, they're not a principal.
117 				GlobalVariables.getMessageMap().putError("leavePayout.principalId","leavePayout.principal.noAttributes");
118 				isValid &= false;
119 			}
120 		}
121 		return isValid;
122 
123 	}
124 
125 	private static boolean validateForfeitedAmount(BigDecimal forfeitedAmount) {
126 		if(forfeitedAmount.compareTo(BigDecimal.ZERO) < 0) {
127 			GlobalVariables.getMessageMap().putError("leavePayout.forfeitedAmount", "balanceTransfer.transferAmount.negative");
128 			return false;
129 		}
130 		return true;
131 	}
132 
133 	private static boolean validatePayoutAmount(BigDecimal payoutAmount,
134 			AccrualCategory fromCat, EarnCode earnCode, String principalId,
135 			LocalDate effectiveDate, AccrualCategoryRule accrualRule) {
136 
137 		LeaveSummaryContract leaveSummary = LmServiceLocator.getLeaveSummaryService().getLeaveSummaryAsOfDateForAccrualCategory(principalId, effectiveDate, fromCat.getAccrualCategory());
138 		LeaveSummaryRowContract row = leaveSummary.getLeaveSummaryRowForAccrualCtgy(fromCat.getAccrualCategory());
139 		BigDecimal balance = row.getAccruedBalance();
140 		//transfer amount must be less than the max transfer amount defined in the accrual category rule.
141 		//it cannot be negative.
142 		boolean isValid = true;
143 		
144 		BigDecimal maxPayoutAmount = null;
145 		BigDecimal adjustedMaxPayoutAmount = null;
146 		if(ObjectUtils.isNotNull(accrualRule.getMaxPayoutAmount())) {
147 			maxPayoutAmount = new BigDecimal(accrualRule.getMaxPayoutAmount());
148 			BigDecimal fullTimeEngagement = HrServiceLocator.getJobService().getFteSumForAllActiveLeaveEligibleJobs(principalId, effectiveDate);
149 			adjustedMaxPayoutAmount = maxPayoutAmount.multiply(fullTimeEngagement);
150 		}
151 		
152 		//use override if one exists.
153         EmployeeOverrideContract maxPayoutAmountOverride = LmServiceLocator.getEmployeeOverrideService().getEmployeeOverride(principalId, fromCat.getLeavePlan(), fromCat.getAccrualCategory(), "MPA", effectiveDate);
154 		if(ObjectUtils.isNotNull(maxPayoutAmountOverride))
155 			adjustedMaxPayoutAmount = new BigDecimal(maxPayoutAmountOverride.getOverrideValue());
156 				
157 				
158 		if(ObjectUtils.isNotNull(adjustedMaxPayoutAmount)) {
159 			if(payoutAmount.compareTo(adjustedMaxPayoutAmount) > 0) {
160 				isValid &= false;
161 				String fromUnitOfTime = HrConstants.UNIT_OF_TIME.get(fromCat.getUnitOfTime());
162 				GlobalVariables.getMessageMap().putError("leavePayout.payoutAmount","leavePayout.payoutAmount.maxPayoutAmount",adjustedMaxPayoutAmount.toString(),fromUnitOfTime);
163 			}
164 		}
165 		// check for a positive amount.
166 		if(payoutAmount.compareTo(BigDecimal.ZERO) < 0 ) {
167 			isValid &= false;
168 			GlobalVariables.getMessageMap().putError("leavePayout.payoutAmount","leavePayout.payoutAmount.negative");
169 		}
170 		
171 		if(payoutAmount.compareTo(balance) > 0 ) {
172 			isValid &= false;
173 			GlobalVariables.getMessageMap().putError("leavePayout.payoutAmount", "maxBalance.amount.exceedsBalance", balance.toString());
174 		}	
175 		return isValid;
176 	}
177 
178 	private boolean validateMaxBalance() {
179 		//This validation could assert that the payout amount, together with forfeiture
180 		//brings the balance for the given accrual category back to, or under, the balance limit
181 		//without exceeding the total accrued balance.
182 		return true;
183 	}
184 	
185 	private boolean validateMaxCarryOver() {
186 		//This validation could assert that the payout amount, together with forfeiture
187 		//brings the balance for the given accrual category back to, or under, the max carry over limit
188 		//without exceeding the total accrued balance.
189 		return true;
190 	}
191 	
192 }