001/*
002 * Copyright 2008 The Kuali Foundation
003 * 
004 * Licensed under the Educational Community License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 * 
008 * http://www.opensource.org/licenses/ecl2.php
009 * 
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 * See the License for the specific language governing permissions and
014 * limitations under the License.
015 */
016package org.kuali.ole.fp.document.validation.impl;
017
018import static org.kuali.ole.sys.OLEKeyConstants.ERROR_DOCUMENT_FUND_GROUP_SET_DOES_NOT_BALANCE;
019
020import java.util.ArrayList;
021import java.util.Iterator;
022import java.util.List;
023
024import org.kuali.ole.fp.document.TransferOfFundsDocument;
025import org.kuali.ole.sys.businessobject.AccountingLine;
026import org.kuali.ole.sys.context.SpringContext;
027import org.kuali.ole.sys.document.AccountingDocument;
028import org.kuali.ole.sys.document.validation.GenericValidation;
029import org.kuali.ole.sys.document.validation.event.AttributedDocumentEvent;
030import org.kuali.ole.sys.document.validation.impl.AccountingDocumentRuleBaseConstants;
031import org.kuali.rice.core.api.parameter.ParameterEvaluator;
032import org.kuali.rice.core.api.parameter.ParameterEvaluatorService;
033import org.kuali.rice.core.api.util.type.KualiDecimal;
034import org.kuali.rice.coreservice.framework.parameter.ParameterService;
035import org.kuali.rice.krad.util.GlobalVariables;
036
037/**
038 * Validation for Transfer of Funds document that tests if the fund groups represented by a given document are in balance.
039 */
040public class TransferOfFundsFundGroupsBalancedValidation extends GenericValidation {
041    private AccountingDocument accountingDocumentForValidation;
042    private ParameterService parameterService;
043
044    /**
045     * This is a helper method that wraps the fund group balancing check. This check can be configured by updating the 
046     * application parameter table that is associated with this check. See the document's specification for details.
047     * @see org.kuali.ole.sys.document.validation.Validation#validate(org.kuali.ole.sys.document.validation.event.AttributedDocumentEvent)
048     */
049    public boolean validate(AttributedDocumentEvent event) {
050        return isFundGroupSetBalanceValid(accountingDocumentForValidation, TransferOfFundsDocument.class, AccountingDocumentRuleBaseConstants.APPLICATION_PARAMETER.FUND_GROUP_BALANCING_SET);
051    }
052    
053    /**
054     * This method will make sure that totals for a specified set of fund groups is valid across the two different accounting line
055     * sections.
056     * 
057     * @param tranDoc
058     * @param fundGroupCodes An array of the fund group codes that will be considered for balancing.
059     * @return True if they balance; false otherwise.
060     */
061    protected boolean isFundGroupSetBalanceValid(AccountingDocument tranDoc, Class componentClass, String parameterName) {
062        // don't need to do any of this if there's no parameter
063        if (!getParameterService().parameterExists(componentClass, parameterName)) {
064            return true;
065        }
066
067        List lines = new ArrayList();
068
069        lines.addAll(tranDoc.getSourceAccountingLines());
070        lines.addAll(tranDoc.getTargetAccountingLines());
071
072        KualiDecimal sourceLinesTotal = KualiDecimal.ZERO;
073        KualiDecimal targetLinesTotal = KualiDecimal.ZERO;
074
075        // iterate over each accounting line and if it has an account with a
076        // fund group that should be balanced, then add that lines amount to the bucket
077        for (Iterator i = lines.iterator(); i.hasNext();) {
078            AccountingLine line = (AccountingLine) i.next();
079            String fundGroupCode = line.getAccount().getSubFundGroup().getFundGroupCode();
080
081            ParameterEvaluator evaluator = /*REFACTORME*/SpringContext.getBean(ParameterEvaluatorService.class).getParameterEvaluator(componentClass, parameterName, fundGroupCode);
082            if (evaluator.evaluationSucceeds()) {
083                KualiDecimal glpeLineAmount = tranDoc.getGeneralLedgerPendingEntryAmountForDetail(line);
084                if (line.isSourceAccountingLine()) {
085                    sourceLinesTotal = sourceLinesTotal.add(glpeLineAmount);
086                }
087                else {
088                    targetLinesTotal = targetLinesTotal.add(glpeLineAmount);
089                }
090            }
091        }
092
093        // check that the amounts balance across sections
094        boolean isValid = true;
095
096        if (sourceLinesTotal.compareTo(targetLinesTotal) != 0) {
097            isValid = false;
098
099            // creating an evaluator to just format the fund codes into a nice string
100            ParameterEvaluator evaluator = /*REFACTORME*/SpringContext.getBean(ParameterEvaluatorService.class).getParameterEvaluator(componentClass, parameterName, "");
101            GlobalVariables.getMessageMap().putError("document.sourceAccountingLines", ERROR_DOCUMENT_FUND_GROUP_SET_DOES_NOT_BALANCE, new String[] { tranDoc.getSourceAccountingLinesSectionTitle(), tranDoc.getTargetAccountingLinesSectionTitle(), evaluator.getParameterValuesForMessage() });
102        }
103
104        return isValid;
105    }
106
107    /**
108     * Gets the accountingDocumentForValidation attribute. 
109     * @return Returns the accountingDocumentForValidation.
110     */
111    public AccountingDocument getAccountingDocumentForValidation() {
112        return accountingDocumentForValidation;
113    }
114
115    /**
116     * Sets the accountingDocumentForValidation attribute value.
117     * @param accountingDocumentForValidation The accountingDocumentForValidation to set.
118     */
119    public void setAccountingDocumentForValidation(AccountingDocument accountingDocumentForValidation) {
120        this.accountingDocumentForValidation = accountingDocumentForValidation;
121    }
122
123    /**
124     * Gets the parameterService attribute. 
125     * @return Returns the parameterService.
126     */
127    public ParameterService getParameterService() {
128        return parameterService;
129    }
130
131    /**
132     * Sets the parameterService attribute value.
133     * @param parameterService The parameterService to set.
134     */
135    public void setParameterService(ParameterService parameterService) {
136        this.parameterService = parameterService;
137    }
138}